mirror of
				https://github.com/containers/podman.git
				synced 2025-11-04 08:56:05 +08:00 
			
		
		
		
	Use "slices" from the standard library, this package was added in go 1.21 so we can use it now. Signed-off-by: Paul Holzinger <pholzing@redhat.com>
		
			
				
	
	
		
			477 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			477 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package manifests
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"encoding/json"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"mime/multipart"
 | 
						|
	"net/http"
 | 
						|
	"net/textproto"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"slices"
 | 
						|
	"strconv"
 | 
						|
	"strings"
 | 
						|
	"sync"
 | 
						|
 | 
						|
	"github.com/containers/common/libimage/define"
 | 
						|
	"github.com/containers/image/v5/manifest"
 | 
						|
	imageTypes "github.com/containers/image/v5/types"
 | 
						|
	"github.com/containers/podman/v5/pkg/auth"
 | 
						|
	"github.com/containers/podman/v5/pkg/bindings"
 | 
						|
	"github.com/containers/podman/v5/pkg/bindings/images"
 | 
						|
	entitiesTypes "github.com/containers/podman/v5/pkg/domain/entities/types"
 | 
						|
	"github.com/containers/podman/v5/pkg/errorhandling"
 | 
						|
	dockerAPI "github.com/docker/docker/api/types"
 | 
						|
	jsoniter "github.com/json-iterator/go"
 | 
						|
)
 | 
						|
 | 
						|
// Create creates a manifest for the given name.  Optional images to be associated with
 | 
						|
// the new manifest can also be specified.  The all boolean specifies to add all entries
 | 
						|
// of a list if the name provided is a manifest list.  The ID of the new manifest list
 | 
						|
// is returned as a string.
 | 
						|
func Create(ctx context.Context, name string, images []string, options *CreateOptions) (string, error) {
 | 
						|
	var idr dockerAPI.IDResponse
 | 
						|
	if options == nil {
 | 
						|
		options = new(CreateOptions)
 | 
						|
	}
 | 
						|
	conn, err := bindings.GetClient(ctx)
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	if len(name) < 1 {
 | 
						|
		return "", errors.New("creating a manifest requires at least one name argument")
 | 
						|
	}
 | 
						|
	params, err := options.ToParams()
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
 | 
						|
	for _, i := range images {
 | 
						|
		params.Add("images", i)
 | 
						|
	}
 | 
						|
 | 
						|
	response, err := conn.DoRequest(ctx, nil, http.MethodPost, "/manifests/%s", params, nil, name)
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	defer response.Body.Close()
 | 
						|
 | 
						|
	return idr.ID, response.Process(&idr)
 | 
						|
}
 | 
						|
 | 
						|
// Exists returns true if a given manifest list exists
 | 
						|
func Exists(ctx context.Context, name string, options *ExistsOptions) (bool, error) {
 | 
						|
	conn, err := bindings.GetClient(ctx)
 | 
						|
	if err != nil {
 | 
						|
		return false, err
 | 
						|
	}
 | 
						|
	response, err := conn.DoRequest(ctx, nil, http.MethodGet, "/manifests/%s/exists", nil, nil, name)
 | 
						|
	if err != nil {
 | 
						|
		return false, err
 | 
						|
	}
 | 
						|
	defer response.Body.Close()
 | 
						|
 | 
						|
	return response.IsSuccess(), nil
 | 
						|
}
 | 
						|
 | 
						|
// Inspect returns a manifest list for a given name.
 | 
						|
func Inspect(ctx context.Context, name string, options *InspectOptions) (*manifest.Schema2List, error) {
 | 
						|
	conn, err := bindings.GetClient(ctx)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if options == nil {
 | 
						|
		options = new(InspectOptions)
 | 
						|
	}
 | 
						|
 | 
						|
	params, err := options.ToParams()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	// SkipTLSVerify is special.  We need to delete the param added by
 | 
						|
	// ToParams() and change the key and flip the bool
 | 
						|
	if options.SkipTLSVerify != nil {
 | 
						|
		params.Del("SkipTLSVerify")
 | 
						|
		params.Set("tlsVerify", strconv.FormatBool(!options.GetSkipTLSVerify()))
 | 
						|
	}
 | 
						|
 | 
						|
	header, err := auth.MakeXRegistryAuthHeader(&imageTypes.SystemContext{AuthFilePath: options.GetAuthfile()}, "", "")
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	response, err := conn.DoRequest(ctx, nil, http.MethodGet, "/manifests/%s/json", params, header, name)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	defer response.Body.Close()
 | 
						|
 | 
						|
	var list manifest.Schema2List
 | 
						|
	return &list, response.Process(&list)
 | 
						|
}
 | 
						|
 | 
						|
// InspectListData returns a manifest list for a given name.
 | 
						|
// Contains exclusive field like `annotations` which is only
 | 
						|
// present in OCI spec and not in docker image spec.
 | 
						|
func InspectListData(ctx context.Context, name string, options *InspectOptions) (*define.ManifestListData, error) {
 | 
						|
	conn, err := bindings.GetClient(ctx)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if options == nil {
 | 
						|
		options = new(InspectOptions)
 | 
						|
	}
 | 
						|
 | 
						|
	params, err := options.ToParams()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	// SkipTLSVerify is special.  We need to delete the param added by
 | 
						|
	// ToParams() and change the key and flip the bool
 | 
						|
	if options.SkipTLSVerify != nil {
 | 
						|
		params.Del("SkipTLSVerify")
 | 
						|
		params.Set("tlsVerify", strconv.FormatBool(!options.GetSkipTLSVerify()))
 | 
						|
	}
 | 
						|
 | 
						|
	header, err := auth.MakeXRegistryAuthHeader(&imageTypes.SystemContext{AuthFilePath: options.GetAuthfile()}, "", "")
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	response, err := conn.DoRequest(ctx, nil, http.MethodGet, "/manifests/%s/json", params, header, name)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	defer response.Body.Close()
 | 
						|
 | 
						|
	var list define.ManifestListData
 | 
						|
	return &list, response.Process(&list)
 | 
						|
}
 | 
						|
 | 
						|
// Add adds a manifest to a given manifest list.  Additional options for the manifest
 | 
						|
// can also be specified.  The ID of the new manifest list is returned as a string
 | 
						|
func Add(ctx context.Context, name string, options *AddOptions) (string, error) {
 | 
						|
	if options == nil {
 | 
						|
		options = new(AddOptions)
 | 
						|
	}
 | 
						|
 | 
						|
	optionsv4 := ModifyOptions{
 | 
						|
		All:           options.All,
 | 
						|
		Annotations:   options.Annotation,
 | 
						|
		Arch:          options.Arch,
 | 
						|
		Features:      options.Features,
 | 
						|
		Images:        options.Images,
 | 
						|
		OS:            options.OS,
 | 
						|
		OSFeatures:    options.OSFeatures,
 | 
						|
		OSVersion:     options.OSVersion,
 | 
						|
		Variant:       options.Variant,
 | 
						|
		Username:      options.Username,
 | 
						|
		Password:      options.Password,
 | 
						|
		Authfile:      options.Authfile,
 | 
						|
		SkipTLSVerify: options.SkipTLSVerify,
 | 
						|
	}
 | 
						|
	optionsv4.WithOperation("update")
 | 
						|
	return Modify(ctx, name, options.Images, &optionsv4)
 | 
						|
}
 | 
						|
 | 
						|
// AddArtifact creates an artifact manifest and adds it to a given manifest
 | 
						|
// list.  Additional options for the manifest can also be specified.  The ID of
 | 
						|
// the new manifest list is returned as a string
 | 
						|
func AddArtifact(ctx context.Context, name string, options *AddArtifactOptions) (string, error) {
 | 
						|
	if options == nil {
 | 
						|
		options = new(AddArtifactOptions)
 | 
						|
	}
 | 
						|
	optionsv4 := ModifyOptions{
 | 
						|
		Annotations: options.Annotation,
 | 
						|
		Arch:        options.Arch,
 | 
						|
		Features:    options.Features,
 | 
						|
		OS:          options.OS,
 | 
						|
		OSFeatures:  options.OSFeatures,
 | 
						|
		OSVersion:   options.OSVersion,
 | 
						|
		Variant:     options.Variant,
 | 
						|
 | 
						|
		ArtifactType:          options.Type,
 | 
						|
		ArtifactConfigType:    options.ConfigType,
 | 
						|
		ArtifactLayerType:     options.LayerType,
 | 
						|
		ArtifactConfig:        options.Config,
 | 
						|
		ArtifactExcludeTitles: options.ExcludeTitles,
 | 
						|
		ArtifactSubject:       options.Subject,
 | 
						|
		ArtifactAnnotations:   options.Annotations,
 | 
						|
	}
 | 
						|
	if len(options.Files) > 0 {
 | 
						|
		optionsv4.WithArtifactFiles(options.Files)
 | 
						|
	}
 | 
						|
	optionsv4.WithOperation("update")
 | 
						|
	return Modify(ctx, name, nil, &optionsv4)
 | 
						|
}
 | 
						|
 | 
						|
// Remove deletes a manifest entry from a manifest list.  Both name and the digest to be
 | 
						|
// removed are mandatory inputs.  The ID of the new manifest list is returned as a string.
 | 
						|
func Remove(ctx context.Context, name, digest string, _ *RemoveOptions) (string, error) {
 | 
						|
	optionsv4 := new(ModifyOptions).WithOperation("remove")
 | 
						|
	return Modify(ctx, name, []string{digest}, optionsv4)
 | 
						|
}
 | 
						|
 | 
						|
// Delete removes specified manifest from local storage.
 | 
						|
func Delete(ctx context.Context, name string) (*entitiesTypes.ManifestRemoveReport, error) {
 | 
						|
	var report entitiesTypes.ManifestRemoveReport
 | 
						|
	conn, err := bindings.GetClient(ctx)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	response, err := conn.DoRequest(ctx, nil, http.MethodDelete, "/manifests/%s", nil, nil, name)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	defer response.Body.Close()
 | 
						|
 | 
						|
	if err := response.Process(&report); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	return &report, errorhandling.JoinErrors(errorhandling.StringsToErrors(report.Errors))
 | 
						|
}
 | 
						|
 | 
						|
// Push takes a manifest list and pushes to a destination.  If the destination is not specified,
 | 
						|
// the name will be used instead.  If the optional all boolean is specified, all images specified
 | 
						|
// in the list will be pushed as well.
 | 
						|
func Push(ctx context.Context, name, destination string, options *images.PushOptions) (string, error) {
 | 
						|
	if options == nil {
 | 
						|
		options = new(images.PushOptions)
 | 
						|
	}
 | 
						|
	if len(destination) < 1 {
 | 
						|
		destination = name
 | 
						|
	}
 | 
						|
	conn, err := bindings.GetClient(ctx)
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
 | 
						|
	header, err := auth.MakeXRegistryAuthHeader(&imageTypes.SystemContext{AuthFilePath: options.GetAuthfile()}, options.GetUsername(), options.GetPassword())
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
 | 
						|
	params, err := options.ToParams()
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	// SkipTLSVerify is special.  It's not being serialized by ToParams()
 | 
						|
	// because we need to flip the boolean.
 | 
						|
	if options.SkipTLSVerify != nil {
 | 
						|
		params.Set("tlsVerify", strconv.FormatBool(!options.GetSkipTLSVerify()))
 | 
						|
	}
 | 
						|
 | 
						|
	response, err := conn.DoRequest(ctx, nil, http.MethodPost, "/manifests/%s/registry/%s", params, header, name, destination)
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	defer response.Body.Close()
 | 
						|
 | 
						|
	if !response.IsSuccess() {
 | 
						|
		return "", response.Process(err)
 | 
						|
	}
 | 
						|
 | 
						|
	var writer io.Writer
 | 
						|
	if options.GetQuiet() {
 | 
						|
		writer = io.Discard
 | 
						|
	} else if progressWriter := options.GetProgressWriter(); progressWriter != nil {
 | 
						|
		writer = progressWriter
 | 
						|
	} else {
 | 
						|
		// Historically push writes status to stderr
 | 
						|
		writer = os.Stderr
 | 
						|
	}
 | 
						|
 | 
						|
	dec := json.NewDecoder(response.Body)
 | 
						|
	for {
 | 
						|
		var report entitiesTypes.ManifestPushReport
 | 
						|
		if err := dec.Decode(&report); err != nil {
 | 
						|
			return "", err
 | 
						|
		}
 | 
						|
 | 
						|
		select {
 | 
						|
		case <-response.Request.Context().Done():
 | 
						|
			return "", context.Canceled
 | 
						|
		default:
 | 
						|
			// non-blocking select
 | 
						|
		}
 | 
						|
 | 
						|
		switch {
 | 
						|
		case report.ID != "":
 | 
						|
			return report.ID, nil
 | 
						|
		case report.Stream != "":
 | 
						|
			fmt.Fprint(writer, report.Stream)
 | 
						|
		case report.Error != "":
 | 
						|
			// There can only be one error.
 | 
						|
			return "", errors.New(report.Error)
 | 
						|
		default:
 | 
						|
			return "", fmt.Errorf("failed to parse push results stream, unexpected input: %v", report)
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Modify modifies the given manifest list using options and the optional list of images
 | 
						|
func Modify(ctx context.Context, name string, images []string, options *ModifyOptions) (string, error) {
 | 
						|
	if options == nil || *options.Operation == "" {
 | 
						|
		return "", errors.New(`the field ModifyOptions.Operation must be set to either "update" or "remove"`)
 | 
						|
	}
 | 
						|
	options.WithImages(images)
 | 
						|
 | 
						|
	var artifactFiles, artifactBaseNames []string
 | 
						|
	if options.ArtifactFiles != nil && len(*options.ArtifactFiles) > 0 {
 | 
						|
		artifactFiles = slices.Clone(*options.ArtifactFiles)
 | 
						|
		artifactBaseNames = make([]string, 0, len(artifactFiles))
 | 
						|
		for _, filename := range artifactFiles {
 | 
						|
			artifactBaseNames = append(artifactBaseNames, filepath.Base(filename))
 | 
						|
		}
 | 
						|
		options.ArtifactFiles = &artifactBaseNames
 | 
						|
	}
 | 
						|
 | 
						|
	conn, err := bindings.GetClient(ctx)
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	opts, err := jsoniter.MarshalToString(options)
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	reader := io.Reader(strings.NewReader(opts))
 | 
						|
	if options.Body != nil {
 | 
						|
		reader = io.MultiReader(reader, *options.Body)
 | 
						|
	}
 | 
						|
	var artifactContentType string
 | 
						|
	var artifactWriterGroup sync.WaitGroup
 | 
						|
	var artifactWriterError error
 | 
						|
	if len(artifactFiles) > 0 {
 | 
						|
		// get ready to upload the passed-in files
 | 
						|
		bodyReader, bodyWriter := io.Pipe()
 | 
						|
		defer bodyReader.Close()
 | 
						|
		requestBodyReader := reader
 | 
						|
		reader = bodyReader
 | 
						|
		// upload the files in another goroutine
 | 
						|
		writer := multipart.NewWriter(bodyWriter)
 | 
						|
		artifactContentType = writer.FormDataContentType()
 | 
						|
		artifactWriterGroup.Add(1)
 | 
						|
		go func() {
 | 
						|
			defer bodyWriter.Close()
 | 
						|
			defer writer.Close()
 | 
						|
			// start with the body we would have uploaded if we weren't
 | 
						|
			// attaching artifacts
 | 
						|
			headers := textproto.MIMEHeader{
 | 
						|
				"Content-Type": []string{"application/json"},
 | 
						|
			}
 | 
						|
			requestPartWriter, err := writer.CreatePart(headers)
 | 
						|
			if err != nil {
 | 
						|
				artifactWriterError = fmt.Errorf("creating form part for request: %v", err)
 | 
						|
				return
 | 
						|
			}
 | 
						|
			if _, err := io.Copy(requestPartWriter, requestBodyReader); err != nil {
 | 
						|
				artifactWriterError = fmt.Errorf("uploading request as form part: %v", err)
 | 
						|
				return
 | 
						|
			}
 | 
						|
			// now walk the list of files we're attaching
 | 
						|
			for _, file := range artifactFiles {
 | 
						|
				if err := func() error {
 | 
						|
					f, err := os.Open(file)
 | 
						|
					if err != nil {
 | 
						|
						return err
 | 
						|
					}
 | 
						|
					defer f.Close()
 | 
						|
					fileBase := filepath.Base(file)
 | 
						|
					formFile, err := writer.CreateFormFile(fileBase, fileBase)
 | 
						|
					if err != nil {
 | 
						|
						return err
 | 
						|
					}
 | 
						|
					st, err := f.Stat()
 | 
						|
					if err != nil {
 | 
						|
						return err
 | 
						|
					}
 | 
						|
					// upload the file contents
 | 
						|
					n, err := io.Copy(formFile, f)
 | 
						|
					if err != nil {
 | 
						|
						return fmt.Errorf("uploading contents of artifact file %s: %w", filepath.Base(file), err)
 | 
						|
					}
 | 
						|
					if n != st.Size() {
 | 
						|
						return fmt.Errorf("short write while uploading contents of artifact file %s: %d != %d", filepath.Base(file), n, st.Size())
 | 
						|
					}
 | 
						|
					return nil
 | 
						|
				}(); err != nil {
 | 
						|
					artifactWriterError = err
 | 
						|
					break
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}()
 | 
						|
	}
 | 
						|
 | 
						|
	header, err := auth.MakeXRegistryAuthHeader(&imageTypes.SystemContext{AuthFilePath: options.GetAuthfile()}, options.GetUsername(), options.GetPassword())
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	if artifactContentType != "" {
 | 
						|
		header["Content-Type"] = []string{artifactContentType}
 | 
						|
	}
 | 
						|
 | 
						|
	params, err := options.ToParams()
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	// SkipTLSVerify is special.  It's not being serialized by ToParams()
 | 
						|
	// because we need to flip the boolean.
 | 
						|
	if options.SkipTLSVerify != nil {
 | 
						|
		params.Set("tlsVerify", strconv.FormatBool(!options.GetSkipTLSVerify()))
 | 
						|
	}
 | 
						|
 | 
						|
	response, err := conn.DoRequest(ctx, reader, http.MethodPut, "/manifests/%s", params, header, name)
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	defer response.Body.Close()
 | 
						|
 | 
						|
	artifactWriterGroup.Wait()
 | 
						|
	if artifactWriterError != nil {
 | 
						|
		return "", fmt.Errorf("uploading artifacts: %w", err)
 | 
						|
	}
 | 
						|
 | 
						|
	data, err := io.ReadAll(response.Body)
 | 
						|
	if err != nil {
 | 
						|
		return "", fmt.Errorf("unable to process API response: %w", err)
 | 
						|
	}
 | 
						|
 | 
						|
	if response.IsSuccess() || response.IsRedirection() {
 | 
						|
		var report entitiesTypes.ManifestModifyReport
 | 
						|
		if err = jsoniter.Unmarshal(data, &report); err != nil {
 | 
						|
			return "", fmt.Errorf("unable to decode API response: %w", err)
 | 
						|
		}
 | 
						|
 | 
						|
		err = errorhandling.JoinErrors(report.Errors)
 | 
						|
		if err != nil {
 | 
						|
			errModel := errorhandling.ErrorModel{
 | 
						|
				Because:      errorhandling.Cause(err).Error(),
 | 
						|
				Message:      err.Error(),
 | 
						|
				ResponseCode: response.StatusCode,
 | 
						|
			}
 | 
						|
			return report.ID, &errModel
 | 
						|
		}
 | 
						|
		return report.ID, nil
 | 
						|
	}
 | 
						|
 | 
						|
	errModel := errorhandling.ErrorModel{
 | 
						|
		ResponseCode: response.StatusCode,
 | 
						|
	}
 | 
						|
	if err = jsoniter.Unmarshal(data, &errModel); err != nil {
 | 
						|
		return "", fmt.Errorf("unable to decode API response: %w", err)
 | 
						|
	}
 | 
						|
	return "", &errModel
 | 
						|
}
 | 
						|
 | 
						|
// Annotate modifies the given manifest list using options and the optional list of images
 | 
						|
//
 | 
						|
// As of 4.0.0
 | 
						|
func Annotate(ctx context.Context, name string, images []string, options *ModifyOptions) (string, error) {
 | 
						|
	options.WithOperation("annotate")
 | 
						|
	return Modify(ctx, name, images, options)
 | 
						|
}
 |