* bug: Use DSSE media types for artifactType * Don't serialize DSSE extension if not present * Update pkg/attestation/types.go Co-authored-by: Joel Kamp <joel.kamp@docker.com> * Don't error on no referrers --------- Co-authored-by: Joel Kamp <joel.kamp@docker.com>
145 lines
4.0 KiB
Go
145 lines
4.0 KiB
Go
package oci
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/docker/attest/pkg/attestation"
|
|
att "github.com/docker/attest/pkg/attestation"
|
|
v1 "github.com/google/go-containerregistry/pkg/v1"
|
|
"github.com/google/go-containerregistry/pkg/v1/layout"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
// implementation of AttestationResolver that closes over attestations from an oci layout
|
|
type OCILayoutResolver struct {
|
|
*attestation.AttestationManifest
|
|
*ImageSpec
|
|
}
|
|
|
|
func NewOCILayoutAttestationResolver(src *ImageSpec) (*OCILayoutResolver, error) {
|
|
r := &OCILayoutResolver{
|
|
ImageSpec: src,
|
|
}
|
|
_, err := r.fetchAttestationManifest()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return r, nil
|
|
}
|
|
|
|
func (r *OCILayoutResolver) fetchAttestationManifest() (*attestation.AttestationManifest, error) {
|
|
if r.AttestationManifest == nil {
|
|
m, err := attestationManifestFromOCILayout(r.Identifier, r.ImageSpec.Platform)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
r.AttestationManifest = m
|
|
}
|
|
|
|
return r.AttestationManifest, nil
|
|
}
|
|
|
|
func (r *OCILayoutResolver) Attestations(ctx context.Context, predicateType string) ([]*att.Envelope, error) {
|
|
var envs []*att.Envelope
|
|
dsseMediaType, err := attestation.DSSEMediaType(predicateType)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get DSSE media type for predicate '%s': %w", predicateType, err)
|
|
}
|
|
for _, attestationLayer := range r.AttestationManifest.OriginalLayers {
|
|
mt, err := attestationLayer.Layer.MediaType()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get layer media type: %w", err)
|
|
}
|
|
mts := string(mt)
|
|
if mts != dsseMediaType {
|
|
continue
|
|
}
|
|
var env = new(att.Envelope)
|
|
// parse layer blob as json
|
|
r, err := attestationLayer.Layer.Uncompressed()
|
|
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get layer contents: %w", err)
|
|
}
|
|
defer r.Close()
|
|
err = json.NewDecoder(r).Decode(env)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to decode envelope: %w", err)
|
|
}
|
|
envs = append(envs, env)
|
|
}
|
|
return envs, nil
|
|
}
|
|
|
|
func (r *OCILayoutResolver) ImageName(ctx context.Context) (string, error) {
|
|
return r.SubjectName, nil
|
|
}
|
|
|
|
func (r *OCILayoutResolver) ImageDescriptor(ctx context.Context) (*v1.Descriptor, error) {
|
|
return r.SubjectDescriptor, nil
|
|
}
|
|
|
|
func (r *OCILayoutResolver) ImagePlatform(ctx context.Context) (*v1.Platform, error) {
|
|
return r.ImageSpec.Platform, nil
|
|
}
|
|
|
|
func attestationManifestFromOCILayout(path string, platform *v1.Platform) (*attestation.AttestationManifest, error) {
|
|
idx, err := layout.ImageIndexFromPath(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
idxm, err := idx.IndexManifest()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get digest: %w", err)
|
|
}
|
|
|
|
idxDescriptor := idxm.Manifests[0]
|
|
name := idxDescriptor.Annotations["org.opencontainers.image.ref.name"]
|
|
idxDigest := idxDescriptor.Digest
|
|
|
|
mfs, err := idx.ImageIndex(idxDigest)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to extract ImageIndex for digest %s: %w", idxDigest.String(), err)
|
|
}
|
|
mfs2, err := mfs.IndexManifest()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to extract IndexManifest from ImageIndex: %w", err)
|
|
}
|
|
var subjectDescriptor *v1.Descriptor
|
|
for _, mf := range mfs2.Manifests {
|
|
if mf.Platform.Equals(*platform) {
|
|
subjectDescriptor = &mf
|
|
break
|
|
}
|
|
}
|
|
for _, mf := range mfs2.Manifests {
|
|
if mf.Annotations[att.DockerReferenceType] != attestation.AttestationManifestType {
|
|
continue
|
|
}
|
|
|
|
if mf.Annotations[att.DockerReferenceDigest] != subjectDescriptor.Digest.String() {
|
|
continue
|
|
}
|
|
|
|
attestationImage, err := mfs.Image(mf.Digest)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to extract attestation image with digest %s: %w", mf.Digest.String(), err)
|
|
}
|
|
layers, err := attestation.GetAttestationsFromImage(attestationImage)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get attestations from image: %w", err)
|
|
}
|
|
attest := &attestation.AttestationManifest{
|
|
OriginalLayers: layers,
|
|
OriginalDescriptor: &mf,
|
|
SubjectName: name,
|
|
SubjectDescriptor: subjectDescriptor,
|
|
}
|
|
return attest, nil
|
|
}
|
|
return nil, errors.New("attestation manifest not found")
|
|
}
|