image_tag.go 3.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. package containerd
  2. import (
  3. "context"
  4. cerrdefs "github.com/containerd/containerd/errdefs"
  5. containerdimages "github.com/containerd/containerd/images"
  6. "github.com/containerd/log"
  7. "github.com/distribution/reference"
  8. "github.com/docker/docker/api/types/events"
  9. "github.com/docker/docker/errdefs"
  10. "github.com/docker/docker/image"
  11. "github.com/docker/docker/internal/compatcontext"
  12. "github.com/pkg/errors"
  13. )
  14. // TagImage creates an image named as newTag and targeting the given descriptor id.
  15. func (i *ImageService) TagImage(ctx context.Context, imageID image.ID, newTag reference.Named) error {
  16. targetImage, err := i.resolveImage(ctx, imageID.String())
  17. if err != nil {
  18. return errors.Wrapf(err, "failed to resolve image id %q to a descriptor", imageID.String())
  19. }
  20. newImg := containerdimages.Image{
  21. Name: newTag.String(),
  22. Target: targetImage.Target,
  23. Labels: targetImage.Labels,
  24. }
  25. is := i.client.ImageService()
  26. _, err = is.Create(ctx, newImg)
  27. if err != nil {
  28. if !cerrdefs.IsAlreadyExists(err) {
  29. return errdefs.System(errors.Wrapf(err, "failed to create image with name %s and target %s", newImg.Name, newImg.Target.Digest.String()))
  30. }
  31. replacedImg, err := is.Get(ctx, newImg.Name)
  32. if err != nil {
  33. return errdefs.Unknown(errors.Wrapf(err, "creating image %s failed because it already exists, but accessing it also failed", newImg.Name))
  34. }
  35. // Check if image we would replace already resolves to the same target.
  36. // No need to do anything.
  37. if replacedImg.Target.Digest == targetImage.Target.Digest {
  38. i.LogImageEvent(imageID.String(), reference.FamiliarString(newTag), events.ActionTag)
  39. return nil
  40. }
  41. // If there already exists an image with this tag, delete it
  42. if err := i.softImageDelete(ctx, replacedImg); err != nil {
  43. return errors.Wrapf(err, "failed to delete previous image %s", replacedImg.Name)
  44. }
  45. if _, err = is.Create(compatcontext.WithoutCancel(ctx), newImg); err != nil {
  46. return errdefs.System(errors.Wrapf(err, "failed to create an image %s with target %s after deleting the existing one",
  47. newImg.Name, imageID.String()))
  48. }
  49. }
  50. logger := log.G(ctx).WithFields(log.Fields{
  51. "imageID": imageID.String(),
  52. "tag": newTag.String(),
  53. })
  54. logger.Info("image created")
  55. defer i.LogImageEvent(imageID.String(), reference.FamiliarString(newTag), events.ActionTag)
  56. // The tag succeeded, check if the source image is dangling
  57. sourceDanglingImg, err := is.Get(compatcontext.WithoutCancel(ctx), danglingImageName(targetImage.Target.Digest))
  58. if err != nil {
  59. if !cerrdefs.IsNotFound(err) {
  60. logger.WithError(err).Warn("unexpected error when checking if source image is dangling")
  61. }
  62. return nil
  63. }
  64. builderLabel, ok := sourceDanglingImg.Labels[imageLabelClassicBuilderParent]
  65. if ok {
  66. newImg.Labels = map[string]string{
  67. imageLabelClassicBuilderParent: builderLabel,
  68. }
  69. if _, err := is.Update(compatcontext.WithoutCancel(ctx), newImg, "labels"); err != nil {
  70. logger.WithError(err).Warnf("failed to set %s label on the newly tagged image", imageLabelClassicBuilderParent)
  71. }
  72. }
  73. // Delete the source dangling image, as it's no longer dangling.
  74. if err := is.Delete(compatcontext.WithoutCancel(ctx), sourceDanglingImg.Name); err != nil {
  75. logger.WithError(err).Warn("unexpected error when deleting dangling image")
  76. }
  77. return nil
  78. }