2021-10-05 12:12:22 +03:00
|
|
|
package storage_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
godigest "github.com/opencontainers/go-digest"
|
|
|
|
ispec "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
|
. "github.com/smartystreets/goconvey/convey"
|
2022-10-20 19:39:20 +03:00
|
|
|
|
2021-10-05 12:12:22 +03:00
|
|
|
"zotregistry.io/zot/pkg/extensions/monitoring"
|
|
|
|
"zotregistry.io/zot/pkg/log"
|
|
|
|
"zotregistry.io/zot/pkg/storage"
|
2022-11-03 00:53:08 +02:00
|
|
|
"zotregistry.io/zot/pkg/storage/cache"
|
2023-05-26 21:08:19 +03:00
|
|
|
common "zotregistry.io/zot/pkg/storage/common"
|
|
|
|
storageConstants "zotregistry.io/zot/pkg/storage/constants"
|
2022-09-30 20:35:16 +03:00
|
|
|
"zotregistry.io/zot/pkg/storage/local"
|
2022-12-11 02:38:01 +02:00
|
|
|
"zotregistry.io/zot/pkg/test"
|
2021-10-05 12:12:22 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
repoName = "test"
|
2022-12-11 02:38:01 +02:00
|
|
|
tag = "1.0"
|
2021-10-05 12:12:22 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestCheckAllBlobsIntegrity(t *testing.T) {
|
2022-03-07 16:55:12 +08:00
|
|
|
dir := t.TempDir()
|
2021-10-05 12:12:22 +03:00
|
|
|
|
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
|
|
|
metrics := monitoring.NewMetricsServer(false, log)
|
2022-11-03 00:53:08 +02:00
|
|
|
cacheDriver, _ := storage.Create("boltdb", cache.BoltDBDriverParameters{
|
|
|
|
RootDir: dir,
|
|
|
|
Name: "cache",
|
|
|
|
UseRelPaths: true,
|
|
|
|
}, log)
|
2023-05-26 21:08:19 +03:00
|
|
|
imgStore := local.NewImageStore(dir, true, storageConstants.DefaultGCDelay,
|
2022-11-03 00:53:08 +02:00
|
|
|
true, true, log, metrics, nil, cacheDriver)
|
2021-10-05 12:12:22 +03:00
|
|
|
|
|
|
|
Convey("Scrub only one repo", t, func(c C) {
|
|
|
|
// initialize repo
|
2022-03-07 16:55:12 +08:00
|
|
|
err := imgStore.InitRepo(repoName)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
2021-12-13 19:23:31 +00:00
|
|
|
ok := imgStore.DirExists(path.Join(imgStore.RootDir(), repoName))
|
2021-10-05 12:12:22 +03:00
|
|
|
So(ok, ShouldBeTrue)
|
|
|
|
storeController := storage.StoreController{}
|
2021-12-13 19:23:31 +00:00
|
|
|
storeController.DefaultStore = imgStore
|
|
|
|
So(storeController.GetImageStore(repoName), ShouldResemble, imgStore)
|
2021-10-05 12:12:22 +03:00
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
storeCtlr := storage.StoreController{}
|
|
|
|
storeCtlr.DefaultStore = imgStore
|
2021-10-05 12:12:22 +03:00
|
|
|
|
2022-12-11 02:38:01 +02:00
|
|
|
config, layers, manifest, err := test.GetImageComponents(1000)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2022-12-11 02:38:01 +02:00
|
|
|
layerReader := bytes.NewReader(layers[0])
|
|
|
|
layerDigest := godigest.FromBytes(layers[0])
|
|
|
|
_, _, err = imgStore.FullBlobUpload(repoName, layerReader, layerDigest)
|
2022-03-21 17:37:23 +00:00
|
|
|
So(err, ShouldBeNil)
|
2021-10-05 12:12:22 +03:00
|
|
|
|
2022-12-11 02:38:01 +02:00
|
|
|
configBlob, err := json.Marshal(config)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
configReader := bytes.NewReader(configBlob)
|
|
|
|
configDigest := godigest.FromBytes(configBlob)
|
|
|
|
_, _, err = imgStore.FullBlobUpload(repoName, configReader, configDigest)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2022-12-11 02:38:01 +02:00
|
|
|
manifestBlob, err := json.Marshal(manifest)
|
|
|
|
So(err, ShouldBeNil)
|
2023-05-12 19:32:01 +03:00
|
|
|
manifestDigest, _, err := imgStore.PutImageManifest(repoName, tag, ispec.MediaTypeImageManifest, manifestBlob)
|
2022-12-11 02:38:01 +02:00
|
|
|
So(err, ShouldBeNil)
|
2022-10-22 23:46:13 +03:00
|
|
|
|
2021-10-05 12:12:22 +03:00
|
|
|
Convey("Blobs integrity not affected", func() {
|
|
|
|
buff := bytes.NewBufferString("")
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
res, err := storeCtlr.CheckAllBlobsIntegrity()
|
2021-10-05 12:12:22 +03:00
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
space := regexp.MustCompile(`\s+`)
|
|
|
|
str := space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual := strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test 1.0 ok")
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("Manifest integrity affected", func() {
|
|
|
|
// get content of manifest file
|
2022-12-11 02:38:01 +02:00
|
|
|
content, _, _, err := imgStore.GetImageManifest(repoName, manifestDigest.String())
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
// delete content of manifest file
|
2022-12-11 02:38:01 +02:00
|
|
|
manifestDig := manifestDigest.Encoded()
|
|
|
|
manifestFile := path.Join(imgStore.RootDir(), repoName, "/blobs/sha256", manifestDig)
|
2021-10-05 12:12:22 +03:00
|
|
|
err = os.Truncate(manifestFile, 0)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff := bytes.NewBufferString("")
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
res, err := storeCtlr.CheckAllBlobsIntegrity()
|
2021-10-05 12:12:22 +03:00
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
space := regexp.MustCompile(`\s+`)
|
|
|
|
str := space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual := strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
// verify error message
|
|
|
|
So(actual, ShouldContainSubstring, "test 1.0 affected parse application/vnd.oci.image.manifest.v1+json")
|
|
|
|
|
2023-05-26 21:08:19 +03:00
|
|
|
index, err := common.GetIndex(imgStore, repoName, log.With().Caller().Logger())
|
2022-12-11 02:38:01 +02:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
So(len(index.Manifests), ShouldEqual, 1)
|
|
|
|
manifestDescriptor := index.Manifests[0]
|
|
|
|
|
|
|
|
repoDir := path.Join(dir, repoName)
|
|
|
|
imageRes := storage.CheckLayers(repoName, tag, repoDir, manifestDescriptor)
|
|
|
|
So(imageRes.Status, ShouldEqual, "affected")
|
|
|
|
So(imageRes.Error, ShouldEqual, "unexpected end of JSON input")
|
|
|
|
|
2021-10-05 12:12:22 +03:00
|
|
|
// put manifest content back to file
|
2022-09-02 14:56:02 +02:00
|
|
|
err = os.WriteFile(manifestFile, content, 0o600)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("Config integrity affected", func() {
|
|
|
|
// get content of config file
|
2022-10-22 23:46:13 +03:00
|
|
|
content, err := imgStore.GetBlobContent(repoName, configDigest)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
// delete content of config file
|
2022-12-11 02:38:01 +02:00
|
|
|
configDig := configDigest.Encoded()
|
|
|
|
configFile := path.Join(imgStore.RootDir(), repoName, "/blobs/sha256", configDig)
|
2021-10-05 12:12:22 +03:00
|
|
|
err = os.Truncate(configFile, 0)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff := bytes.NewBufferString("")
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
res, err := storeCtlr.CheckAllBlobsIntegrity()
|
2021-10-05 12:12:22 +03:00
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
space := regexp.MustCompile(`\s+`)
|
|
|
|
str := space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual := strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test 1.0 affected stat: parse application/vnd.oci.image.config.v1+json")
|
|
|
|
|
|
|
|
// put config content back to file
|
2022-09-02 14:56:02 +02:00
|
|
|
err = os.WriteFile(configFile, content, 0o600)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("Layers integrity affected", func() {
|
|
|
|
// get content of layer
|
2022-10-22 23:46:13 +03:00
|
|
|
content, err := imgStore.GetBlobContent(repoName, layerDigest)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
// delete content of layer file
|
2022-12-11 02:38:01 +02:00
|
|
|
layerDig := layerDigest.Encoded()
|
|
|
|
layerFile := path.Join(imgStore.RootDir(), repoName, "/blobs/sha256", layerDig)
|
2021-10-05 12:12:22 +03:00
|
|
|
err = os.Truncate(layerFile, 0)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff := bytes.NewBufferString("")
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
res, err := storeCtlr.CheckAllBlobsIntegrity()
|
2021-10-05 12:12:22 +03:00
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
space := regexp.MustCompile(`\s+`)
|
|
|
|
str := space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual := strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test 1.0 affected blob: bad blob digest")
|
|
|
|
|
|
|
|
// put layer content back to file
|
2022-09-02 14:56:02 +02:00
|
|
|
err = os.WriteFile(layerFile, content, 0o600)
|
2021-10-05 12:12:22 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("Layer not found", func() {
|
2022-12-11 02:38:01 +02:00
|
|
|
// change layer file permissions
|
|
|
|
layerDig := layerDigest.Encoded()
|
|
|
|
repoDir := path.Join(dir, repoName)
|
|
|
|
layerFile := path.Join(repoDir, "/blobs/sha256", layerDig)
|
|
|
|
err = os.Chmod(layerFile, 0x0200)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-05-26 21:08:19 +03:00
|
|
|
index, err := common.GetIndex(imgStore, repoName, log.With().Caller().Logger())
|
2022-12-11 02:38:01 +02:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
So(len(index.Manifests), ShouldEqual, 1)
|
|
|
|
manifestDescriptor := index.Manifests[0]
|
|
|
|
|
|
|
|
imageRes := storage.CheckLayers(repoName, tag, repoDir, manifestDescriptor)
|
|
|
|
So(imageRes.Status, ShouldEqual, "affected")
|
|
|
|
So(imageRes.Error, ShouldEqual, "blob: not found")
|
|
|
|
err = os.Chmod(layerFile, 0x0600)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2021-10-05 12:12:22 +03:00
|
|
|
// delete layer file
|
|
|
|
err = os.Remove(layerFile)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff := bytes.NewBufferString("")
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
res, err := storeCtlr.CheckAllBlobsIntegrity()
|
2021-10-05 12:12:22 +03:00
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
space := regexp.MustCompile(`\s+`)
|
|
|
|
str := space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual := strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test 1.0 affected blob: not found")
|
|
|
|
})
|
2022-12-11 02:38:01 +02:00
|
|
|
|
|
|
|
Convey("Scrub index", func() {
|
|
|
|
newConfig, newLayers, newManifest, err := test.GetImageComponents(10)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
newLayerReader := bytes.NewReader(newLayers[0])
|
|
|
|
newLayerDigest := godigest.FromBytes(newLayers[0])
|
|
|
|
_, _, err = imgStore.FullBlobUpload(repoName, newLayerReader, newLayerDigest)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
newConfigBlob, err := json.Marshal(newConfig)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
newConfigReader := bytes.NewReader(newConfigBlob)
|
|
|
|
newConfigDigest := godigest.FromBytes(newConfigBlob)
|
|
|
|
_, _, err = imgStore.FullBlobUpload(repoName, newConfigReader, newConfigDigest)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
newManifestBlob, err := json.Marshal(newManifest)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
newManifestReader := bytes.NewReader(newManifestBlob)
|
|
|
|
newManifestDigest := godigest.FromBytes(newManifestBlob)
|
|
|
|
_, _, err = imgStore.FullBlobUpload(repoName, newManifestReader, newManifestDigest)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
var index ispec.Index
|
|
|
|
index.SchemaVersion = 2
|
|
|
|
index.Manifests = []ispec.Descriptor{
|
|
|
|
{
|
|
|
|
MediaType: ispec.MediaTypeImageManifest,
|
|
|
|
Digest: newManifestDigest,
|
|
|
|
Size: int64(len(newManifestBlob)),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
indexBlob, err := json.Marshal(index)
|
|
|
|
So(err, ShouldBeNil)
|
2023-05-12 19:32:01 +03:00
|
|
|
indexDigest, _, err := imgStore.PutImageManifest(repoName, "", ispec.MediaTypeImageIndex, indexBlob)
|
2022-12-11 02:38:01 +02:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff := bytes.NewBufferString("")
|
|
|
|
|
|
|
|
res, err := storeCtlr.CheckAllBlobsIntegrity()
|
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
space := regexp.MustCompile(`\s+`)
|
|
|
|
str := space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual := strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test 1.0 ok")
|
|
|
|
So(actual, ShouldContainSubstring, "test ok")
|
|
|
|
|
|
|
|
// test scrub index - errors
|
|
|
|
indexFile := path.Join(imgStore.RootDir(), repoName, "/blobs/sha256", indexDigest.Encoded())
|
|
|
|
err = os.Chmod(indexFile, 0o000)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff = bytes.NewBufferString("")
|
|
|
|
|
|
|
|
res, err = storeCtlr.CheckAllBlobsIntegrity()
|
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
str = space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual = strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test affected")
|
|
|
|
|
|
|
|
err = os.Chmod(indexFile, 0o600)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
err = os.Truncate(indexFile, 0)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff = bytes.NewBufferString("")
|
|
|
|
|
|
|
|
res, err = storeCtlr.CheckAllBlobsIntegrity()
|
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
str = space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual = strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test affected")
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("Manifest not found", func() {
|
|
|
|
// delete manifest file
|
|
|
|
manifestDig := manifestDigest.Encoded()
|
|
|
|
manifestFile := path.Join(imgStore.RootDir(), repoName, "/blobs/sha256", manifestDig)
|
|
|
|
err = os.Remove(manifestFile)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
buff := bytes.NewBufferString("")
|
|
|
|
|
|
|
|
res, err := storeCtlr.CheckAllBlobsIntegrity()
|
|
|
|
res.PrintScrubResults(buff)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
space := regexp.MustCompile(`\s+`)
|
|
|
|
str := space.ReplaceAllString(buff.String(), " ")
|
|
|
|
actual := strings.TrimSpace(str)
|
|
|
|
So(actual, ShouldContainSubstring, "IMAGE NAME TAG STATUS ERROR")
|
|
|
|
So(actual, ShouldContainSubstring, "test 1.0 affected")
|
|
|
|
So(actual, ShouldContainSubstring, "no such file or directory")
|
|
|
|
|
2023-05-26 21:08:19 +03:00
|
|
|
index, err := common.GetIndex(imgStore, repoName, log.With().Caller().Logger())
|
2022-12-11 02:38:01 +02:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
So(len(index.Manifests), ShouldEqual, 2)
|
|
|
|
manifestDescriptor := index.Manifests[0]
|
|
|
|
|
|
|
|
repoDir := path.Join(dir, repoName)
|
|
|
|
imageRes := storage.CheckLayers(repoName, tag, repoDir, manifestDescriptor)
|
|
|
|
So(imageRes.Status, ShouldEqual, "affected")
|
|
|
|
So(imageRes.Error, ShouldContainSubstring, "no such file or directory")
|
|
|
|
})
|
2021-10-05 12:12:22 +03:00
|
|
|
})
|
|
|
|
}
|