Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: support copying referrers for multi-arch images #1122

Merged
merged 16 commits into from
Sep 20, 2023
72 changes: 55 additions & 17 deletions cmd/oras/root/cp.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ package root

import (
"context"
"encoding/json"
"fmt"
"strings"
"sync"
Expand All @@ -28,6 +29,7 @@ import (
"oras.land/oras-go/v2/content"
"oras.land/oras/cmd/oras/internal/display"
"oras.land/oras/cmd/oras/internal/option"
"oras.land/oras/internal/docker"
"oras.land/oras/internal/graph"
)

Expand Down Expand Up @@ -137,36 +139,27 @@ func runCopy(ctx context.Context, opts copyOptions) error {
var desc ocispec.Descriptor
rOpts := oras.DefaultResolveOptions
rOpts.TargetPlatform = opts.Platform.Platform
if dstRef := opts.To.Reference; dstRef == "" {
if opts.recursive {
desc, err = oras.Resolve(ctx, src, opts.From.Reference, rOpts)
if err != nil {
return fmt.Errorf("failed to resolve %s: %w", opts.From.Reference, err)
}
if opts.recursive {
err = oras.ExtendedCopyGraph(ctx, src, dst, desc, extendedCopyOptions.ExtendedCopyGraphOptions)
} else {
err = oras.CopyGraph(ctx, src, dst, desc, extendedCopyOptions.CopyGraphOptions)
}
err = recursiveCopy(ctx, src, dst, opts.To.Reference, desc, extendedCopyOptions)
} else {
if opts.recursive {
srcRef := opts.From.Reference
if rOpts.TargetPlatform != nil {
// resolve source reference to specified platform
desc, err := oras.Resolve(ctx, src, opts.From.Reference, rOpts)
if err != nil {
return fmt.Errorf("failed to resolve %s: %w", opts.From.Reference, err)
}
srcRef = desc.Digest.String()
if opts.To.Reference == "" {
desc, err = oras.Resolve(ctx, src, opts.From.Reference, rOpts)
if err != nil {
return fmt.Errorf("failed to resolve %s: %w", opts.From.Reference, err)
}
desc, err = oras.ExtendedCopy(ctx, src, srcRef, dst, dstRef, extendedCopyOptions)
err = oras.CopyGraph(ctx, src, dst, desc, extendedCopyOptions.CopyGraphOptions)
} else {
copyOptions := oras.CopyOptions{
CopyGraphOptions: extendedCopyOptions.CopyGraphOptions,
}
if opts.Platform.Platform != nil {
copyOptions.WithTargetPlatform(opts.Platform.Platform)
}
desc, err = oras.Copy(ctx, src, opts.From.Reference, dst, dstRef, copyOptions)
desc, err = oras.Copy(ctx, src, opts.From.Reference, dst, opts.To.Reference, copyOptions)
}
}
if err != nil {
Expand All @@ -191,3 +184,48 @@ func runCopy(ctx context.Context, opts copyOptions) error {

return nil
}

// recursiveCopy copies an artifact and its referrers from one target to another.
// If the artifact is a manifest list or index, referrers of its manifests are copied as well.
func recursiveCopy(ctx context.Context, src oras.ReadOnlyGraphTarget, dst oras.Target, dstRef string, root ocispec.Descriptor, opts oras.ExtendedCopyOptions) error {
var err error
if root.MediaType == ocispec.MediaTypeImageIndex || root.MediaType == docker.MediaTypeManifestList {
var fetched []byte
qweeah marked this conversation as resolved.
Show resolved Hide resolved
fetched, err := content.FetchAll(ctx, src, root)
if err != nil {
return err
}
var index ocispec.Index
if err = json.Unmarshal(fetched, &index); err != nil {
return nil
}

// point referrers of child manifests to root
findPredecessor := opts.FindPredecessors
var referrers []ocispec.Descriptor
for _, desc := range index.Manifests {
descs, err := findPredecessor(ctx, src, desc)
if err != nil {
return err
}
referrers = append(referrers, descs...)
}
qweeah marked this conversation as resolved.
Show resolved Hide resolved
opts.FindPredecessors = func(ctx context.Context, src content.ReadOnlyGraphStorage, desc ocispec.Descriptor) ([]ocispec.Descriptor, error) {
descs, err := findPredecessor(ctx, src, desc)
if err != nil {
return nil, err
}
if content.Equal(desc, root) {
descs = append(descs, referrers...)
}
return descs, nil
}
qweeah marked this conversation as resolved.
Show resolved Hide resolved
}

if dstRef == "" || dstRef == root.Digest.String() {
err = oras.ExtendedCopyGraph(ctx, src, dst, root, opts.ExtendedCopyGraphOptions)
} else {
_, err = oras.ExtendedCopy(ctx, src, root.Digest.String(), dst, dstRef, opts)
}
return err
}
3 changes: 2 additions & 1 deletion internal/docker/mediatype.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,5 +17,6 @@ package docker

// docker media types
const (
MediaTypeManifest = "application/vnd.docker.distribution.manifest.v2+json"
MediaTypeManifest = "application/vnd.docker.distribution.manifest.v2+json"
MediaTypeManifestList = "application/vnd.docker.distribution.manifest.list.v2+json"
)
30 changes: 13 additions & 17 deletions test/e2e/suite/command/cp.go
Original file line number Diff line number Diff line change
Expand Up @@ -115,15 +115,15 @@ var _ = Describe("1.1 registry users:", func() {
})

It("should copy an image and its referrers to a new repository", func() {
stateKeys := append(append(foobarStates, foobar.ImageReferrersStateKeys...), foobar.ImageReferrerConfigStateKeys...)
stateKeys := append(append(foobar.ImageLayerStateKeys, foobar.ManifestStateKey, foobar.ImageReferrerConfigStateKeys[0]), foobar.ImageReferrersStateKeys...)
src := RegistryRef(ZOTHost, ArtifactRepo, foobar.Tag)
dst := RegistryRef(ZOTHost, cpTestRepo("referrers"), foobar.Digest)
ORAS("cp", "-r", src, dst, "-v").MatchStatus(stateKeys, true, len(stateKeys)).Exec()
CompareRef(src, dst)
})

It("should copy a multi-arch image and its referrers to a new repository via tag", func() {
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.IndexReferrerConfigStateKey)
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.LinuxAMD64ReferrerConfigStateKey)
src := RegistryRef(ZOTHost, ArtifactRepo, ma.Tag)
dstRepo := cpTestRepo("index-referrers")
dst := RegistryRef(ZOTHost, dstRepo, "copiedTag")
Expand All @@ -142,13 +142,12 @@ var _ = Describe("1.1 registry users:", func() {
Expect(len(index.Manifests)).To(Equal(1))
Expect(index.Manifests[0].Digest.String()).To(Equal(ma.IndexReferrerDigest))
ORAS("manifest", "fetch", RegistryRef(ZOTHost, dstRepo, ma.LinuxAMD64Referrer.Digest.String())).
WithDescription("not copy referrer of successor").
ExpectFailure().
WithDescription("copy referrer of successor").
Exec()
})

It("should copy a multi-arch image and its referrers to a new repository via digest", func() {
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.IndexReferrerConfigStateKey)
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.LinuxAMD64ReferrerConfigStateKey)
src := RegistryRef(ZOTHost, ArtifactRepo, ma.Tag)
dstRepo := cpTestRepo("index-referrers-digest")
dst := RegistryRef(ZOTHost, dstRepo, ma.Digest)
Expand All @@ -168,7 +167,6 @@ var _ = Describe("1.1 registry users:", func() {
Expect(index.Manifests[0].Digest.String()).To(Equal(ma.IndexReferrerDigest))
ORAS("manifest", "fetch", RegistryRef(ZOTHost, dstRepo, ma.LinuxAMD64Referrer.Digest.String())).
WithDescription("not copy referrer of successor").
ExpectFailure().
Exec()
})

Expand Down Expand Up @@ -270,7 +268,7 @@ var _ = Describe("OCI spec 1.0 registry users:", func() {
When("running `cp`", func() {
It("should copy an image artifact and its referrers from a registry to a fallback registry", func() {
repo := cpTestRepo("to-fallback")
stateKeys := append(append(foobarStates, foobar.ImageReferrersStateKeys...), foobar.ImageReferrerConfigStateKeys...)
stateKeys := append(append(foobar.ImageLayerStateKeys, foobar.ManifestStateKey, foobar.ImageReferrerConfigStateKeys[0]), foobar.ImageReferrersStateKeys...)
src := RegistryRef(ZOTHost, ArtifactRepo, foobar.SignatureImageReferrer.Digest.String())
dst := RegistryRef(FallbackHost, repo, "")
ORAS("cp", "-r", src, dst, "-v").MatchStatus(stateKeys, true, len(stateKeys)).Exec()
Expand All @@ -280,7 +278,7 @@ var _ = Describe("OCI spec 1.0 registry users:", func() {
})
It("should copy an image artifact and its referrers from a fallback registry to a registry", func() {
repo := cpTestRepo("from-fallback")
stateKeys := append(append(foobarStates, foobar.ImageReferrersStateKeys...), foobar.ImageReferrerConfigStateKeys...)
stateKeys := append(append(foobar.ImageLayerStateKeys, foobar.ManifestStateKey, foobar.ImageReferrerConfigStateKeys[0]), foobar.ImageReferrersStateKeys...)
src := RegistryRef(FallbackHost, ArtifactRepo, foobar.SBOMImageReferrer.Digest.String())
dst := RegistryRef(ZOTHost, repo, "")
ORAS("cp", "-r", src, dst, "-v").MatchStatus(stateKeys, true, len(stateKeys)).Exec()
Expand Down Expand Up @@ -439,7 +437,7 @@ var _ = Describe("OCI layout users:", func() {
})

It("should copy a tagged image and its referrers from a registry to an OCI image layout", func() {
stateKeys := append(append(foobarStates, foobar.ImageReferrersStateKeys...), foobar.ImageReferrerConfigStateKeys...)
stateKeys := append(append(foobar.ImageLayerStateKeys, foobar.ManifestStateKey, foobar.ImageReferrerConfigStateKeys[0]), foobar.ImageReferrersStateKeys...)
dst := LayoutRef(GinkgoT().TempDir(), "copied")
src := RegistryRef(ZOTHost, ArtifactRepo, foobar.Tag)
// test
Expand All @@ -451,7 +449,7 @@ var _ = Describe("OCI layout users:", func() {
})

It("should copy a image and its referrers from a registry to an OCI image layout via digest", func() {
stateKeys := append(append(foobarStates, foobar.ImageReferrersStateKeys...), foobar.ImageReferrerConfigStateKeys...)
stateKeys := append(append(foobar.ImageLayerStateKeys, foobar.ManifestStateKey, foobar.ImageReferrerConfigStateKeys[0]), foobar.ImageReferrersStateKeys...)
toDir := GinkgoT().TempDir()
src := RegistryRef(ZOTHost, ArtifactRepo, foobar.Digest)
// test
Expand All @@ -463,7 +461,7 @@ var _ = Describe("OCI layout users:", func() {
})

It("should copy a multi-arch image and its referrers from a registry to an OCI image layout a via tag", func() {
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.IndexReferrerConfigStateKey)
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.LinuxAMD64ReferrerConfigStateKey)
src := RegistryRef(ZOTHost, ArtifactRepo, ma.Tag)
toDir := GinkgoT().TempDir()
dst := LayoutRef(toDir, "copied")
Expand All @@ -485,13 +483,12 @@ var _ = Describe("OCI layout users:", func() {
Expect(len(index.Manifests)).To(Equal(1))
Expect(index.Manifests[0].Digest.String()).To(Equal(ma.IndexReferrerDigest))
ORAS("manifest", "fetch", Flags.Layout, LayoutRef(toDir, ma.LinuxAMD64Referrer.Digest.String())).
WithDescription("not copy referrer of successor").
ExpectFailure().
WithDescription("copy referrer of successor").
Exec()
})

It("should copy a multi-arch image and its referrers from an OCI image layout to a registry via digest", func() {
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.IndexReferrerConfigStateKey)
stateKeys := append(ma.IndexStateKeys, ma.IndexZOTReferrerStateKey, ma.LinuxAMD64ReferrerConfigStateKey)
fromDir := GinkgoT().TempDir()
src := LayoutRef(fromDir, ma.Tag)
dst := RegistryRef(ZOTHost, cpTestRepo("recursive-from-layout"), "copied")
Expand All @@ -514,9 +511,8 @@ var _ = Describe("OCI layout users:", func() {
Expect(json.Unmarshal(bytes, &index)).ShouldNot(HaveOccurred())
Expect(len(index.Manifests)).To(Equal(1))
Expect(index.Manifests[0].Digest.String()).To(Equal(ma.IndexReferrerDigest))
ORAS("manifest", "fetch", LayoutRef(fromDir, ma.LinuxAMD64Referrer.Digest.String())).
WithDescription("not copy referrer of successor").
ExpectFailure().
ORAS("manifest", "fetch", dst).
WithDescription("copy referrer of successor").
Exec()
})

Expand Down
2 changes: 1 addition & 1 deletion test/e2e/suite/command/pull.go
Original file line number Diff line number Diff line change
Expand Up @@ -122,7 +122,7 @@ var _ = Describe("OCI spec 1.1 registry users:", func() {

It("should copy an artifact with blob", func() {
repo := cpTestRepo("artifact-with-blob")
stateKeys := append(append(foobarStates, foobar.ImageReferrersStateKeys...), foobar.ImageReferrerConfigStateKeys...)
stateKeys := append(append(foobar.ImageLayerStateKeys, foobar.ManifestStateKey, foobar.ImageReferrerConfigStateKeys[0]), foobar.ImageReferrersStateKeys...)
src := RegistryRef(ZOTHost, ArtifactRepo, foobar.SignatureImageReferrer.Digest.String())
dst := RegistryRef(FallbackHost, repo, "")
ORAS("cp", "-r", src, dst, "-v").MatchStatus(stateKeys, true, len(stateKeys)).Exec()
Expand Down