2016-11-17 18:28:05 +00:00
|
|
|
package ocischema
|
|
|
|
|
|
|
|
import (
|
2017-09-01 02:14:40 +00:00
|
|
|
"context"
|
2018-07-19 21:07:26 +00:00
|
|
|
"errors"
|
2017-09-01 02:14:40 +00:00
|
|
|
|
2020-08-24 11:18:39 +00:00
|
|
|
"github.com/distribution/distribution/v3"
|
2016-11-17 18:28:05 +00:00
|
|
|
"github.com/opencontainers/go-digest"
|
2023-04-30 16:16:51 +00:00
|
|
|
"github.com/opencontainers/image-spec/specs-go"
|
2019-10-09 12:02:21 +00:00
|
|
|
v1 "github.com/opencontainers/image-spec/specs-go/v1"
|
2016-11-17 18:28:05 +00:00
|
|
|
)
|
|
|
|
|
2018-06-19 18:48:42 +00:00
|
|
|
// Builder is a type for constructing manifests.
|
2018-03-15 20:04:58 +00:00
|
|
|
type Builder struct {
|
2016-11-17 18:28:05 +00:00
|
|
|
// bs is a BlobService used to publish the configuration blob.
|
|
|
|
bs distribution.BlobService
|
|
|
|
|
|
|
|
// configJSON references
|
|
|
|
configJSON []byte
|
|
|
|
|
|
|
|
// layers is a list of layer descriptors that gets built by successive
|
|
|
|
// calls to AppendReference.
|
|
|
|
layers []distribution.Descriptor
|
2017-07-31 23:34:11 +00:00
|
|
|
|
|
|
|
// Annotations contains arbitrary metadata relating to the targeted content.
|
|
|
|
annotations map[string]string
|
2018-03-15 20:04:58 +00:00
|
|
|
|
|
|
|
// For testing purposes
|
|
|
|
mediaType string
|
2016-11-17 18:28:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewManifestBuilder is used to build new manifests for the current schema
|
|
|
|
// version. It takes a BlobService so it can publish the configuration blob
|
2017-07-31 23:34:11 +00:00
|
|
|
// as part of the Build process, and annotations.
|
2023-05-02 16:09:48 +00:00
|
|
|
func NewManifestBuilder(bs distribution.BlobService, configJSON []byte, annotations map[string]string) *Builder {
|
2018-03-15 20:04:58 +00:00
|
|
|
mb := &Builder{
|
2017-07-31 23:34:11 +00:00
|
|
|
bs: bs,
|
|
|
|
configJSON: make([]byte, len(configJSON)),
|
|
|
|
annotations: annotations,
|
2018-03-15 20:04:58 +00:00
|
|
|
mediaType: v1.MediaTypeImageManifest,
|
2016-11-17 18:28:05 +00:00
|
|
|
}
|
|
|
|
copy(mb.configJSON, configJSON)
|
|
|
|
|
|
|
|
return mb
|
|
|
|
}
|
|
|
|
|
2018-06-19 18:48:42 +00:00
|
|
|
// SetMediaType assigns the passed mediatype or error if the mediatype is not a
|
|
|
|
// valid media type for oci image manifests currently: "" or "application/vnd.oci.image.manifest.v1+json"
|
2018-07-19 21:07:26 +00:00
|
|
|
func (mb *Builder) SetMediaType(mediaType string) error {
|
2018-03-15 20:04:58 +00:00
|
|
|
if mediaType != "" && mediaType != v1.MediaTypeImageManifest {
|
2019-02-05 00:01:04 +00:00
|
|
|
return errors.New("invalid media type for OCI image manifest")
|
2018-03-15 20:04:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mb.mediaType = mediaType
|
2018-07-19 21:07:26 +00:00
|
|
|
return nil
|
2018-03-15 20:04:58 +00:00
|
|
|
}
|
|
|
|
|
2016-11-17 18:28:05 +00:00
|
|
|
// Build produces a final manifest from the given references.
|
2018-03-15 20:04:58 +00:00
|
|
|
func (mb *Builder) Build(ctx context.Context) (distribution.Manifest, error) {
|
2016-11-17 18:28:05 +00:00
|
|
|
m := Manifest{
|
2023-04-30 16:16:51 +00:00
|
|
|
Versioned: specs.Versioned{SchemaVersion: 2},
|
|
|
|
MediaType: mb.mediaType,
|
2017-07-31 23:34:11 +00:00
|
|
|
Layers: make([]distribution.Descriptor, len(mb.layers)),
|
|
|
|
Annotations: mb.annotations,
|
2016-11-17 18:28:05 +00:00
|
|
|
}
|
|
|
|
copy(m.Layers, mb.layers)
|
|
|
|
|
|
|
|
configDigest := digest.FromBytes(mb.configJSON)
|
|
|
|
|
|
|
|
var err error
|
|
|
|
m.Config, err = mb.bs.Stat(ctx, configDigest)
|
|
|
|
switch err {
|
|
|
|
case nil:
|
|
|
|
// Override MediaType, since Put always replaces the specified media
|
|
|
|
// type with application/octet-stream in the descriptor it returns.
|
2017-07-11 19:19:47 +00:00
|
|
|
m.Config.MediaType = v1.MediaTypeImageConfig
|
2016-11-17 18:28:05 +00:00
|
|
|
return FromStruct(m)
|
|
|
|
case distribution.ErrBlobUnknown:
|
|
|
|
// nop
|
|
|
|
default:
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add config to the blob store
|
2017-07-11 19:19:47 +00:00
|
|
|
m.Config, err = mb.bs.Put(ctx, v1.MediaTypeImageConfig, mb.configJSON)
|
2016-11-17 18:28:05 +00:00
|
|
|
// Override MediaType, since Put always replaces the specified media
|
|
|
|
// type with application/octet-stream in the descriptor it returns.
|
2017-07-11 19:19:47 +00:00
|
|
|
m.Config.MediaType = v1.MediaTypeImageConfig
|
2016-11-17 18:28:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return FromStruct(m)
|
|
|
|
}
|
|
|
|
|
|
|
|
// AppendReference adds a reference to the current ManifestBuilder.
|
Descriptor: do not implement Describable interface
Commit cb6f0023500c3d2afb8c9f3ee4a0097526192156 implemented a generic
Manifest interface to represent manifests in the registry and remove
references to schema specific manifests.
As part of this refactor, the Describable interface was introduced,
which allowed for a single ManifestBuilder interface to handle both
schema1 and schema2 manifests. Implementations of Describable are
generally objects which can be described, not simply descriptors, but
for convenience, this interface was also implemented on Descriptor in
2ff77c00bad887928be04367f0dd58f6aed5b756.
This interface served its purpose, but no longer needed for most cases;
schema2 (and OCI) descriptors do not need this method, making it only
needed for `schema1.Reference`, which is now deprecated.
Requiring this interface to be implemented limits interoperability
between distribution's Descriptor and the OCI Descriptor types, which
are identical in every other way, except for the presence of the
Describable interface.
This patch:
- Removes the `Descriptor.Descriptor()` method (no longer implementing
the `Describable` interface).
- Updates ManifestBuilder interface and implementations to accept either
- Updates ManifestBuilder interface and implementations to accept a
`Descriptor`.
After this patch, the caller is responsible for changing a describable
type into a descriptor;
builder.AppendReference(describable.Descriptor())
Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
2023-04-30 15:53:17 +00:00
|
|
|
func (mb *Builder) AppendReference(ref distribution.Descriptor) error {
|
|
|
|
mb.layers = append(mb.layers, ref)
|
2016-11-17 18:28:05 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// References returns the current references added to this builder.
|
2018-03-15 20:04:58 +00:00
|
|
|
func (mb *Builder) References() []distribution.Descriptor {
|
2016-11-17 18:28:05 +00:00
|
|
|
return mb.layers
|
|
|
|
}
|