2023-03-28 12:20:09 -05:00
|
|
|
package meta_test
|
2023-01-09 15:37:44 -05:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2023-05-24 11:46:16 -05:00
|
|
|
notreg "github.com/notaryproject/notation-go/registry"
|
2023-01-09 15:37:44 -05:00
|
|
|
godigest "github.com/opencontainers/go-digest"
|
|
|
|
ispec "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
|
. "github.com/smartystreets/goconvey/convey"
|
|
|
|
|
|
|
|
zerr "zotregistry.io/zot/errors"
|
|
|
|
"zotregistry.io/zot/pkg/extensions/monitoring"
|
|
|
|
"zotregistry.io/zot/pkg/log"
|
2023-03-28 12:20:09 -05:00
|
|
|
"zotregistry.io/zot/pkg/meta"
|
2023-07-18 12:27:26 -05:00
|
|
|
"zotregistry.io/zot/pkg/meta/boltdb"
|
|
|
|
mTypes "zotregistry.io/zot/pkg/meta/types"
|
2023-01-09 15:37:44 -05:00
|
|
|
"zotregistry.io/zot/pkg/storage"
|
|
|
|
"zotregistry.io/zot/pkg/storage/local"
|
|
|
|
"zotregistry.io/zot/pkg/test"
|
2023-09-15 11:53:15 -05:00
|
|
|
. "zotregistry.io/zot/pkg/test/image-utils"
|
2023-01-09 15:37:44 -05:00
|
|
|
"zotregistry.io/zot/pkg/test/mocks"
|
|
|
|
)
|
|
|
|
|
|
|
|
var ErrTestError = errors.New("test error")
|
|
|
|
|
|
|
|
func TestOnUpdateManifest(t *testing.T) {
|
|
|
|
Convey("On UpdateManifest", t, func() {
|
|
|
|
rootDir := t.TempDir()
|
|
|
|
storeController := storage.StoreController{}
|
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
metrics := monitoring.NewMetricsServer(false, log)
|
2023-09-01 12:54:39 -05:00
|
|
|
storeController.DefaultStore = local.NewImageStore(rootDir, true, true, 1*time.Second,
|
|
|
|
1*time.Second, true, true, log, metrics, nil, nil,
|
2023-01-09 15:37:44 -05:00
|
|
|
)
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
params := boltdb.DBParameters{
|
2023-01-09 15:37:44 -05:00
|
|
|
RootDir: rootDir,
|
2023-03-28 12:20:09 -05:00
|
|
|
}
|
2023-07-18 12:27:26 -05:00
|
|
|
boltDriver, err := boltdb.GetBoltDriver(params)
|
2023-03-28 12:20:09 -05:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB, err := boltdb.New(boltDriver, log)
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-28 09:53:46 -05:00
|
|
|
config, layers, manifest, err := test.GetRandomImageComponents(100) //nolint:staticcheck
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-02-27 14:23:18 -05:00
|
|
|
err = test.WriteImageToFileSystem(
|
2023-09-15 11:53:15 -05:00
|
|
|
Image{
|
2023-07-28 09:53:46 -05:00
|
|
|
Config: config, Manifest: manifest, Layers: layers,
|
|
|
|
}, "repo", "tag1", storeController)
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
manifestBlob, err := json.Marshal(manifest)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
digest := godigest.FromBytes(manifestBlob)
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
err = meta.OnUpdateManifest("repo", "tag1", "", digest, manifestBlob, storeController, metaDB, log)
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
repoMeta, err := metaDB.GetRepoMeta("repo")
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
So(repoMeta.Tags, ShouldContainKey, "tag1")
|
|
|
|
})
|
2023-02-27 14:23:18 -05:00
|
|
|
|
|
|
|
Convey("metadataSuccessfullySet is false", t, func() {
|
|
|
|
rootDir := t.TempDir()
|
|
|
|
storeController := storage.StoreController{}
|
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
metrics := monitoring.NewMetricsServer(false, log)
|
2023-09-01 12:54:39 -05:00
|
|
|
storeController.DefaultStore = local.NewImageStore(rootDir, true, true, 1*time.Second,
|
|
|
|
1*time.Second, true, true, log, metrics, nil, nil,
|
2023-02-27 14:23:18 -05:00
|
|
|
)
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{
|
|
|
|
SetManifestDataFn: func(manifestDigest godigest.Digest, mm mTypes.ManifestData) error {
|
2023-02-27 14:23:18 -05:00
|
|
|
return ErrTestError
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2023-03-28 12:20:09 -05:00
|
|
|
err := meta.OnUpdateManifest("repo", "tag1", ispec.MediaTypeImageManifest, "digest",
|
2023-07-18 12:27:26 -05:00
|
|
|
[]byte("{}"), storeController, metaDB, log)
|
2023-02-27 14:23:18 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
2023-01-09 15:37:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestUpdateErrors(t *testing.T) {
|
|
|
|
Convey("Update operations", t, func() {
|
2023-05-24 11:46:16 -05:00
|
|
|
Convey("On UpdateManifest", func() {
|
|
|
|
imageStore := mocks.MockedImageStore{}
|
|
|
|
storeController := storage.StoreController{DefaultStore: &imageStore}
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{}
|
2023-05-24 11:46:16 -05:00
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
|
|
|
Convey("CheckIsImageSignature errors", func() {
|
|
|
|
badManifestBlob := []byte("bad")
|
|
|
|
|
|
|
|
imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) {
|
|
|
|
return []byte{}, "", "", zerr.ErrManifestNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
imageStore.DeleteImageManifestFn = func(repo, reference string, detectCollision bool) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
err := meta.OnUpdateManifest("repo", "tag1", "digest", "media", badManifestBlob,
|
2023-07-18 12:27:26 -05:00
|
|
|
storeController, metaDB, log)
|
2023-05-24 11:46:16 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("GetSignatureLayersInfo errors", func() {
|
|
|
|
// get notation signature layers info
|
|
|
|
badNotationManifestContent := ispec.Manifest{
|
|
|
|
Subject: &ispec.Descriptor{
|
|
|
|
Digest: "123",
|
|
|
|
},
|
|
|
|
Config: ispec.Descriptor{MediaType: notreg.ArtifactTypeNotation},
|
|
|
|
}
|
|
|
|
|
|
|
|
badNotationManifestBlob, err := json.Marshal(badNotationManifestContent)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) {
|
|
|
|
return badNotationManifestBlob, "", "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
err = meta.OnUpdateManifest("repo", "tag1", "", "digest", badNotationManifestBlob,
|
2023-07-18 12:27:26 -05:00
|
|
|
storeController, metaDB, log)
|
2023-05-24 11:46:16 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("UpdateSignaturesValidity", func() {
|
|
|
|
notationManifestContent := ispec.Manifest{
|
|
|
|
Subject: &ispec.Descriptor{
|
|
|
|
Digest: "123",
|
|
|
|
},
|
|
|
|
Config: ispec.Descriptor{MediaType: notreg.ArtifactTypeNotation},
|
|
|
|
Layers: []ispec.Descriptor{{
|
|
|
|
MediaType: ispec.MediaTypeImageLayer,
|
|
|
|
Digest: godigest.FromString("blob digest"),
|
|
|
|
}},
|
|
|
|
}
|
|
|
|
|
|
|
|
notationManifestBlob, err := json.Marshal(notationManifestContent)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) {
|
|
|
|
return notationManifestBlob, "", "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
imageStore.GetBlobContentFn = func(repo string, digest godigest.Digest) ([]byte, error) {
|
|
|
|
return []byte{}, nil
|
|
|
|
}
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB.UpdateSignaturesValidityFn = func(repo string, manifestDigest godigest.Digest) error {
|
2023-05-24 11:46:16 -05:00
|
|
|
return ErrTestError
|
|
|
|
}
|
|
|
|
|
|
|
|
err = meta.OnUpdateManifest("repo", "tag1", "", "digest", notationManifestBlob,
|
2023-07-18 12:27:26 -05:00
|
|
|
storeController, metaDB, log)
|
2023-05-24 11:46:16 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2023-01-09 15:37:44 -05:00
|
|
|
Convey("On DeleteManifest", func() {
|
|
|
|
imageStore := mocks.MockedImageStore{}
|
|
|
|
storeController := storage.StoreController{DefaultStore: &imageStore}
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{}
|
2023-01-09 15:37:44 -05:00
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
|
|
|
Convey("CheckIsImageSignature errors", func() {
|
2023-05-15 04:02:23 -05:00
|
|
|
badManifestBlob := []byte("bad")
|
2023-01-09 15:37:44 -05:00
|
|
|
|
|
|
|
imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) {
|
|
|
|
return []byte{}, "", "", zerr.ErrManifestNotFound
|
|
|
|
}
|
|
|
|
|
2023-05-15 04:02:23 -05:00
|
|
|
err := meta.OnDeleteManifest("repo", "tag1", "digest", "media", badManifestBlob,
|
2023-07-18 12:27:26 -05:00
|
|
|
storeController, metaDB, log)
|
2023-03-10 13:37:29 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("DeleteReferrers errors", func() {
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB.DeleteReferrerFn = func(repo string, referredDigest, referrerDigest godigest.Digest) error {
|
2023-03-10 13:37:29 -05:00
|
|
|
return ErrTestError
|
|
|
|
}
|
|
|
|
|
2023-03-28 12:20:09 -05:00
|
|
|
err := meta.OnDeleteManifest("repo", "tag1", "digest", "media",
|
2023-03-10 13:37:29 -05:00
|
|
|
[]byte(`{"subject": {"digest": "dig"}}`),
|
2023-07-18 12:27:26 -05:00
|
|
|
storeController, metaDB, log)
|
2023-03-10 13:37:29 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
2023-01-09 15:37:44 -05:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("On GetManifest", func() {
|
|
|
|
imageStore := mocks.MockedImageStore{}
|
|
|
|
storeController := storage.StoreController{DefaultStore: &imageStore}
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{}
|
2023-01-09 15:37:44 -05:00
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
|
|
|
Convey("CheckIsImageSignature errors", func() {
|
2023-05-15 04:02:23 -05:00
|
|
|
badManifestBlob := []byte("bad")
|
2023-01-09 15:37:44 -05:00
|
|
|
|
|
|
|
imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) {
|
|
|
|
return []byte{}, "", "", zerr.ErrManifestNotFound
|
|
|
|
}
|
|
|
|
|
2023-05-15 04:02:23 -05:00
|
|
|
err := meta.OnGetManifest("repo", "tag1", badManifestBlob,
|
2023-07-18 12:27:26 -05:00
|
|
|
storeController, metaDB, log)
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2023-05-15 04:02:23 -05:00
|
|
|
Convey("SetImageMetaFromInput", func() {
|
2023-01-09 15:37:44 -05:00
|
|
|
imageStore := mocks.MockedImageStore{}
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{}
|
2023-01-09 15:37:44 -05:00
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageManifest, "digest",
|
|
|
|
[]byte("BadManifestBlob"), imageStore, metaDB, log)
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
|
|
|
|
// reference is digest
|
|
|
|
|
|
|
|
manifestContent := ispec.Manifest{}
|
|
|
|
manifestBlob, err := json.Marshal(manifestContent)
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) {
|
|
|
|
return manifestBlob, "", "", nil
|
|
|
|
}
|
|
|
|
imageStore.GetBlobContentFn = func(repo string, digest godigest.Digest) ([]byte, error) {
|
|
|
|
return []byte("{}"), nil
|
|
|
|
}
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
err = meta.SetImageMetaFromInput("repo", string(godigest.FromString("reference")), "", "digest",
|
|
|
|
manifestBlob, imageStore, metaDB, log)
|
2023-01-09 15:37:44 -05:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
})
|
2023-03-10 13:37:29 -05:00
|
|
|
|
2023-05-15 04:02:23 -05:00
|
|
|
Convey("SetImageMetaFromInput SetData errors", func() {
|
2023-03-10 13:37:29 -05:00
|
|
|
imageStore := mocks.MockedImageStore{}
|
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{
|
|
|
|
SetManifestDataFn: func(manifestDigest godigest.Digest, mm mTypes.ManifestData) error {
|
2023-03-10 13:37:29 -05:00
|
|
|
return ErrTestError
|
|
|
|
},
|
|
|
|
}
|
2023-07-18 12:27:26 -05:00
|
|
|
err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageManifest, "digest",
|
|
|
|
[]byte("{}"), imageStore, metaDB, log)
|
2023-03-10 13:37:29 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
2023-05-10 12:15:33 -05:00
|
|
|
})
|
2023-03-10 13:37:29 -05:00
|
|
|
|
2023-05-15 04:02:23 -05:00
|
|
|
Convey("SetImageMetaFromInput SetIndexData errors", func() {
|
2023-05-10 12:15:33 -05:00
|
|
|
imageStore := mocks.MockedImageStore{}
|
|
|
|
log := log.NewLogger("debug", "")
|
2023-03-10 13:37:29 -05:00
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{
|
|
|
|
SetIndexDataFn: func(digest godigest.Digest, indexData mTypes.IndexData) error {
|
2023-03-10 13:37:29 -05:00
|
|
|
return ErrTestError
|
|
|
|
},
|
|
|
|
}
|
2023-07-18 12:27:26 -05:00
|
|
|
err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageIndex, "digest",
|
|
|
|
[]byte("{}"), imageStore, metaDB, log)
|
2023-03-10 13:37:29 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
|
2023-05-15 04:02:23 -05:00
|
|
|
Convey("SetImageMetaFromInput SetReferrer errors", func() {
|
2023-03-10 13:37:29 -05:00
|
|
|
imageStore := mocks.MockedImageStore{
|
|
|
|
GetBlobContentFn: func(repo string, digest godigest.Digest) ([]byte, error) {
|
|
|
|
return []byte("{}"), nil
|
|
|
|
},
|
|
|
|
}
|
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
metaDB := mocks.MetaDBMock{
|
|
|
|
SetReferrerFn: func(repo string, referredDigest godigest.Digest, referrer mTypes.ReferrerInfo) error {
|
2023-03-10 13:37:29 -05:00
|
|
|
return ErrTestError
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageManifest, "digest",
|
|
|
|
[]byte(`{"subject": {"digest": "subjDigest"}}`), imageStore, metaDB, log)
|
2023-03-10 13:37:29 -05:00
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
2023-01-09 15:37:44 -05:00
|
|
|
})
|
|
|
|
}
|