mirror of
				https://github.com/containers/podman.git
				synced 2025-10-31 18:08:51 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			298 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			298 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package manifests
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"encoding/json"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"net/http"
 | |
| 	"os"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/containers/image/v5/manifest"
 | |
| 	imageTypes "github.com/containers/image/v5/types"
 | |
| 	"github.com/containers/podman/v4/pkg/auth"
 | |
| 	"github.com/containers/podman/v4/pkg/bindings"
 | |
| 	"github.com/containers/podman/v4/pkg/bindings/images"
 | |
| 	"github.com/containers/podman/v4/pkg/domain/entities"
 | |
| 	"github.com/containers/podman/v4/pkg/errorhandling"
 | |
| 	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 entities.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, _ *InspectOptions) (*manifest.Schema2List, error) {
 | |
| 	conn, err := bindings.GetClient(ctx)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	response, err := conn.DoRequest(ctx, nil, http.MethodGet, "/manifests/%s/json", nil, nil, name)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	defer response.Body.Close()
 | |
| 
 | |
| 	var list manifest.Schema2List
 | |
| 	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:    nil,
 | |
| 		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)
 | |
| }
 | |
| 
 | |
| // 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) (*entities.ManifestRemoveReport, error) {
 | |
| 	var report entities.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 entities.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)
 | |
| 
 | |
| 	conn, err := bindings.GetClient(ctx)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	opts, err := jsoniter.MarshalToString(options)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	reader := strings.NewReader(opts)
 | |
| 
 | |
| 	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, reader, http.MethodPut, "/manifests/%s", params, header, name)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	defer response.Body.Close()
 | |
| 
 | |
| 	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 entities.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)
 | |
| }
 | 
