From 575fdb8345030749068b55229f1454bb0548b7a1 Mon Sep 17 00:00:00 2001 From: Petu Eusebiu Date: Thu, 12 Oct 2023 16:51:59 +0300 Subject: [PATCH] feat(api): repair corrupted blobs when pushed again CheckBlob() returns ErrBlobNotFound on corrupted blobs closes #1922 Signed-off-by: Petu Eusebiu --- pkg/storage/common/common.go | 88 ++++++++ pkg/storage/common/common_test.go | 82 +++++++ pkg/storage/imagestore/imagestore.go | 69 ++++-- pkg/storage/local/local_test.go | 6 +- pkg/storage/s3/s3_test.go | 44 ++-- pkg/storage/storage_test.go | 213 ++++++++++++++++++ pkg/storage/types/types.go | 2 +- pkg/test/mocks/image_store_mock.go | 6 +- test/gc-stress/config-gc-bench-s3-minio.json | 4 +- .../config-gc-referrers-bench-s3-minio.json | 4 +- 10 files changed, 467 insertions(+), 51 deletions(-) diff --git a/pkg/storage/common/common.go b/pkg/storage/common/common.go index 533c078f5..8573b4a6b 100644 --- a/pkg/storage/common/common.go +++ b/pkg/storage/common/common.go @@ -809,6 +809,94 @@ func GetOrasManifestByDigest(imgStore storageTypes.ImageStore, repo string, dige return artManifest, nil } +// Get blob descriptor from it's manifest contents, if blob can not be found it will return error. +func GetBlobDescriptorFromRepo(imgStore storageTypes.ImageStore, repo string, blobDigest godigest.Digest, + log zlog.Logger, +) (ispec.Descriptor, error) { + index, err := GetIndex(imgStore, repo, log) + if err != nil { + return ispec.Descriptor{}, err + } + + return GetBlobDescriptorFromIndex(imgStore, index, repo, blobDigest, log) +} + +func GetBlobDescriptorFromIndex(imgStore storageTypes.ImageStore, index ispec.Index, repo string, + blobDigest godigest.Digest, log zlog.Logger, +) (ispec.Descriptor, error) { + for _, desc := range index.Manifests { + if desc.Digest == blobDigest { + return desc, nil + } + + switch desc.MediaType { + case ispec.MediaTypeImageManifest: + if foundDescriptor, err := getBlobDescriptorFromManifest(imgStore, repo, blobDigest, desc, log); err == nil { + return foundDescriptor, nil + } + case ispec.MediaTypeImageIndex: + indexImage, err := GetImageIndex(imgStore, repo, desc.Digest, log) + if err != nil { + return ispec.Descriptor{}, err + } + + if foundDescriptor, err := GetBlobDescriptorFromIndex(imgStore, indexImage, repo, blobDigest, log); err == nil { + return foundDescriptor, nil + } + case oras.MediaTypeArtifactManifest: + if foundDescriptor, err := getBlobDescriptorFromORASManifest(imgStore, repo, blobDigest, desc, log); err == nil { + return foundDescriptor, nil + } + } + } + + return ispec.Descriptor{}, zerr.ErrBlobNotFound +} + +func getBlobDescriptorFromManifest(imgStore storageTypes.ImageStore, repo string, blobDigest godigest.Digest, + desc ispec.Descriptor, log zlog.Logger, +) (ispec.Descriptor, error) { + manifest, err := GetImageManifest(imgStore, repo, desc.Digest, log) + if err != nil { + return ispec.Descriptor{}, err + } + + if manifest.Config.Digest == blobDigest { + return manifest.Config, nil + } + + for _, layer := range manifest.Layers { + if layer.Digest == blobDigest { + return layer, nil + } + } + + return ispec.Descriptor{}, zerr.ErrBlobNotFound +} + +func getBlobDescriptorFromORASManifest(imgStore storageTypes.ImageStore, repo string, blobDigest godigest.Digest, + desc ispec.Descriptor, log zlog.Logger, +) (ispec.Descriptor, error) { + manifest, err := GetOrasManifestByDigest(imgStore, repo, desc.Digest, log) + if err != nil { + return ispec.Descriptor{}, err + } + + for _, layer := range manifest.Blobs { + if layer.Digest == blobDigest { + return ispec.Descriptor{ + MediaType: layer.MediaType, + Size: layer.Size, + Digest: layer.Digest, + ArtifactType: layer.ArtifactType, + Annotations: layer.Annotations, + }, nil + } + } + + return ispec.Descriptor{}, zerr.ErrBlobNotFound +} + func IsSupportedMediaType(mediaType string) bool { return mediaType == ispec.MediaTypeImageIndex || mediaType == ispec.MediaTypeImageManifest || diff --git a/pkg/storage/common/common_test.go b/pkg/storage/common/common_test.go index 9a113fcf2..a593005f5 100644 --- a/pkg/storage/common/common_test.go +++ b/pkg/storage/common/common_test.go @@ -19,6 +19,7 @@ import ( "zotregistry.io/zot/pkg/storage" "zotregistry.io/zot/pkg/storage/cache" common "zotregistry.io/zot/pkg/storage/common" + "zotregistry.io/zot/pkg/storage/imagestore" "zotregistry.io/zot/pkg/storage/local" . "zotregistry.io/zot/pkg/test/image-utils" "zotregistry.io/zot/pkg/test/mocks" @@ -420,6 +421,87 @@ func TestGetImageIndexErrors(t *testing.T) { }) } +func TestGetBlobDescriptorFromRepo(t *testing.T) { + log := log.Logger{Logger: zerolog.New(os.Stdout)} + metrics := monitoring.NewMetricsServer(false, log) + + tdir := t.TempDir() + cacheDriver, _ := storage.Create("boltdb", cache.BoltDBDriverParameters{ + RootDir: tdir, + Name: "cache", + UseRelPaths: true, + }, log) + + driver := local.New(true) + imgStore := imagestore.NewImageStore(tdir, tdir, true, + true, log, metrics, nil, driver, cacheDriver) + + repoName := "zot-test" + + Convey("Test error paths", t, func() { + storeController := storage.StoreController{DefaultStore: imgStore} + + image := CreateRandomMultiarch() + + tag := "index" + + err := WriteMultiArchImageToFileSystem(image, repoName, tag, storeController) + So(err, ShouldBeNil) + + blob := image.Images[0].Layers[0] + blobDigest := godigest.FromBytes(blob) + blobSize := len(blob) + + desc, err := common.GetBlobDescriptorFromIndex(imgStore, ispec.Index{Manifests: []ispec.Descriptor{ + { + Digest: image.Digest(), + MediaType: ispec.MediaTypeImageIndex, + }, + }}, repoName, blobDigest, log) + So(err, ShouldBeNil) + So(desc.Digest, ShouldEqual, blobDigest) + So(desc.Size, ShouldEqual, blobSize) + + desc, err = common.GetBlobDescriptorFromRepo(imgStore, repoName, blobDigest, log) + So(err, ShouldBeNil) + So(desc.Digest, ShouldEqual, blobDigest) + So(desc.Size, ShouldEqual, blobSize) + + indexBlobPath := imgStore.BlobPath(repoName, image.Digest()) + err = os.Chmod(indexBlobPath, 0o000) + So(err, ShouldBeNil) + + defer func() { + err = os.Chmod(indexBlobPath, 0o644) + So(err, ShouldBeNil) + }() + + _, err = common.GetBlobDescriptorFromIndex(imgStore, ispec.Index{Manifests: []ispec.Descriptor{ + { + Digest: image.Digest(), + MediaType: ispec.MediaTypeImageIndex, + }, + }}, repoName, blobDigest, log) + So(err, ShouldNotBeNil) + + manifestDigest := image.Images[0].Digest() + manifestBlobPath := imgStore.BlobPath(repoName, manifestDigest) + err = os.Chmod(manifestBlobPath, 0o000) + So(err, ShouldBeNil) + + defer func() { + err = os.Chmod(manifestBlobPath, 0o644) + So(err, ShouldBeNil) + }() + + _, err = common.GetBlobDescriptorFromRepo(imgStore, repoName, blobDigest, log) + So(err, ShouldNotBeNil) + + _, err = common.GetBlobDescriptorFromRepo(imgStore, repoName, manifestDigest, log) + So(err, ShouldBeNil) + }) +} + func TestIsSignature(t *testing.T) { Convey("Unknown media type", t, func(c C) { isSingature := common.IsSignature(ispec.Descriptor{ diff --git a/pkg/storage/imagestore/imagestore.go b/pkg/storage/imagestore/imagestore.go index 3ef39c7d5..2508a59fd 100644 --- a/pkg/storage/imagestore/imagestore.go +++ b/pkg/storage/imagestore/imagestore.go @@ -866,20 +866,11 @@ func (is *ImageStore) FinishBlobUpload(repo, uuid string, body io.Reader, dstDig return err } - fileReader, err := is.storeDriver.Reader(src, 0) - if err != nil { - is.log.Error().Err(err).Str("blob", src).Msg("failed to open file") - - return zerr.ErrUploadNotFound - } - - defer fileReader.Close() - - srcDigest, err := godigest.FromReader(fileReader) + srcDigest, err := getBlobDigest(is, src) if err != nil { is.log.Error().Err(err).Str("blob", src).Msg("failed to open blob") - return zerr.ErrBadBlobDigest + return err } if srcDigest != dstDigest { @@ -906,7 +897,7 @@ func (is *ImageStore) FinishBlobUpload(repo, uuid string, body io.Reader, dstDig defer is.Unlock(&lockLatency) if is.dedupe && fmt.Sprintf("%v", is.cache) != fmt.Sprintf("%v", nil) { - err = is.DedupeBlob(src, dstDigest, dst) + err = is.DedupeBlob(src, dstDigest, repo, dst) if err := inject.Error(err); err != nil { is.log.Error().Err(err).Str("src", src).Str("dstDigest", dstDigest.String()). Str("dst", dst).Msg("unable to dedupe blob") @@ -985,7 +976,7 @@ func (is *ImageStore) FullBlobUpload(repo string, body io.Reader, dstDigest godi dst := is.BlobPath(repo, dstDigest) if is.dedupe && fmt.Sprintf("%v", is.cache) != fmt.Sprintf("%v", nil) { - if err := is.DedupeBlob(src, dstDigest, dst); err != nil { + if err := is.DedupeBlob(src, dstDigest, repo, dst); err != nil { is.log.Error().Err(err).Str("src", src).Str("dstDigest", dstDigest.String()). Str("dst", dst).Msg("unable to dedupe blob") @@ -1003,7 +994,7 @@ func (is *ImageStore) FullBlobUpload(repo string, body io.Reader, dstDigest godi return uuid, int64(nbytes), nil } -func (is *ImageStore) DedupeBlob(src string, dstDigest godigest.Digest, dst string) error { +func (is *ImageStore) DedupeBlob(src string, dstDigest godigest.Digest, dstRepo string, dst string) error { retry: is.log.Debug().Str("src", src).Str("dstDigest", dstDigest.String()).Str("dst", dst).Msg("dedupe: enter") @@ -1033,12 +1024,11 @@ retry: } else { // cache record exists, but due to GC and upgrades from older versions, // disk content and cache records may go out of sync - if is.cache.UsesRelativePaths() { dstRecord = path.Join(is.rootDir, dstRecord) } - _, err := is.storeDriver.Stat(dstRecord) + blobInfo, err := is.storeDriver.Stat(dstRecord) if err != nil { is.log.Error().Err(err).Str("blobPath", dstRecord).Msg("dedupe: unable to stat") // the actual blob on disk may have been removed by GC, so sync the cache @@ -1066,6 +1056,22 @@ retry: return err } + } else { + // if it's same file then it was already uploaded, check if blob is corrupted + if desc, err := common.GetBlobDescriptorFromRepo(is, dstRepo, dstDigest, is.log); err == nil { + // blob corrupted, replace content + if desc.Size != blobInfo.Size() { + if err := is.storeDriver.Move(src, dst); err != nil { + is.log.Error().Err(err).Str("src", src).Str("dst", dst).Msg("dedupe: unable to rename blob") + + return err + } + + is.log.Debug().Str("src", src).Msg("dedupe: remove") + + return nil + } + } } // remove temp blobupload @@ -1134,9 +1140,20 @@ func (is *ImageStore) CheckBlob(repo string, digest godigest.Digest) (bool, int6 binfo, err := is.storeDriver.Stat(blobPath) if err == nil && binfo.Size() > 0 { - is.log.Debug().Str("blob path", blobPath).Msg("blob path found") + // try to find blob size in blob descriptors, if blob can not be found + desc, err := common.GetBlobDescriptorFromRepo(is, repo, digest, is.log) + if err != nil || desc.Size == binfo.Size() { + // blob not found in descriptors, can not compare, just return + is.log.Debug().Str("blob path", blobPath).Msg("blob path found") + + return true, binfo.Size(), nil //nolint: nilerr + } - return true, binfo.Size(), nil + if desc.Size != binfo.Size() { + is.log.Debug().Str("blob path", blobPath).Msg("blob path found, but it's corrupted") + + return false, -1, zerr.ErrBlobNotFound + } } // otherwise is a 'deduped' blob (empty file) @@ -1634,6 +1651,22 @@ func (is *ImageStore) deleteBlob(repo string, digest godigest.Digest) error { return nil } +func getBlobDigest(imgStore *ImageStore, path string) (godigest.Digest, error) { + fileReader, err := imgStore.storeDriver.Reader(path, 0) + if err != nil { + return "", zerr.ErrUploadNotFound + } + + defer fileReader.Close() + + digest, err := godigest.FromReader(fileReader) + if err != nil { + return "", zerr.ErrBadBlobDigest + } + + return digest, nil +} + func (is *ImageStore) GetAllBlobs(repo string) ([]string, error) { dir := path.Join(is.rootDir, repo, "blobs", "sha256") diff --git a/pkg/storage/local/local_test.go b/pkg/storage/local/local_test.go index 5ee8c3017..2715a29c8 100644 --- a/pkg/storage/local/local_test.go +++ b/pkg/storage/local/local_test.go @@ -818,7 +818,7 @@ func FuzzDedupeBlob(f *testing.F) { t.Error(err) } - err = imgStore.DedupeBlob(src, blobDigest, dst) + err = imgStore.DedupeBlob(src, blobDigest, "repoName", dst) if err != nil { t.Error(err) } @@ -1514,7 +1514,7 @@ func TestDedupe(t *testing.T) { Convey("Dedupe", t, func(c C) { Convey("Nil ImageStore", func() { var is storageTypes.ImageStore - So(func() { _ = is.DedupeBlob("", "", "") }, ShouldPanic) + So(func() { _ = is.DedupeBlob("", "", "", "") }, ShouldPanic) }) Convey("Valid ImageStore", func() { @@ -1530,7 +1530,7 @@ func TestDedupe(t *testing.T) { il := local.NewImageStore(dir, true, true, log, metrics, nil, cacheDriver) - So(il.DedupeBlob("", "", ""), ShouldNotBeNil) + So(il.DedupeBlob("", "", "", ""), ShouldNotBeNil) }) }) } diff --git a/pkg/storage/s3/s3_test.go b/pkg/storage/s3/s3_test.go index 381e50989..6ba070b26 100644 --- a/pkg/storage/s3/s3_test.go +++ b/pkg/storage/s3/s3_test.go @@ -3552,7 +3552,7 @@ func TestS3DedupeErr(t *testing.T) { digest := godigest.NewDigestFromEncoded(godigest.SHA256, "digest") // trigger unable to insert blob record - err := imgStore.DedupeBlob("", digest, "") + err := imgStore.DedupeBlob("", digest, "", "") So(err, ShouldNotBeNil) imgStore = createMockStorage(testDir, tdir, true, &StorageDriverMock{ @@ -3565,11 +3565,11 @@ func TestS3DedupeErr(t *testing.T) { }) // trigger unable to rename blob - err = imgStore.DedupeBlob("", digest, "dst") + err = imgStore.DedupeBlob("", digest, "", "dst") So(err, ShouldNotBeNil) // trigger retry - err = imgStore.DedupeBlob("", digest, "dst") + err = imgStore.DedupeBlob("", digest, "", "dst") So(err, ShouldNotBeNil) }) @@ -3586,11 +3586,11 @@ func TestS3DedupeErr(t *testing.T) { }) digest := godigest.NewDigestFromEncoded(godigest.SHA256, "digest") - err := imgStore.DedupeBlob("", digest, "dst") + err := imgStore.DedupeBlob("", digest, "", "dst") So(err, ShouldBeNil) // error will be triggered in driver.SameFile() - err = imgStore.DedupeBlob("", digest, "dst2") + err = imgStore.DedupeBlob("", digest, "", "dst2") So(err, ShouldBeNil) }) @@ -3606,10 +3606,10 @@ func TestS3DedupeErr(t *testing.T) { }) digest := godigest.NewDigestFromEncoded(godigest.SHA256, "digest") - err := imgStore.DedupeBlob("", digest, "dst") + err := imgStore.DedupeBlob("", digest, "", "dst") So(err, ShouldBeNil) - err = imgStore.DedupeBlob("", digest, "dst2") + err = imgStore.DedupeBlob("", digest, "", "dst2") So(err, ShouldNotBeNil) }) @@ -3622,10 +3622,10 @@ func TestS3DedupeErr(t *testing.T) { }) digest := godigest.NewDigestFromEncoded(godigest.SHA256, "digest") - err := imgStore.DedupeBlob("", digest, "dst") + err := imgStore.DedupeBlob("", digest, "", "dst") So(err, ShouldBeNil) - err = imgStore.DedupeBlob("", digest, "") + err = imgStore.DedupeBlob("", digest, "", "") So(err, ShouldNotBeNil) }) @@ -3641,10 +3641,10 @@ func TestS3DedupeErr(t *testing.T) { }) digest := godigest.NewDigestFromEncoded(godigest.SHA256, "digest") - err := imgStore.DedupeBlob("", digest, "dst") + err := imgStore.DedupeBlob("", digest, "", "dst") So(err, ShouldBeNil) - err = imgStore.DedupeBlob("", digest, "dst") + err = imgStore.DedupeBlob("", digest, "", "dst") So(err, ShouldNotBeNil) }) @@ -3665,7 +3665,7 @@ func TestS3DedupeErr(t *testing.T) { digest := godigest.NewDigestFromEncoded(godigest.SHA256, "7173b809ca12ec5dee4506cd86be934c4596dd234ee82c0662eac04a8c2c71dc") - err := imgStore.DedupeBlob("repo", digest, "dst") + err := imgStore.DedupeBlob("repo", digest, "", "dst") So(err, ShouldBeNil) _, _, err = imgStore.CheckBlob("repo", digest) @@ -3686,7 +3686,7 @@ func TestS3DedupeErr(t *testing.T) { digest := godigest.NewDigestFromEncoded(godigest.SHA256, "7173b809ca12ec5dee4506cd86be934c4596dd234ee82c0662eac04a8c2c71dc") - err := imgStore.DedupeBlob("repo", digest, "dst") + err := imgStore.DedupeBlob("repo", digest, "", "dst") So(err, ShouldBeNil) _, _, err = imgStore.CheckBlob("repo", digest) @@ -3704,7 +3704,7 @@ func TestS3DedupeErr(t *testing.T) { digest := godigest.NewDigestFromEncoded(godigest.SHA256, "7173b809ca12ec5dee4506cd86be934c4596dd234ee82c0662eac04a8c2c71dc") - err := imgStore.DedupeBlob("repo", digest, "dst") + err := imgStore.DedupeBlob("repo", digest, "", "dst") So(err, ShouldBeNil) _, _, err = imgStore.CheckBlob("repo", digest) @@ -3719,10 +3719,10 @@ func TestS3DedupeErr(t *testing.T) { digest := godigest.NewDigestFromEncoded(godigest.SHA256, "7173b809ca12ec5dee4506cd86be934c4596dd234ee82c0662eac04a8c2c71dc") - err := imgStore.DedupeBlob("/src/dst", digest, "/repo1/dst1") + err := imgStore.DedupeBlob("/src/dst", digest, "", "/repo1/dst1") So(err, ShouldBeNil) - err = imgStore.DedupeBlob("/src/dst", digest, "/repo2/dst2") + err = imgStore.DedupeBlob("/src/dst", digest, "", "/repo2/dst2") So(err, ShouldBeNil) // copy cache db to the new imagestore @@ -3767,10 +3767,10 @@ func TestS3DedupeErr(t *testing.T) { digest := godigest.NewDigestFromEncoded(godigest.SHA256, "7173b809ca12ec5dee4506cd86be934c4596dd234ee82c0662eac04a8c2c71dc") - err := imgStore.DedupeBlob("/src/dst", digest, "/repo1/dst1") + err := imgStore.DedupeBlob("/src/dst", digest, "", "/repo1/dst1") So(err, ShouldBeNil) - err = imgStore.DedupeBlob("/src/dst", digest, "/repo2/dst2") + err = imgStore.DedupeBlob("/src/dst", digest, "", "/repo2/dst2") So(err, ShouldBeNil) // copy cache db to the new imagestore @@ -3871,7 +3871,7 @@ func TestS3DedupeErr(t *testing.T) { }, }) - err := imgStore.DedupeBlob("repo", digest, blobPath) + err := imgStore.DedupeBlob("repo", digest, "", blobPath) So(err, ShouldBeNil) _, _, err = imgStore.CheckBlob("repo2", digest) @@ -3922,11 +3922,11 @@ func TestInjectDedupe(t *testing.T) { return &FileInfoMock{}, errS3 }, }) - err := imgStore.DedupeBlob("blob", "digest", "newblob") + err := imgStore.DedupeBlob("blob", "digest", "", "newblob") So(err, ShouldBeNil) injected := inject.InjectFailure(0) - err = imgStore.DedupeBlob("blob", "digest", "newblob") + err = imgStore.DedupeBlob("blob", "digest", "", "newblob") if injected { So(err, ShouldNotBeNil) } else { @@ -3934,7 +3934,7 @@ func TestInjectDedupe(t *testing.T) { } injected = inject.InjectFailure(1) - err = imgStore.DedupeBlob("blob", "digest", "newblob") + err = imgStore.DedupeBlob("blob", "digest", "", "newblob") if injected { So(err, ShouldNotBeNil) } else { diff --git a/pkg/storage/storage_test.go b/pkg/storage/storage_test.go index 023990677..15baff299 100644 --- a/pkg/storage/storage_test.go +++ b/pkg/storage/storage_test.go @@ -1156,6 +1156,219 @@ func TestDeleteBlobsInUse(t *testing.T) { } } +func TestReuploadCorruptedBlob(t *testing.T) { + for _, testcase := range testCases { + testcase := testcase + t.Run(testcase.testCaseName, func(t *testing.T) { + var imgStore storageTypes.ImageStore + var testDir, tdir string + var store driver.StorageDriver + var driver storageTypes.Driver + + log := log.Logger{Logger: zerolog.New(os.Stdout)} + metrics := monitoring.NewMetricsServer(false, log) + + if testcase.storageType == storageConstants.S3StorageDriverName { + tskip.SkipS3(t) + + uuid, err := guuid.NewV4() + if err != nil { + panic(err) + } + + testDir = path.Join("/oci-repo-test", uuid.String()) + tdir = t.TempDir() + + store, imgStore, _ = createObjectsStore(testDir, tdir) + driver = s3.New(store) + defer cleanupStorage(store, testDir) + } else { + tdir = t.TempDir() + cacheDriver, _ := storage.Create("boltdb", cache.BoltDBDriverParameters{ + RootDir: tdir, + Name: "cache", + UseRelPaths: true, + }, log) + driver = local.New(true) + imgStore = imagestore.NewImageStore(tdir, tdir, true, + true, log, metrics, nil, driver, cacheDriver) + } + + Convey("Test errors paths", t, func() { + storeController := storage.StoreController{DefaultStore: imgStore} + + image := CreateRandomImage() + + err := WriteImageToFileSystem(image, repoName, tag, storeController) + So(err, ShouldBeNil) + }) + + Convey("Test reupload repair corrupted image", t, func() { + storeController := storage.StoreController{DefaultStore: imgStore} + + image := CreateRandomImage() + + err := WriteImageToFileSystem(image, repoName, tag, storeController) + So(err, ShouldBeNil) + + blob := image.Layers[0] + blobDigest := godigest.FromBytes(blob) + blobSize := len(blob) + blobPath := imgStore.BlobPath(repoName, blobDigest) + + ok, size, err := imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(size, ShouldEqual, blobSize) + So(err, ShouldBeNil) + + _, err = driver.WriteFile(blobPath, []byte("corrupted")) + So(err, ShouldBeNil) + + ok, size, err = imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeFalse) + So(size, ShouldNotEqual, blobSize) + So(err, ShouldEqual, zerr.ErrBlobNotFound) + + err = WriteImageToFileSystem(image, repoName, tag, storeController) + So(err, ShouldBeNil) + + ok, size, _, err = imgStore.StatBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(blobSize, ShouldEqual, size) + So(err, ShouldBeNil) + + ok, size, err = imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(size, ShouldEqual, blobSize) + So(err, ShouldBeNil) + }) + + Convey("Test reupload repair corrupted image index", t, func() { + storeController := storage.StoreController{DefaultStore: imgStore} + + image := CreateRandomMultiarch() + + tag := "index" + + err := WriteMultiArchImageToFileSystem(image, repoName, tag, storeController) + So(err, ShouldBeNil) + + blob := image.Images[0].Layers[0] + blobDigest := godigest.FromBytes(blob) + blobSize := len(blob) + blobPath := imgStore.BlobPath(repoName, blobDigest) + + ok, size, err := imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(size, ShouldEqual, blobSize) + So(err, ShouldBeNil) + + _, err = driver.WriteFile(blobPath, []byte("corrupted")) + So(err, ShouldBeNil) + + ok, size, err = imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeFalse) + So(size, ShouldNotEqual, blobSize) + So(err, ShouldEqual, zerr.ErrBlobNotFound) + + err = WriteMultiArchImageToFileSystem(image, repoName, tag, storeController) + So(err, ShouldBeNil) + + ok, size, _, err = imgStore.StatBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(blobSize, ShouldEqual, size) + So(err, ShouldBeNil) + + ok, size, err = imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(size, ShouldEqual, blobSize) + So(err, ShouldBeNil) + }) + + Convey("Test reupload repair corrupted oras artifact", t, func() { + storeController := storage.StoreController{DefaultStore: imgStore} + + image := CreateRandomImage() + + tag := "oras-artifact" + err := WriteImageToFileSystem(image, repoName, tag, storeController) + So(err, ShouldBeNil) + + body := []byte("this is a blob") + blobDigest := godigest.FromBytes(body) + blobPath := imgStore.BlobPath(repoName, blobDigest) + buf := bytes.NewBuffer(body) + blobSize := int64(buf.Len()) + + _, size, err := imgStore.FullBlobUpload(repoName, buf, blobDigest) + So(err, ShouldBeNil) + So(size, ShouldEqual, blobSize) + + artifactManifest := artifactspec.Manifest{} + artifactType := "signature" + artifactManifest.ArtifactType = artifactType + artifactManifest.Subject = &artifactspec.Descriptor{ + MediaType: ispec.MediaTypeImageManifest, + Digest: image.Digest(), + Size: image.ManifestDescriptor.Size, + } + + artifactManifest.Blobs = []artifactspec.Descriptor{ + { + Digest: blobDigest, + Size: blobSize, + }, + } + + manBuf, err := json.Marshal(artifactManifest) + So(err, ShouldBeNil) + + manBufLen := len(manBuf) + manDigest := godigest.FromBytes(manBuf) + + _, _, err = imgStore.PutImageManifest(repoName, manDigest.String(), artifactspec.MediaTypeArtifactManifest, manBuf) + So(err, ShouldBeNil) + + descriptors, err := imgStore.GetOrasReferrers(repoName, image.Digest(), artifactType) + So(err, ShouldBeNil) + So(descriptors, ShouldNotBeEmpty) + So(descriptors[0].ArtifactType, ShouldEqual, artifactType) + So(descriptors[0].MediaType, ShouldEqual, artifactspec.MediaTypeArtifactManifest) + So(descriptors[0].Size, ShouldEqual, manBufLen) + So(descriptors[0].Digest, ShouldEqual, manDigest) + + ok, size, err := imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(size, ShouldEqual, blobSize) + So(err, ShouldBeNil) + + _, err = driver.WriteFile(blobPath, []byte("corrupted")) + So(err, ShouldBeNil) + + ok, size, err = imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeFalse) + So(size, ShouldNotEqual, blobSize) + So(err, ShouldEqual, zerr.ErrBlobNotFound) + + buf = bytes.NewBuffer(body) + _, size, err = imgStore.FullBlobUpload(repoName, buf, blobDigest) + So(err, ShouldBeNil) + So(size, ShouldEqual, blobSize) + + ok, size, _, err = imgStore.StatBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(blobSize, ShouldEqual, size) + So(err, ShouldBeNil) + + ok, size, err = imgStore.CheckBlob(repoName, blobDigest) + So(ok, ShouldBeTrue) + So(size, ShouldEqual, blobSize) + So(err, ShouldBeNil) + }) + }) + } +} + func TestStorageHandler(t *testing.T) { for _, testcase := range testCases { testcase := testcase diff --git a/pkg/storage/types/types.go b/pkg/storage/types/types.go index ab51c2abf..4f7c36412 100644 --- a/pkg/storage/types/types.go +++ b/pkg/storage/types/types.go @@ -42,7 +42,7 @@ type ImageStore interface { //nolint:interfacebloat BlobUploadInfo(repo, uuid string) (int64, error) FinishBlobUpload(repo, uuid string, body io.Reader, digest godigest.Digest) error FullBlobUpload(repo string, body io.Reader, digest godigest.Digest) (string, int64, error) - DedupeBlob(src string, dstDigest godigest.Digest, dst string) error + DedupeBlob(src string, dstDigest godigest.Digest, dstRepo, dst string) error DeleteBlobUpload(repo, uuid string) error BlobPath(repo string, digest godigest.Digest) string CheckBlob(repo string, digest godigest.Digest) (bool, int64, error) diff --git a/pkg/test/mocks/image_store_mock.go b/pkg/test/mocks/image_store_mock.go index c691f013c..e5eda2c5c 100644 --- a/pkg/test/mocks/image_store_mock.go +++ b/pkg/test/mocks/image_store_mock.go @@ -32,7 +32,7 @@ type MockedImageStore struct { PutBlobChunkFn func(repo string, uuid string, from int64, to int64, body io.Reader) (int64, error) FinishBlobUploadFn func(repo string, uuid string, body io.Reader, digest godigest.Digest) error FullBlobUploadFn func(repo string, body io.Reader, digest godigest.Digest) (string, int64, error) - DedupeBlobFn func(src string, dstDigest godigest.Digest, dst string) error + DedupeBlobFn func(src string, dstDigest godigest.Digest, dstRepo, dst string) error DeleteBlobUploadFn func(repo string, uuid string) error BlobPathFn func(repo string, digest godigest.Digest) string CheckBlobFn func(repo string, digest godigest.Digest) (bool, int64, error) @@ -240,9 +240,9 @@ func (is MockedImageStore) FullBlobUpload(repo string, body io.Reader, digest go return "", 0, nil } -func (is MockedImageStore) DedupeBlob(src string, dstDigest godigest.Digest, dst string) error { +func (is MockedImageStore) DedupeBlob(src string, dstDigest godigest.Digest, dstRepo, dst string) error { if is.DedupeBlobFn != nil { - return is.DedupeBlobFn(src, dstDigest, dst) + return is.DedupeBlobFn(src, dstDigest, dstRepo, dst) } return nil diff --git a/test/gc-stress/config-gc-bench-s3-minio.json b/test/gc-stress/config-gc-bench-s3-minio.json index 7c59e661a..89ae27709 100644 --- a/test/gc-stress/config-gc-bench-s3-minio.json +++ b/test/gc-stress/config-gc-bench-s3-minio.json @@ -4,8 +4,8 @@ "rootDirectory": "/tmp/zot/s3", "gc": true, "gcReferrers": false, - "gcDelay": "3m", - "untaggedImageRetentionDelay": "3m", + "gcDelay": "4m", + "untaggedImageRetentionDelay": "4m", "gcInterval": "1s", "storageDriver": { "name": "s3", diff --git a/test/gc-stress/config-gc-referrers-bench-s3-minio.json b/test/gc-stress/config-gc-referrers-bench-s3-minio.json index 58ba09934..cac4aa270 100644 --- a/test/gc-stress/config-gc-referrers-bench-s3-minio.json +++ b/test/gc-stress/config-gc-referrers-bench-s3-minio.json @@ -4,8 +4,8 @@ "rootDirectory": "/tmp/zot/s3", "gc": true, "gcReferrers": true, - "gcDelay": "3m", - "untaggedImageRetentionDelay": "3m", + "gcDelay": "4m", + "untaggedImageRetentionDelay": "4m", "gcInterval": "1s", "storageDriver": { "name": "s3",