mirror of
https://github.com/containers/podman.git
synced 2025-07-15 03:02:52 +08:00

We now no longer write containers.conf, instead system connections and farms are written to a new file called podman-connections.conf. This is a major rework and I had to change a lot of things to get this to compile again with my c/common changes. It is a breaking change for users as connections/farms added before this commit can now no longer be removed or modified directly. However because the logic keeps reading from containers.conf the old connections can still be used to connect to a remote host. Signed-off-by: Paul Holzinger <pholzing@redhat.com>
627 lines
20 KiB
Go
627 lines
20 KiB
Go
package manifests
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
|
|
"github.com/containers/image/v5/manifest"
|
|
digest "github.com/opencontainers/go-digest"
|
|
imgspec "github.com/opencontainers/image-spec/specs-go"
|
|
v1 "github.com/opencontainers/image-spec/specs-go/v1"
|
|
)
|
|
|
|
// List is a generic interface for manipulating a manifest list or an image
|
|
// index.
|
|
type List interface {
|
|
AddInstance(manifestDigest digest.Digest, manifestSize int64, manifestType, os, architecture, osVersion string, osFeatures []string, variant string, features []string, annotations []string) error
|
|
Remove(instanceDigest digest.Digest) error
|
|
SetURLs(instanceDigest digest.Digest, urls []string) error
|
|
URLs(instanceDigest digest.Digest) ([]string, error)
|
|
ClearAnnotations(instanceDigest *digest.Digest) error
|
|
SetAnnotations(instanceDigest *digest.Digest, annotations map[string]string) error
|
|
Annotations(instanceDigest *digest.Digest) (map[string]string, error)
|
|
SetOS(instanceDigest digest.Digest, os string) error
|
|
OS(instanceDigest digest.Digest) (string, error)
|
|
SetArchitecture(instanceDigest digest.Digest, arch string) error
|
|
Architecture(instanceDigest digest.Digest) (string, error)
|
|
SetOSVersion(instanceDigest digest.Digest, osVersion string) error
|
|
OSVersion(instanceDigest digest.Digest) (string, error)
|
|
SetVariant(instanceDigest digest.Digest, variant string) error
|
|
Variant(instanceDigest digest.Digest) (string, error)
|
|
SetFeatures(instanceDigest digest.Digest, features []string) error
|
|
Features(instanceDigest digest.Digest) ([]string, error)
|
|
SetOSFeatures(instanceDigest digest.Digest, osFeatures []string) error
|
|
OSFeatures(instanceDigest digest.Digest) ([]string, error)
|
|
SetMediaType(instanceDigest digest.Digest, mediaType string) error
|
|
MediaType(instanceDigest digest.Digest) (string, error)
|
|
SetArtifactType(instanceDigest digest.Digest, artifactType string) error
|
|
ArtifactType(instanceDigest digest.Digest) (string, error)
|
|
Serialize(mimeType string) ([]byte, error)
|
|
Instances() []digest.Digest
|
|
OCIv1() *v1.Index
|
|
Docker() *manifest.Schema2List
|
|
|
|
findDocker(instanceDigest digest.Digest) (*manifest.Schema2ManifestDescriptor, error)
|
|
findOCIv1(instanceDigest digest.Digest) (*v1.Descriptor, error)
|
|
}
|
|
|
|
type list struct {
|
|
docker manifest.Schema2List
|
|
oci v1.Index
|
|
}
|
|
|
|
// OCIv1 returns the list as a Docker schema 2 list. The returned structure should NOT be modified.
|
|
func (l *list) Docker() *manifest.Schema2List {
|
|
return &l.docker
|
|
}
|
|
|
|
// OCIv1 returns the list as an OCI image index. The returned structure should NOT be modified.
|
|
func (l *list) OCIv1() *v1.Index {
|
|
return &l.oci
|
|
}
|
|
|
|
// Create creates a new list.
|
|
func Create() List {
|
|
return &list{
|
|
docker: manifest.Schema2List{
|
|
SchemaVersion: 2,
|
|
MediaType: manifest.DockerV2ListMediaType,
|
|
},
|
|
oci: v1.Index{
|
|
Versioned: imgspec.Versioned{SchemaVersion: 2},
|
|
MediaType: v1.MediaTypeImageIndex,
|
|
},
|
|
}
|
|
}
|
|
|
|
// AddInstance adds an entry for the specified manifest digest, with assorted
|
|
// additional information specified in parameters, to the list or index.
|
|
func (l *list) AddInstance(manifestDigest digest.Digest, manifestSize int64, manifestType, osName, architecture, osVersion string, osFeatures []string, variant string, features, annotations []string) error { // nolint:revive
|
|
// FIXME: the annotations argument is currently ignored
|
|
if err := l.Remove(manifestDigest); err != nil && !errors.Is(err, os.ErrNotExist) {
|
|
return err
|
|
}
|
|
|
|
schema2platform := manifest.Schema2PlatformSpec{
|
|
Architecture: architecture,
|
|
OS: osName,
|
|
OSVersion: osVersion,
|
|
OSFeatures: osFeatures,
|
|
Variant: variant,
|
|
Features: features,
|
|
}
|
|
l.docker.Manifests = append(l.docker.Manifests, manifest.Schema2ManifestDescriptor{
|
|
Schema2Descriptor: manifest.Schema2Descriptor{
|
|
MediaType: manifestType,
|
|
Size: manifestSize,
|
|
Digest: manifestDigest,
|
|
},
|
|
Platform: schema2platform,
|
|
})
|
|
|
|
ociv1platform := &v1.Platform{
|
|
Architecture: architecture,
|
|
OS: osName,
|
|
OSVersion: osVersion,
|
|
OSFeatures: osFeatures,
|
|
Variant: variant,
|
|
}
|
|
if ociv1platform.Architecture == "" && ociv1platform.OS == "" && ociv1platform.OSVersion == "" && ociv1platform.Variant == "" && len(ociv1platform.OSFeatures) == 0 {
|
|
ociv1platform = nil
|
|
}
|
|
l.oci.Manifests = append(l.oci.Manifests, v1.Descriptor{
|
|
MediaType: manifestType,
|
|
Size: manifestSize,
|
|
Digest: manifestDigest,
|
|
Platform: ociv1platform,
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
// Remove filters out any instances in the list which match the specified digest.
|
|
func (l *list) Remove(instanceDigest digest.Digest) error {
|
|
err := fmt.Errorf("no instance matching digest %q found in manifest list: %w", instanceDigest, os.ErrNotExist)
|
|
newDockerManifests := make([]manifest.Schema2ManifestDescriptor, 0, len(l.docker.Manifests))
|
|
for i := range l.docker.Manifests {
|
|
if l.docker.Manifests[i].Digest != instanceDigest {
|
|
newDockerManifests = append(newDockerManifests, l.docker.Manifests[i])
|
|
} else {
|
|
err = nil
|
|
}
|
|
}
|
|
l.docker.Manifests = newDockerManifests
|
|
newOCIv1Manifests := make([]v1.Descriptor, 0, len(l.oci.Manifests))
|
|
for i := range l.oci.Manifests {
|
|
if l.oci.Manifests[i].Digest != instanceDigest {
|
|
newOCIv1Manifests = append(newOCIv1Manifests, l.oci.Manifests[i])
|
|
} else {
|
|
err = nil
|
|
}
|
|
}
|
|
l.oci.Manifests = newOCIv1Manifests
|
|
return err
|
|
}
|
|
|
|
func (l *list) findDocker(instanceDigest digest.Digest) (*manifest.Schema2ManifestDescriptor, error) {
|
|
for i := range l.docker.Manifests {
|
|
if l.docker.Manifests[i].Digest == instanceDigest {
|
|
return &l.docker.Manifests[i], nil
|
|
}
|
|
}
|
|
return nil, fmt.Errorf("no Docker manifest matching digest %q was found in list: %w", instanceDigest.String(), ErrDigestNotFound)
|
|
}
|
|
|
|
func (l *list) findOCIv1(instanceDigest digest.Digest) (*v1.Descriptor, error) {
|
|
for i := range l.oci.Manifests {
|
|
if l.oci.Manifests[i].Digest == instanceDigest {
|
|
return &l.oci.Manifests[i], nil
|
|
}
|
|
}
|
|
return nil, fmt.Errorf("no OCI manifest matching digest %q was found in list: %w", instanceDigest.String(), ErrDigestNotFound)
|
|
}
|
|
|
|
// SetURLs sets the URLs where the manifest might also be found.
|
|
func (l *list) SetURLs(instanceDigest digest.Digest, urls []string) error {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci.URLs = append([]string{}, urls...)
|
|
if len(oci.URLs) == 0 {
|
|
oci.URLs = nil
|
|
}
|
|
docker.URLs = append([]string{}, urls...)
|
|
if len(docker.URLs) == 0 {
|
|
docker.URLs = nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// URLs retrieves the locations from which this object might possibly be downloaded.
|
|
func (l *list) URLs(instanceDigest digest.Digest) ([]string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return append([]string{}, oci.URLs...), nil
|
|
}
|
|
|
|
// ClearAnnotations removes all annotations from the image index, or from a
|
|
// specific manifest.
|
|
// The field is specific to the OCI image index format, and is not present in Docker manifest lists.
|
|
func (l *list) ClearAnnotations(instanceDigest *digest.Digest) error {
|
|
a := &l.oci.Annotations
|
|
if instanceDigest != nil {
|
|
oci, err := l.findOCIv1(*instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
a = &oci.Annotations
|
|
}
|
|
*a = nil
|
|
return nil
|
|
}
|
|
|
|
// SetAnnotations sets annotations on the image index, or on a specific
|
|
// manifest.
|
|
// The field is specific to the OCI image index format, and is not present in Docker manifest lists.
|
|
func (l *list) SetAnnotations(instanceDigest *digest.Digest, annotations map[string]string) error {
|
|
a := &l.oci.Annotations
|
|
if instanceDigest != nil {
|
|
oci, err := l.findOCIv1(*instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
a = &oci.Annotations
|
|
}
|
|
if *a == nil {
|
|
(*a) = make(map[string]string)
|
|
}
|
|
for k, v := range annotations {
|
|
(*a)[k] = v
|
|
}
|
|
if len(*a) == 0 {
|
|
*a = nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Annotations retrieves the annotations which have been set on the image index, or on one instance.
|
|
// The field is specific to the OCI image index format, and is not present in Docker manifest lists.
|
|
func (l *list) Annotations(instanceDigest *digest.Digest) (map[string]string, error) {
|
|
a := l.oci.Annotations
|
|
if instanceDigest != nil {
|
|
oci, err := l.findOCIv1(*instanceDigest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
a = oci.Annotations
|
|
}
|
|
annotations := make(map[string]string)
|
|
for k, v := range a {
|
|
annotations[k] = v
|
|
}
|
|
return annotations, nil
|
|
}
|
|
|
|
// SetOS sets the OS field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) SetOS(instanceDigest digest.Digest, os string) error {
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
docker.Platform.OS = os
|
|
if oci.Platform == nil {
|
|
oci.Platform = &v1.Platform{}
|
|
}
|
|
oci.Platform.OS = os
|
|
if oci.Platform.Architecture == "" && oci.Platform.OS == "" && oci.Platform.OSVersion == "" && oci.Platform.Variant == "" && len(oci.Platform.OSFeatures) == 0 {
|
|
oci.Platform = nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// OS retrieves the OS field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) OS(instanceDigest digest.Digest) (string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
platform := oci.Platform
|
|
if platform == nil {
|
|
platform = &v1.Platform{}
|
|
}
|
|
return platform.OS, nil
|
|
}
|
|
|
|
// SetArchitecture sets the Architecture field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) SetArchitecture(instanceDigest digest.Digest, arch string) error {
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
docker.Platform.Architecture = arch
|
|
if oci.Platform == nil {
|
|
oci.Platform = &v1.Platform{}
|
|
}
|
|
oci.Platform.Architecture = arch
|
|
if oci.Platform.Architecture == "" && oci.Platform.OS == "" && oci.Platform.OSVersion == "" && oci.Platform.Variant == "" && len(oci.Platform.OSFeatures) == 0 {
|
|
oci.Platform = nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Architecture retrieves the Architecture field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) Architecture(instanceDigest digest.Digest) (string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
platform := oci.Platform
|
|
if platform == nil {
|
|
platform = &v1.Platform{}
|
|
}
|
|
return platform.Architecture, nil
|
|
}
|
|
|
|
// SetOSVersion sets the OSVersion field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) SetOSVersion(instanceDigest digest.Digest, osVersion string) error {
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
docker.Platform.OSVersion = osVersion
|
|
if oci.Platform == nil {
|
|
oci.Platform = &v1.Platform{}
|
|
}
|
|
oci.Platform.OSVersion = osVersion
|
|
if oci.Platform.Architecture == "" && oci.Platform.OS == "" && oci.Platform.OSVersion == "" && oci.Platform.Variant == "" && len(oci.Platform.OSFeatures) == 0 {
|
|
oci.Platform = nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// OSVersion retrieves the OSVersion field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) OSVersion(instanceDigest digest.Digest) (string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
platform := oci.Platform
|
|
if platform == nil {
|
|
platform = &v1.Platform{}
|
|
}
|
|
return platform.OSVersion, nil
|
|
}
|
|
|
|
// SetVariant sets the Variant field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) SetVariant(instanceDigest digest.Digest, variant string) error {
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
docker.Platform.Variant = variant
|
|
if oci.Platform == nil {
|
|
oci.Platform = &v1.Platform{}
|
|
}
|
|
oci.Platform.Variant = variant
|
|
if oci.Platform.Architecture == "" && oci.Platform.OS == "" && oci.Platform.OSVersion == "" && oci.Platform.Variant == "" && len(oci.Platform.OSFeatures) == 0 {
|
|
oci.Platform = nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Variant retrieves the Variant field in the platform information associated with the instance with the specified digest.
|
|
func (l *list) Variant(instanceDigest digest.Digest) (string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
platform := oci.Platform
|
|
if platform == nil {
|
|
platform = &v1.Platform{}
|
|
}
|
|
return platform.Variant, nil
|
|
}
|
|
|
|
// SetFeatures sets the features list in the platform information associated with the instance with the specified digest.
|
|
// The field is specific to the Docker manifest list format, and is not present in OCI's image indexes.
|
|
func (l *list) SetFeatures(instanceDigest digest.Digest, features []string) error {
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
docker.Platform.Features = append([]string{}, features...)
|
|
if len(docker.Platform.Features) == 0 {
|
|
docker.Platform.Features = nil
|
|
}
|
|
// no OCI equivalent
|
|
return nil
|
|
}
|
|
|
|
// Features retrieves the features list from the platform information associated with the instance with the specified digest.
|
|
// The field is specific to the Docker manifest list format, and is not present in OCI's image indexes.
|
|
func (l *list) Features(instanceDigest digest.Digest) ([]string, error) {
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return append([]string{}, docker.Platform.Features...), nil
|
|
}
|
|
|
|
// SetOSFeatures sets the OS features list in the platform information associated with the instance with the specified digest.
|
|
func (l *list) SetOSFeatures(instanceDigest digest.Digest, osFeatures []string) error {
|
|
docker, err := l.findDocker(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
docker.Platform.OSFeatures = append([]string{}, osFeatures...)
|
|
if oci.Platform == nil {
|
|
oci.Platform = &v1.Platform{}
|
|
}
|
|
oci.Platform.OSFeatures = append([]string{}, osFeatures...)
|
|
if len(oci.Platform.OSFeatures) == 0 {
|
|
oci.Platform.OSFeatures = nil
|
|
}
|
|
if oci.Platform.Architecture == "" && oci.Platform.OS == "" && oci.Platform.OSVersion == "" && oci.Platform.Variant == "" && len(oci.Platform.OSFeatures) == 0 {
|
|
oci.Platform = nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// OSFeatures retrieves the OS features list from the platform information associated with the instance with the specified digest.
|
|
func (l *list) OSFeatures(instanceDigest digest.Digest) ([]string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
platform := oci.Platform
|
|
if platform == nil {
|
|
platform = &v1.Platform{}
|
|
}
|
|
return append([]string{}, platform.OSFeatures...), nil
|
|
}
|
|
|
|
// SetMediaType sets the MediaType field in the instance with the specified digest.
|
|
func (l *list) SetMediaType(instanceDigest digest.Digest, mediaType string) error {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci.MediaType = mediaType
|
|
return nil
|
|
}
|
|
|
|
// MediaType retrieves the MediaType field in the instance with the specified digest.
|
|
func (l *list) MediaType(instanceDigest digest.Digest) (string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return oci.MediaType, nil
|
|
}
|
|
|
|
// SetArtifactType sets the ArtifactType field in the instance with the specified digest.
|
|
func (l *list) SetArtifactType(instanceDigest digest.Digest, artifactType string) error {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oci.ArtifactType = artifactType
|
|
return nil
|
|
}
|
|
|
|
// ArtifactType retrieves the ArtifactType field in the instance with the specified digest.
|
|
func (l *list) ArtifactType(instanceDigest digest.Digest) (string, error) {
|
|
oci, err := l.findOCIv1(instanceDigest)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return oci.ArtifactType, nil
|
|
}
|
|
|
|
// FromBlob builds a list from an encoded manifest list or image index.
|
|
func FromBlob(manifestBytes []byte) (List, error) {
|
|
manifestType := manifest.GuessMIMEType(manifestBytes)
|
|
list := &list{
|
|
docker: manifest.Schema2List{
|
|
SchemaVersion: 2,
|
|
MediaType: manifest.DockerV2ListMediaType,
|
|
},
|
|
oci: v1.Index{
|
|
Versioned: imgspec.Versioned{SchemaVersion: 2},
|
|
MediaType: v1.MediaTypeImageIndex,
|
|
},
|
|
}
|
|
switch manifestType {
|
|
default:
|
|
return nil, fmt.Errorf("unable to load manifest list: unsupported format %q: %w", manifestType, ErrManifestTypeNotSupported)
|
|
case manifest.DockerV2ListMediaType:
|
|
if err := json.Unmarshal(manifestBytes, &list.docker); err != nil {
|
|
return nil, fmt.Errorf("unable to parse Docker manifest list from image: %w", err)
|
|
}
|
|
for _, m := range list.docker.Manifests {
|
|
list.oci.Manifests = append(list.oci.Manifests, v1.Descriptor{
|
|
MediaType: m.Schema2Descriptor.MediaType,
|
|
Size: m.Schema2Descriptor.Size,
|
|
Digest: m.Schema2Descriptor.Digest,
|
|
Platform: &v1.Platform{
|
|
Architecture: m.Platform.Architecture,
|
|
OS: m.Platform.OS,
|
|
OSVersion: m.Platform.OSVersion,
|
|
OSFeatures: m.Platform.OSFeatures,
|
|
Variant: m.Platform.Variant,
|
|
},
|
|
})
|
|
}
|
|
case v1.MediaTypeImageIndex:
|
|
if err := json.Unmarshal(manifestBytes, &list.oci); err != nil {
|
|
return nil, fmt.Errorf("unable to parse OCIv1 manifest list: %w", err)
|
|
}
|
|
for _, m := range list.oci.Manifests {
|
|
platform := m.Platform
|
|
if platform == nil {
|
|
platform = &v1.Platform{}
|
|
}
|
|
list.docker.Manifests = append(list.docker.Manifests, manifest.Schema2ManifestDescriptor{
|
|
Schema2Descriptor: manifest.Schema2Descriptor{
|
|
MediaType: m.MediaType,
|
|
Size: m.Size,
|
|
Digest: m.Digest,
|
|
},
|
|
Platform: manifest.Schema2PlatformSpec{
|
|
Architecture: platform.Architecture,
|
|
OS: platform.OS,
|
|
OSVersion: platform.OSVersion,
|
|
OSFeatures: platform.OSFeatures,
|
|
Variant: platform.Variant,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
return list, nil
|
|
}
|
|
|
|
func (l *list) preferOCI() bool {
|
|
// If we have any data that's only in the OCI format, use that.
|
|
if l.oci.ArtifactType != "" {
|
|
return true
|
|
}
|
|
if l.oci.Subject != nil {
|
|
return true
|
|
}
|
|
for _, m := range l.oci.Manifests {
|
|
if m.ArtifactType != "" {
|
|
return true
|
|
}
|
|
if len(m.Annotations) > 0 {
|
|
return true
|
|
}
|
|
if len(m.Data) > 0 {
|
|
return true
|
|
}
|
|
}
|
|
// If we have any data that's only in the Docker format, use that.
|
|
for _, m := range l.docker.Manifests {
|
|
if len(m.Platform.Features) > 0 {
|
|
return false
|
|
}
|
|
}
|
|
// If we have no manifests, remember that the Docker format is
|
|
// explicitly typed, so use that. Otherwise, default to using the OCI
|
|
// format.
|
|
return len(l.docker.Manifests) != 0
|
|
}
|
|
|
|
// Serialize encodes the list using the specified format, or by selecting one
|
|
// which it thinks is appropriate.
|
|
func (l *list) Serialize(mimeType string) ([]byte, error) {
|
|
var (
|
|
res []byte
|
|
err error
|
|
)
|
|
switch mimeType {
|
|
case "":
|
|
if l.preferOCI() {
|
|
res, err = json.Marshal(&l.oci)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("marshalling OCI image index: %w", err)
|
|
}
|
|
} else {
|
|
res, err = json.Marshal(&l.docker)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("marshalling Docker manifest list: %w", err)
|
|
}
|
|
}
|
|
case v1.MediaTypeImageIndex:
|
|
res, err = json.Marshal(&l.oci)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("marshalling OCI image index: %w", err)
|
|
}
|
|
case manifest.DockerV2ListMediaType:
|
|
res, err = json.Marshal(&l.docker)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("marshalling Docker manifest list: %w", err)
|
|
}
|
|
default:
|
|
return nil, fmt.Errorf("serializing list to type %q not implemented: %w", mimeType, ErrManifestTypeNotSupported)
|
|
}
|
|
return res, nil
|
|
}
|
|
|
|
// Instances returns the list of image instances mentioned in this list.
|
|
func (l *list) Instances() []digest.Digest {
|
|
instances := make([]digest.Digest, 0, len(l.oci.Manifests))
|
|
for _, instance := range l.oci.Manifests {
|
|
instances = append(instances, instance.Digest)
|
|
}
|
|
return instances
|
|
}
|