mirror of https://github.com/containers/podman.git
217 lines
8.0 KiB
Go
217 lines
8.0 KiB
Go
package tunnel
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/containers/image/v5/types"
|
|
"github.com/containers/podman/v5/pkg/bindings/images"
|
|
"github.com/containers/podman/v5/pkg/bindings/manifests"
|
|
"github.com/containers/podman/v5/pkg/domain/entities"
|
|
envLib "github.com/containers/podman/v5/pkg/env"
|
|
"golang.org/x/exp/slices"
|
|
)
|
|
|
|
// ManifestCreate implements manifest create via ImageEngine
|
|
func (ir *ImageEngine) ManifestCreate(ctx context.Context, name string, images []string, opts entities.ManifestCreateOptions) (string, error) {
|
|
options := new(manifests.CreateOptions).WithAll(opts.All).WithAmend(opts.Amend).WithAnnotation(opts.Annotations)
|
|
imageID, err := manifests.Create(ir.ClientCtx, name, images, options)
|
|
if err != nil {
|
|
return imageID, fmt.Errorf("creating manifest: %w", err)
|
|
}
|
|
return imageID, err
|
|
}
|
|
|
|
// ManifestExists checks if a manifest list with the given name exists
|
|
func (ir *ImageEngine) ManifestExists(ctx context.Context, name string) (*entities.BoolReport, error) {
|
|
exists, err := manifests.Exists(ir.ClientCtx, name, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &entities.BoolReport{Value: exists}, nil
|
|
}
|
|
|
|
// ManifestInspect returns contents of manifest list with given name
|
|
func (ir *ImageEngine) ManifestInspect(ctx context.Context, name string, opts entities.ManifestInspectOptions) ([]byte, error) {
|
|
options := new(manifests.InspectOptions).WithAuthfile(opts.Authfile)
|
|
if s := opts.SkipTLSVerify; s != types.OptionalBoolUndefined {
|
|
if s == types.OptionalBoolTrue {
|
|
options.WithSkipTLSVerify(true)
|
|
} else {
|
|
options.WithSkipTLSVerify(false)
|
|
}
|
|
}
|
|
|
|
list, err := manifests.InspectListData(ir.ClientCtx, name, options)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("getting content of manifest list or image %s: %w", name, err)
|
|
}
|
|
|
|
buf, err := json.MarshalIndent(list, "", " ")
|
|
if err != nil {
|
|
return buf, fmt.Errorf("rendering manifest for display: %w", err)
|
|
}
|
|
return buf, err
|
|
}
|
|
|
|
// ManifestAdd adds images to the manifest list
|
|
func (ir *ImageEngine) ManifestAdd(_ context.Context, name string, imageNames []string, opts entities.ManifestAddOptions) (string, error) {
|
|
imageNames = slices.Clone(imageNames)
|
|
for _, image := range opts.Images {
|
|
if !slices.Contains(imageNames, image) {
|
|
imageNames = append(imageNames, image)
|
|
}
|
|
}
|
|
|
|
options := new(manifests.AddOptions).WithAll(opts.All).WithArch(opts.Arch).WithVariant(opts.Variant)
|
|
options.WithFeatures(opts.Features).WithImages(imageNames).WithOS(opts.OS).WithOSVersion(opts.OSVersion)
|
|
options.WithUsername(opts.Username).WithPassword(opts.Password).WithAuthfile(opts.Authfile)
|
|
|
|
if len(opts.Annotation) != 0 {
|
|
annotations := make(map[string]string)
|
|
for _, annotationSpec := range opts.Annotation {
|
|
key, val, hasVal := strings.Cut(annotationSpec, "=")
|
|
if !hasVal {
|
|
return "", fmt.Errorf("no value given for annotation %q", key)
|
|
}
|
|
annotations[key] = val
|
|
}
|
|
opts.Annotations = envLib.Join(opts.Annotations, annotations)
|
|
}
|
|
options.WithAnnotation(opts.Annotations)
|
|
|
|
if s := opts.SkipTLSVerify; s != types.OptionalBoolUndefined {
|
|
if s == types.OptionalBoolTrue {
|
|
options.WithSkipTLSVerify(true)
|
|
} else {
|
|
options.WithSkipTLSVerify(false)
|
|
}
|
|
}
|
|
|
|
id, err := manifests.Add(ir.ClientCtx, name, options)
|
|
if err != nil {
|
|
return id, fmt.Errorf("adding to manifest list %s: %w", name, err)
|
|
}
|
|
return id, nil
|
|
}
|
|
|
|
// ManifestAddArtifact creates artifact manifests and adds them to the manifest list
|
|
func (ir *ImageEngine) ManifestAddArtifact(_ context.Context, name string, files []string, opts entities.ManifestAddArtifactOptions) (string, error) {
|
|
files = slices.Clone(files)
|
|
for _, file := range opts.Files {
|
|
if !slices.Contains(files, file) {
|
|
files = append(files, file)
|
|
}
|
|
}
|
|
options := new(manifests.AddArtifactOptions).WithArch(opts.Arch).WithVariant(opts.Variant)
|
|
options.WithFeatures(opts.Features).WithOS(opts.OS).WithOSVersion(opts.OSVersion).WithOSFeatures(opts.OSFeatures)
|
|
if len(opts.Annotation) != 0 {
|
|
annotations := make(map[string]string)
|
|
for _, annotationSpec := range opts.Annotation {
|
|
key, val, hasVal := strings.Cut(annotationSpec, "=")
|
|
if !hasVal {
|
|
return "", fmt.Errorf("no value given for annotation %q", key)
|
|
}
|
|
annotations[key] = val
|
|
}
|
|
options.WithAnnotation(annotations)
|
|
}
|
|
options.WithType(opts.Type).WithConfigType(opts.ConfigType).WithLayerType(opts.LayerType)
|
|
options.WithConfig(opts.Config)
|
|
options.WithExcludeTitles(opts.ExcludeTitles).WithSubject(opts.Subject)
|
|
options.WithAnnotations(opts.Annotations)
|
|
options.WithFiles(files)
|
|
id, err := manifests.AddArtifact(ir.ClientCtx, name, options)
|
|
if err != nil {
|
|
return id, fmt.Errorf("adding to manifest list %s: %w", name, err)
|
|
}
|
|
return id, nil
|
|
}
|
|
|
|
// ManifestAnnotate updates an entry of the manifest list
|
|
func (ir *ImageEngine) ManifestAnnotate(ctx context.Context, name, images string, opts entities.ManifestAnnotateOptions) (string, error) {
|
|
options := new(manifests.ModifyOptions).WithArch(opts.Arch).WithVariant(opts.Variant)
|
|
options.WithFeatures(opts.Features).WithOS(opts.OS).WithOSVersion(opts.OSVersion)
|
|
|
|
if len(opts.Annotation) != 0 {
|
|
annotations := make(map[string]string)
|
|
for _, annotationSpec := range opts.Annotation {
|
|
key, val, hasVal := strings.Cut(annotationSpec, "=")
|
|
if !hasVal {
|
|
return "", fmt.Errorf("no value given for annotation %q", key)
|
|
}
|
|
annotations[key] = val
|
|
}
|
|
opts.Annotations = envLib.Join(opts.Annotations, annotations)
|
|
}
|
|
options.WithAnnotations(opts.Annotations)
|
|
|
|
id, err := manifests.Annotate(ir.ClientCtx, name, []string{images}, options)
|
|
if err != nil {
|
|
return id, fmt.Errorf("annotating to manifest list %s: %w", name, err)
|
|
}
|
|
return id, nil
|
|
}
|
|
|
|
// ManifestRemoveDigest removes the digest from manifest list
|
|
func (ir *ImageEngine) ManifestRemoveDigest(ctx context.Context, name string, image string) (string, error) {
|
|
updatedListID, err := manifests.Remove(ir.ClientCtx, name, image, nil)
|
|
if err != nil {
|
|
return updatedListID, fmt.Errorf("removing from manifest %s: %w", name, err)
|
|
}
|
|
return fmt.Sprintf("%s :%s\n", updatedListID, image), nil
|
|
}
|
|
|
|
// ManifestRm removes the specified manifest list from storage
|
|
func (ir *ImageEngine) ManifestRm(ctx context.Context, names []string) (*entities.ImageRemoveReport, []error) {
|
|
return ir.Remove(ctx, names, entities.ImageRemoveOptions{LookupManifest: true})
|
|
}
|
|
|
|
// ManifestPush pushes a manifest list or image index to the destination
|
|
func (ir *ImageEngine) ManifestPush(ctx context.Context, name, destination string, opts entities.ImagePushOptions) (string, error) {
|
|
if opts.Signers != nil {
|
|
return "", fmt.Errorf("forwarding Signers is not supported for remote clients")
|
|
}
|
|
|
|
options := new(images.PushOptions)
|
|
options.WithUsername(opts.Username).WithPassword(opts.Password).WithAuthfile(opts.Authfile).WithRemoveSignatures(opts.RemoveSignatures).WithAll(opts.All).WithFormat(opts.Format).WithCompressionFormat(opts.CompressionFormat).WithQuiet(opts.Quiet).WithProgressWriter(opts.Writer).WithAddCompression(opts.AddCompression).WithForceCompressionFormat(opts.ForceCompressionFormat)
|
|
|
|
if s := opts.SkipTLSVerify; s != types.OptionalBoolUndefined {
|
|
if s == types.OptionalBoolTrue {
|
|
options.WithSkipTLSVerify(true)
|
|
} else {
|
|
options.WithSkipTLSVerify(false)
|
|
}
|
|
}
|
|
digest, err := manifests.Push(ir.ClientCtx, name, destination, options)
|
|
if err != nil {
|
|
return "", fmt.Errorf("pushing manifest list %s: %w", name, err)
|
|
}
|
|
|
|
if opts.Rm {
|
|
if _, rmErrors := ir.Remove(ctx, []string{name}, entities.ImageRemoveOptions{LookupManifest: true}); len(rmErrors) > 0 {
|
|
return "", fmt.Errorf("removing manifest after push: %w", rmErrors[0])
|
|
}
|
|
}
|
|
|
|
return digest, err
|
|
}
|
|
|
|
// ManifestListClear clears out all instances from a manifest list
|
|
func (ir *ImageEngine) ManifestListClear(ctx context.Context, name string) (string, error) {
|
|
listContents, err := manifests.InspectListData(ctx, name, &manifests.InspectOptions{})
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
for _, instance := range listContents.Manifests {
|
|
if _, err := manifests.Remove(ctx, name, instance.Digest.String(), &manifests.RemoveOptions{}); err != nil {
|
|
return "", err
|
|
}
|
|
}
|
|
|
|
return name, nil
|
|
}
|