package client import ( "bytes" "context" "encoding/json" "errors" "fmt" "io" "net/http" "strings" "sync" cerrdefs "github.com/containerd/errdefs" ocispec "github.com/opencontainers/image-spec/specs-go/v1" ) type emptyIDError string func (e emptyIDError) InvalidParameter() {} func (e emptyIDError) Error() string { return "invalid " + string(e) + " name or ID: value is empty" } // trimID trims the given object-ID / name, returning an error if it's empty. func trimID(objType, id string) (string, error) { id = strings.TrimSpace(id) if id == "" { return "", emptyIDError(objType) } return id, nil } // encodePlatforms marshals the given platform(s) to JSON format, to // be used for query-parameters for filtering / selecting platforms. func encodePlatforms(platform ...ocispec.Platform) ([]string, error) { if len(platform) == 0 { return []string{}, nil } if len(platform) == 1 { p, err := encodePlatform(&platform[0]) if err != nil { return nil, err } return []string{p}, nil } seen := make(map[string]struct{}, len(platform)) out := make([]string, 0, len(platform)) for i := range platform { p, err := encodePlatform(&platform[i]) if err != nil { return nil, err } if _, ok := seen[p]; !ok { out = append(out, p) seen[p] = struct{}{} } } return out, nil } // encodePlatform marshals the given platform to JSON format, to // be used for query-parameters for filtering / selecting platforms. It // is used as a helper for encodePlatforms, func encodePlatform(platform *ocispec.Platform) (string, error) { p, err := json.Marshal(platform) if err != nil { return "", fmt.Errorf("%w: invalid platform: %v", cerrdefs.ErrInvalidArgument, err) } return string(p), nil } func decodeWithRaw[T any](resp *http.Response, out *T) (raw json.RawMessage, _ error) { if resp == nil || resp.Body == nil { return nil, errors.New("empty response") } defer ensureReaderClosed(resp) var buf bytes.Buffer tr := io.TeeReader(resp.Body, &buf) err := json.NewDecoder(tr).Decode(out) if err != nil { return nil, err } return buf.Bytes(), nil } // newCancelReadCloser wraps rc so it's automatically closed when ctx is canceled. // Close is idempotent and returns the first error from rc.Close. func newCancelReadCloser(ctx context.Context, rc io.ReadCloser) io.ReadCloser { crc := &cancelReadCloser{ rc: rc, close: sync.OnceValue(rc.Close), } context.AfterFunc(ctx, func() { _ = crc.Close() }) return crc } type cancelReadCloser struct { rc io.ReadCloser close func() error } func (c *cancelReadCloser) Read(p []byte) (int, error) { return c.rc.Read(p) } func (c *cancelReadCloser) Close() error { return c.close() }