Skip to content

Commit

Permalink
Merge pull request containerd#9784 from henry118/release/1.7
Browse files Browse the repository at this point in the history
[release/1.7] bug fix: make sure cri image is pinned when it is pulled outside cri
  • Loading branch information
estesp authored Feb 9, 2024
2 parents 2d4823c + 26c0574 commit 8e86a0b
Show file tree
Hide file tree
Showing 3 changed files with 65 additions and 8 deletions.
29 changes: 29 additions & 0 deletions integration/containerd_image_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -237,6 +237,35 @@ func TestContainerdSandboxImage(t *testing.T) {
assert.True(t, pimg.Pinned)
}

func TestContainerdSandboxImagePulledOutsideCRI(t *testing.T) {
var pauseImage = images.Get(images.Pause)
ctx := context.Background()

t.Log("make sure the pause image does not exist")
imageService.RemoveImage(&runtime.ImageSpec{Image: pauseImage})

t.Log("pull pause image")
_, err := containerdClient.Pull(ctx, pauseImage)
assert.NoError(t, err)

t.Log("pause image should be seen by cri plugin")
var pimg *runtime.Image
require.NoError(t, Eventually(func() (bool, error) {
pimg, err = imageService.ImageStatus(&runtime.ImageSpec{Image: pauseImage})
return pimg != nil, err
}, time.Second, 10*time.Second))

t.Log("verify pinned field is set for pause image")
assert.True(t, pimg.Pinned)

t.Log("make sure the pause image exist")
pauseImg, err := containerdClient.GetImage(ctx, pauseImage)
require.NoError(t, err)

t.Log("ensure correct labels are set on pause image")
assert.Equal(t, "pinned", pauseImg.Labels()["io.cri-containerd.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.")
Expand Down
22 changes: 18 additions & 4 deletions pkg/cri/sbserver/image_pull.go
Original file line number Diff line number Diff line change
Expand Up @@ -279,14 +279,28 @@ func (c *criService) createImageReference(ctx context.Context, name string, desc
}
// TODO(random-liu): Figure out which is the more performant sequence create then update or
// update then create.
oldImg, err := c.client.ImageService().Create(ctx, img)
_, err := c.client.ImageService().Create(ctx, img)
if err == nil || !errdefs.IsAlreadyExists(err) {
return err
}
if oldImg.Target.Digest == img.Target.Digest && oldImg.Labels[crilabels.ImageLabelKey] == labels[crilabels.ImageLabelKey] {
// Retrieve oldImg from image store here because Create routine returns an
// empty image on ErrAlreadyExists
oldImg, err := c.client.ImageService().Get(ctx, name)
if err != nil {
return err
}
fieldpaths := []string{"target"}
if oldImg.Labels[crilabels.ImageLabelKey] != labels[crilabels.ImageLabelKey] {
fieldpaths = append(fieldpaths, "labels."+crilabels.ImageLabelKey)
}
if oldImg.Labels[crilabels.PinnedImageLabelKey] != labels[crilabels.PinnedImageLabelKey] &&
labels[crilabels.PinnedImageLabelKey] == crilabels.PinnedImageLabelValue {
fieldpaths = append(fieldpaths, "labels."+crilabels.PinnedImageLabelKey)
}
if oldImg.Target.Digest == img.Target.Digest && len(fieldpaths) < 2 {
return nil
}
_, err = c.client.ImageService().Update(ctx, img, "target", "labels."+crilabels.ImageLabelKey)
_, err = c.client.ImageService().Update(ctx, img, fieldpaths...)
return err
}

Expand Down Expand Up @@ -324,7 +338,7 @@ func (c *criService) updateImage(ctx context.Context, r string) error {
return fmt.Errorf("get image id: %w", err)
}
id := configDesc.Digest.String()
labels := c.getLabels(ctx, id)
labels := c.getLabels(ctx, r)
if err := c.createImageReference(ctx, id, img.Target(), labels); err != nil {
return fmt.Errorf("create image id reference %q: %w", id, err)
}
Expand Down
22 changes: 18 additions & 4 deletions pkg/cri/server/image_pull.go
Original file line number Diff line number Diff line change
Expand Up @@ -281,14 +281,28 @@ func (c *criService) createImageReference(ctx context.Context, name string, desc
}
// TODO(random-liu): Figure out which is the more performant sequence create then update or
// update then create.
oldImg, err := c.client.ImageService().Create(ctx, img)
_, err := c.client.ImageService().Create(ctx, img)
if err == nil || !errdefs.IsAlreadyExists(err) {
return err
}
if oldImg.Target.Digest == img.Target.Digest && oldImg.Labels[crilabels.ImageLabelKey] == labels[crilabels.ImageLabelKey] {
// Retrieve oldImg from image store here because Create routine returns an
// empty image on ErrAlreadyExists
oldImg, err := c.client.ImageService().Get(ctx, name)
if err != nil {
return err
}
fieldpaths := []string{"target"}
if oldImg.Labels[crilabels.ImageLabelKey] != labels[crilabels.ImageLabelKey] {
fieldpaths = append(fieldpaths, "labels."+crilabels.ImageLabelKey)
}
if oldImg.Labels[crilabels.PinnedImageLabelKey] != labels[crilabels.PinnedImageLabelKey] &&
labels[crilabels.PinnedImageLabelKey] == crilabels.PinnedImageLabelValue {
fieldpaths = append(fieldpaths, "labels."+crilabels.PinnedImageLabelKey)
}
if oldImg.Target.Digest == img.Target.Digest && len(fieldpaths) < 2 {
return nil
}
_, err = c.client.ImageService().Update(ctx, img, "target", "labels."+crilabels.ImageLabelKey)
_, err = c.client.ImageService().Update(ctx, img, fieldpaths...)
return err
}

Expand Down Expand Up @@ -326,7 +340,7 @@ func (c *criService) updateImage(ctx context.Context, r string) error {
return fmt.Errorf("get image id: %w", err)
}
id := configDesc.Digest.String()
labels := c.getLabels(ctx, id)
labels := c.getLabels(ctx, r)
if err := c.createImageReference(ctx, id, img.Target(), labels); err != nil {
return fmt.Errorf("create image id reference %q: %w", id, err)
}
Expand Down

0 comments on commit 8e86a0b

Please sign in to comment.