Skip to content

Commit

Permalink
Add a new image label if it is docker schema 1
Browse files Browse the repository at this point in the history
Signed-off-by: Qiutong Song <[email protected]>
(cherry picked from commit 7712375)
Signed-off-by: Qiutong Song <[email protected]>
  • Loading branch information
qiutongs committed Oct 17, 2023
1 parent 9211aff commit cac1bab
Show file tree
Hide file tree
Showing 3 changed files with 60 additions and 4 deletions.
40 changes: 40 additions & 0 deletions integration/containerd_image_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,8 @@ import (
"context"
"errors"
"fmt"
goruntime "runtime"
"strings"
"testing"
"time"

Expand Down Expand Up @@ -225,3 +227,41 @@ func TestContainerdSandboxImage(t *testing.T) {
t.Log("verify pinned field is set for pause image")
assert.True(t, pimg.Pinned)
}

func TestContainerdImageWithDockerSchema1(t *testing.T) {
if goruntime.GOOS == "windows" {
t.Skip("Skipped on Windows because the test image is not a multi-platform one.")
}

var testImage = images.Get(images.DockerSchema1)
digest := strings.Split(testImage, "@")[1]
ctx := context.Background()

t.Logf("make sure the test image doesn't exist in the cri plugin")
i, err := imageService.ImageStatus(&runtime.ImageSpec{Image: testImage})
require.NoError(t, err)
if i != nil {
require.NoError(t, imageService.RemoveImage(&runtime.ImageSpec{Image: testImage}))
}

t.Logf("pull the image into containerd")
//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
_, err = containerdClient.Pull(ctx, testImage, containerd.WithPullUnpack, containerd.WithSchema1Conversion)
require.NoError(t, err)
defer func() {
// Make sure the image is cleaned up in any case.
if err := containerdClient.ImageService().Delete(ctx, testImage); err != nil {
assert.True(t, errdefs.IsNotFound(err), err)
}
assert.NoError(t, imageService.RemoveImage(&runtime.ImageSpec{Image: testImage}))
}()

imgByRef, err := containerdClient.GetImage(ctx, testImage)
require.NoError(t, err)

t.Logf("the image should be marked as managed")
assert.Equal(t, "managed", imgByRef.Labels()["io.cri-containerd.image"])

t.Logf("the image should be marked as dokcker schema1 with its original digest")
assert.Equal(t, digest, imgByRef.Labels()["io.containerd.image/converted-docker-schema1"])
}
5 changes: 5 additions & 0 deletions integration/images/image_list.go
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ type ImageList struct {
VolumeCopyUp string
VolumeOwnership string
ArgsEscaped string
DockerSchema1 string
}

var (
Expand All @@ -55,6 +56,7 @@ func initImages(imageListFile string) {
VolumeCopyUp: "ghcr.io/containerd/volume-copy-up:2.1",
VolumeOwnership: "ghcr.io/containerd/volume-ownership:2.1",
ArgsEscaped: "cplatpublic.azurecr.io/args-escaped-test-image-ns:1.0",
DockerSchema1: "registry.k8s.io/busybox@sha256:4bdd623e848417d96127e16037743f0cd8b528c026e9175e22a84f639eca58ff",
}

if imageListFile != "" {
Expand Down Expand Up @@ -92,6 +94,8 @@ const (
VolumeOwnership
// Test image for ArgsEscaped windows bug
ArgsEscaped
// DockerSchema1 image with docker schema 1
DockerSchema1
)

func initImageMap(imageList ImageList) map[int]string {
Expand All @@ -103,6 +107,7 @@ func initImageMap(imageList ImageList) map[int]string {
images[VolumeCopyUp] = imageList.VolumeCopyUp
images[VolumeOwnership] = imageList.VolumeOwnership
images[ArgsEscaped] = imageList.ArgsEscaped
images[DockerSchema1] = imageList.DockerSchema1
return images
}

Expand Down
19 changes: 15 additions & 4 deletions pull.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,8 @@ import (
)

const (
pullSpanPrefix = "pull"
pullSpanPrefix = "pull"
convertedDockerSchema1LabelKey = "io.containerd.image/converted-docker-schema1"
)

// Pull downloads the provided content into containerd's content store
Expand Down Expand Up @@ -189,9 +190,10 @@ func (c *Client) fetch(ctx context.Context, rCtx *RemoteContext, ref string, lim
var (
handler images.Handler

isConvertible bool
converterFunc func(context.Context, ocispec.Descriptor) (ocispec.Descriptor, error)
limiter *semaphore.Weighted
isConvertible bool
originalSchema1Digest string
converterFunc func(context.Context, ocispec.Descriptor) (ocispec.Descriptor, error)
limiter *semaphore.Weighted
)

if desc.MediaType == images.MediaTypeDockerSchema1Manifest && rCtx.ConvertSchema1 {
Expand All @@ -204,6 +206,8 @@ func (c *Client) fetch(ctx context.Context, rCtx *RemoteContext, ref string, lim
converterFunc = func(ctx context.Context, _ ocispec.Descriptor) (ocispec.Descriptor, error) {
return schema1Converter.Convert(ctx)
}

originalSchema1Digest = desc.Digest.String()
} else {
// Get all the children for a descriptor
childrenHandler := images.ChildrenHandler(store)
Expand Down Expand Up @@ -270,6 +274,13 @@ func (c *Client) fetch(ctx context.Context, rCtx *RemoteContext, ref string, lim
}
}

if originalSchema1Digest != "" {
if rCtx.Labels == nil {
rCtx.Labels = make(map[string]string)
}
rCtx.Labels[convertedDockerSchema1LabelKey] = originalSchema1Digest
}

return images.Image{
Name: name,
Target: desc,
Expand Down

0 comments on commit cac1bab

Please sign in to comment.