diff --git a/Makefile b/Makefile index 4d0287c38e..4ca2c0ac0f 100644 --- a/Makefile +++ b/Makefile @@ -28,12 +28,26 @@ ZUI_VERSION := commit-fad5572 SWAGGER_VERSION := v1.8.12 STACKER := $(TOOLSDIR)/bin/stacker BATS := $(TOOLSDIR)/bin/bats -PROTOC_VERSION := 3.15.8 -PROTOC := $(TOOLSDIR)/bin/protoc TESTDATA := $(TOP_LEVEL)/test/data OS ?= $(shell go env GOOS) ARCH ?= $(shell go env GOARCH) +GO_PROTOC_VERSION := 1.31.0 +PROTOC_VERSION := 3.15.8 +HOST_OS := $(shell go env GOOS) +HOST_ARCH := $(shell go env GOARCH) +ifeq ($(HOST_OS),linux) + PROTOC_OS = linux + PROTOC_ARCH = x86_64 + ifeq ($(HOST_ARCH),arm64) + PROTOC_ARCH = aarch_64 + endif +else ifeq ($(HOST_OS),darwin) + PROTOC_OS = osx + PROTOC_ARCH = universal_binary +endif +PROTOC := $(TOOLSDIR)/bin/protoc + BENCH_OUTPUT ?= stdout ALL_EXTENSIONS = debug,imagetrust,lint,metrics,mgmt,profile,scrub,search,sync,ui,userprefs EXTENSIONS ?= sync,search,scrub,metrics,lint,ui,mgmt,profile,userprefs,imagetrust @@ -104,14 +118,14 @@ gen-protobuf: $(PROTOC) $(PROTOC) --experimental_allow_proto3_optional \ --proto_path=$(TOP_LEVEL)/pkg/meta/proto \ --go_out=$(TOP_LEVEL)/pkg/meta/ \ - --go_opt='Mdescriptor.proto=./proto_go' \ - $(TOP_LEVEL)/pkg/meta/proto/descriptor.proto - $(PROTOC) --experimental_allow_proto3_optional \ - --proto_path=$(TOP_LEVEL)/pkg/meta/proto \ - --go_out=$(TOP_LEVEL)/pkg/meta/ \ + --go_opt='Moci.proto=./proto_go' \ + --go_opt='Mmeta.proto=./proto_go' \ --go_opt='Mconfig.proto=./proto_go' \ + --go_opt='Mmanifest.proto=./proto_go' \ + --go_opt='Mindex.proto=./proto_go' \ --go_opt='Mdescriptor.proto=./proto_go' \ - $(TOP_LEVEL)/pkg/meta/proto/config.proto + --go_opt='Mversioned.proto=./proto_go' \ + $(TOP_LEVEL)/pkg/meta/proto/meta.proto $(PROTOC) --experimental_allow_proto3_optional \ --proto_path=$(TOP_LEVEL)/pkg/meta/proto \ --go_out=$(TOP_LEVEL)/pkg/meta/ \ @@ -120,25 +134,29 @@ gen-protobuf: $(PROTOC) $(PROTOC) --experimental_allow_proto3_optional \ --proto_path=$(TOP_LEVEL)/pkg/meta/proto \ --go_out=$(TOP_LEVEL)/pkg/meta/ \ - --go_opt='Mmanifest.proto=./proto_go' \ --go_opt='Mdescriptor.proto=./proto_go' \ - --go_opt='Mversioned.proto=./proto_go' \ - $(TOP_LEVEL)/pkg/meta/proto/manifest.proto + $(TOP_LEVEL)/pkg/meta/proto/descriptor.proto $(PROTOC) --experimental_allow_proto3_optional \ --proto_path=$(TOP_LEVEL)/pkg/meta/proto \ --go_out=$(TOP_LEVEL)/pkg/meta/ \ - --go_opt='Mindex.proto=./proto_go' \ --go_opt='Mdescriptor.proto=./proto_go' \ --go_opt='Mversioned.proto=./proto_go' \ + --go_opt='Mindex.proto=./proto_go' \ $(TOP_LEVEL)/pkg/meta/proto/index.proto $(PROTOC) --experimental_allow_proto3_optional \ --proto_path=$(TOP_LEVEL)/pkg/meta/proto \ --go_out=$(TOP_LEVEL)/pkg/meta/ \ - --go_opt='MimageData.proto=./proto_go' \ + --go_opt='Moci.proto=./proto_go' \ --go_opt='Mdescriptor.proto=./proto_go' \ - --go_opt='Mversioned.proto=./proto_go' \ --go_opt='Mconfig.proto=./proto_go' \ - $(TOP_LEVEL)/pkg/meta/proto/imageData.proto + $(TOP_LEVEL)/pkg/meta/proto/config.proto + $(PROTOC) --experimental_allow_proto3_optional \ + --proto_path=$(TOP_LEVEL)/pkg/meta/proto \ + --go_out=$(TOP_LEVEL)/pkg/meta/ \ + --go_opt='Mversioned.proto=./proto_go' \ + --go_opt='Mdescriptor.proto=./proto_go' \ + --go_opt='Mmanifest.proto=./proto_go' \ + $(TOP_LEVEL)/pkg/meta/proto/manifest.proto .PHONY: binary-minimal binary-minimal: EXTENSIONS= @@ -263,10 +281,10 @@ $(CRICTL): $(PROTOC): mkdir -p $(TOOLSDIR)/bin - curl -Lo protoc.zip https://github.com/protocolbuffers/protobuf/releases/download/v$(PROTOC_VERSION)/protoc-$(PROTOC_VERSION)-linux-x86_64.zip - unzip -d $(TOOLSDIR) protoc.zip bin/protoc + curl -Lo protoc.zip https://github.com/protocolbuffers/protobuf/releases/download/v$(PROTOC_VERSION)/protoc-$(PROTOC_VERSION)-$(PROTOC_OS)-$(PROTOC_ARCH).zip + unzip -o -d $(TOOLSDIR) protoc.zip bin/protoc chmod +x $(PROTOC) - go install google.golang.org/protobuf/cmd/protoc-gen-go@latest + go install google.golang.org/protobuf/cmd/protoc-gen-go@v$(GO_PROTOC_VERSION) $(ACTION_VALIDATOR): mkdir -p $(TOOLSDIR)/bin diff --git a/codecov.yml b/codecov.yml index 913bdf3a65..734d4be28e 100644 --- a/codecov.yml +++ b/codecov.yml @@ -6,3 +6,4 @@ ignore: - "./pkg/test/mocks/*.go" - "./swagger/*.go" - "./pkg/test/test_http_server.go" + - "./pkg/meta/proto_go/*.go" diff --git a/errors/errors.go b/errors/errors.go index ed2dcf42cf..1960b6baae 100644 --- a/errors/errors.go +++ b/errors/errors.go @@ -108,6 +108,8 @@ var ( ErrManifestConflict = errors.New("manifest: multiple manifests found") ErrManifestMetaNotFound = errors.New("metadb: image metadata not found for given manifest reference") ErrManifestDataNotFound = errors.New("metadb: image data not found for given manifest digest") + ErrImageDataNotFound = errors.New("metadb: image data not found") + ErrUnexpectedMediaType = errors.New("metadb: got unexpected media type") ErrIndexDataNotFount = errors.New("metadb: index data not found for given digest") ErrRepoMetaNotFound = errors.New("metadb: repo metadata not found for given repo name") ErrTagMetaNotFound = errors.New("metadb: tag metadata not found for given repo and tag names") @@ -163,4 +165,5 @@ var ( ErrInvalidOutputFormat = errors.New("cli: invalid output format") ErrFlagValueUnsupported = errors.New("supported values ") ErrUnknownSubcommand = errors.New("cli: unknown subcommand") + ErrMultipleReposSameName = errors.New("test: can't have multiple repos with the same name") ) diff --git a/examples/config-all-remote.json b/examples/config-all-remote.json index a6bca61474..c953576067 100644 --- a/examples/config-all-remote.json +++ b/examples/config-all-remote.json @@ -18,7 +18,8 @@ "region": "us-east-2", "cacheTablename": "ZotBlobTable", "repoMetaTablename": "ZotRepoMetadataTable", - "manifestDataTablename": "ZotManifestDataTable", + "imageDataTablename": "ZotImageDataTable", + "repoBlobsInfoTablename": "ZotRepoBlobsInfoTable", "versionTablename": "ZotVersion" } }, diff --git a/examples/config-dynamodb.json b/examples/config-dynamodb.json index d154dc6524..b60a661d14 100644 --- a/examples/config-dynamodb.json +++ b/examples/config-dynamodb.json @@ -19,7 +19,8 @@ "region": "us-east-2", "cacheTablename": "ZotBlobTable", "repoMetaTablename": "ZotRepoMetadataTable", - "manifestDataTablename": "ZotManifestDataTable", + "imageDataTablename": "ZotImageDataTable", + "repoBlobsInfoTablename": "ZotRepoBlobsInfoTable", "userDataTablename": "ZotUserDataTable", "versionTablename": "ZotVersion" } diff --git a/go.mod b/go.mod index b773fc4767..5ae9225ae3 100644 --- a/go.mod +++ b/go.mod @@ -500,7 +500,7 @@ require ( google.golang.org/appengine v1.6.8 // indirect google.golang.org/genproto v0.0.0-20230913181813-007df8e322eb // indirect google.golang.org/grpc v1.58.2 // indirect - google.golang.org/protobuf v1.31.0 // indirect + google.golang.org/protobuf v1.31.0 gopkg.in/cheggaaa/pb.v1 v1.0.28 // indirect gopkg.in/inf.v0 v0.9.1 // indirect gopkg.in/ini.v1 v1.67.0 // indirect diff --git a/pkg/api/controller.go b/pkg/api/controller.go index fcf2d353cb..e958048d77 100644 --- a/pkg/api/controller.go +++ b/pkg/api/controller.go @@ -293,7 +293,6 @@ func (c *Controller) InitMetaDB(reloadCtx context.Context) error { return err } - // TODO: undo, commented for testing err = meta.ParseStorage(driver, c.StoreController, c.Log) if err != nil { return err diff --git a/pkg/api/controller_test.go b/pkg/api/controller_test.go index 0fdf0c90cd..d32af47d0a 100644 --- a/pkg/api/controller_test.go +++ b/pkg/api/controller_test.go @@ -153,14 +153,15 @@ func TestCreateCacheDatabaseDriver(t *testing.T) { endpoint := os.Getenv("DYNAMODBMOCK_ENDPOINT") conf.Storage.CacheDriver = map[string]interface{}{ - "name": "dynamodb", - "endpoint": endpoint, - "region": "us-east-2", - "cacheTablename": "BlobTable", - "repoMetaTablename": "RepoMetadataTable", - "manifestDataTablename": "ManifestDataTable", - "userDataTablename": "ZotUserDataTable", - "versionTablename": "Version", + "name": "dynamodb", + "endpoint": endpoint, + "region": "us-east-2", + "cacheTablename": "BlobTable", + "repoMetaTablename": "RepoMetadataTable", + "imageDataTablename": "ZotImageDataTable", + "repoBlobsInfoTablename": "ZotRepoBlobsInfoTable", + "userDataTablename": "ZotUserDataTable", + "versionTablename": "Version", } driver := storage.CreateCacheDatabaseDriver(conf.Storage.StorageConfig, log) @@ -169,27 +170,29 @@ func TestCreateCacheDatabaseDriver(t *testing.T) { // negative test cases conf.Storage.CacheDriver = map[string]interface{}{ - "endpoint": endpoint, - "region": "us-east-2", - "cacheTablename": "BlobTable", - "repoMetaTablename": "RepoMetadataTable", - "manifestDataTablename": "ManifestDataTable", - "userDataTablename": "ZotUserDataTable", - "versionTablename": "Version", + "endpoint": endpoint, + "region": "us-east-2", + "cacheTablename": "BlobTable", + "repoMetaTablename": "RepoMetadataTable", + "imageDataTablename": "ZotImageDataTable", + "repoBlobsInfoTablename": "ZotRepoBlobsInfoTable", + "userDataTablename": "ZotUserDataTable", + "versionTablename": "Version", } driver = storage.CreateCacheDatabaseDriver(conf.Storage.StorageConfig, log) So(driver, ShouldBeNil) conf.Storage.CacheDriver = map[string]interface{}{ - "name": "dummy", - "endpoint": endpoint, - "region": "us-east-2", - "cacheTablename": "BlobTable", - "repoMetaTablename": "RepoMetadataTable", - "manifestDataTablename": "ManifestDataTable", - "userDataTablename": "ZotUserDataTable", - "versionTablename": "Version", + "name": "dummy", + "endpoint": endpoint, + "region": "us-east-2", + "cacheTablename": "BlobTable", + "repoMetaTablename": "RepoMetadataTable", + "imageDataTablename": "ZotImageDataTable", + "repoBlobsInfoTablename": "ZotRepoBlobsInfoTable", + "userDataTablename": "ZotUserDataTable", + "versionTablename": "Version", } driver = storage.CreateCacheDatabaseDriver(conf.Storage.StorageConfig, log) @@ -215,43 +218,45 @@ func TestCreateMetaDBDriver(t *testing.T) { } conf.Storage.CacheDriver = map[string]interface{}{ - "name": "dummy", - "endpoint": "http://localhost:4566", - "region": "us-east-2", - "cachetablename": "BlobTable", - "repometatablename": "RepoMetadataTable", - "manifestdatatablename": "ManifestDataTable", - "userdatatablename": "UserDatatable", + "name": "dummy", + "endpoint": "http://localhost:4566", + "region": "us-east-2", + "cachetablename": "BlobTable", + "repometatablename": "RepoMetadataTable", + "imageDataTablename": "ZotImageDataTable", + "repoBlobsInfoTablename": "ZotRepoBlobsInfoTable", + "userdatatablename": "UserDatatable", } testFunc := func() { _, _ = meta.New(conf.Storage.StorageConfig, log) } So(testFunc, ShouldPanic) conf.Storage.CacheDriver = map[string]interface{}{ - "name": "dummy", - "endpoint": "http://localhost:4566", - "region": "us-east-2", - "cachetablename": "", - "repometatablename": "RepoMetadataTable", - "manifestdatatablename": "ManifestDataTable", - "userDataTablename": "ZotUserDataTable", - "versiontablename": 1, + "name": "dummy", + "endpoint": "http://localhost:4566", + "region": "us-east-2", + "cachetablename": "", + "repometatablename": "RepoMetadataTable", + "imageDataTablename": "ZotImageDataTable", + "repoBlobsInfoTablename": "ZotRepoBlobsInfoTable", + "userDataTablename": "ZotUserDataTable", + "versiontablename": 1, } testFunc = func() { _, _ = meta.New(conf.Storage.StorageConfig, log) } So(testFunc, ShouldPanic) conf.Storage.CacheDriver = map[string]interface{}{ - "name": "dummy", - "endpoint": "http://localhost:4566", - "region": "us-east-2", - "cachetablename": "test", - "repometatablename": "RepoMetadataTable", - "manifestdatatablename": "ManifestDataTable", - "indexdatatablename": "IndexDataTable", - "userdatatablename": "ZotUserDataTable", - "apikeytablename": "APIKeyTable", - "versiontablename": "1", + "name": "dummy", + "endpoint": "http://localhost:4566", + "region": "us-east-2", + "cachetablename": "test", + "repometatablename": "RepoMetadataTable", + "imagedatatablename": "ZotImageDataTable", + "repoblobsinfotablename": "ZotRepoBlobsInfoTable", + "userdatatablename": "ZotUserDataTable", + "apikeytablename": "APIKeyTable", + "versiontablename": "1", } testFunc = func() { _, _ = meta.New(conf.Storage.StorageConfig, log) } @@ -424,16 +429,16 @@ func TestObjectStorageController(t *testing.T) { conf.Storage.StorageDriver = storageDriverParams conf.Storage.CacheDriver = map[string]interface{}{ - "name": "dynamodb", - "endpoint": os.Getenv("DYNAMODBMOCK_ENDPOINT"), - "region": "us-east-2", - "cachetablename": "test", - "repometatablename": "RepoMetadataTable", - "manifestdatatablename": "ManifestDataTable", - "indexdatatablename": "IndexDataTable", - "userdatatablename": "ZotUserDataTable", - "apikeytablename": "APIKeyTable1", - "versiontablename": "Version", + "name": "dynamodb", + "endpoint": os.Getenv("DYNAMODBMOCK_ENDPOINT"), + "region": "us-east-2", + "cachetablename": "test", + "repometatablename": "RepoMetadataTable", + "imagedatatablename": "ZotImageDataTable", + "repoblobsinfotablename": "ZotRepoBlobsInfoTable", + "userdatatablename": "ZotUserDataTable", + "apikeytablename": "APIKeyTable1", + "versiontablename": "Version", } mockOIDCServer, err := authutils.MockOIDCRun() @@ -7715,6 +7720,8 @@ func TestInjectTooManyOpenFiles(t *testing.T) { } func TestGCSignaturesAndUntaggedManifestsWithMetaDB(t *testing.T) { + ctx := context.Background() + Convey("Make controller", t, func() { Convey("Garbage collect signatures without subject and manifests without tags", func(c C) { repoName := "testrepo" //nolint:goconst @@ -7746,7 +7753,7 @@ func TestGCSignaturesAndUntaggedManifestsWithMetaDB(t *testing.T) { ctlr.Config.Storage.Dedupe = false cm := test.NewControllerManager(ctlr) - cm.StartServer() + cm.StartServer() //nolint: contextcheck cm.WaitServerToBeReady(port) defer cm.StopServer() @@ -7777,7 +7784,7 @@ func TestGCSignaturesAndUntaggedManifestsWithMetaDB(t *testing.T) { // generate a keypair os.Setenv("COSIGN_PASSWORD", "") - err = generate.GenerateKeyPairCmd(context.TODO(), "", "cosign", nil) + err = generate.GenerateKeyPairCmd(ctx, "", "cosign", nil) So(err, ShouldBeNil) image := fmt.Sprintf("localhost:%s/%s@%s", port, repoName, digest.String()) @@ -7806,7 +7813,7 @@ func TestGCSignaturesAndUntaggedManifestsWithMetaDB(t *testing.T) { So(err, ShouldBeNil) // sign the image - err = signature.SignWithNotation("good", image, tdir, true) + err = signature.SignWithNotation("good", image, tdir, true) //nolint: contextcheck So(err, ShouldBeNil) // get cosign signature manifest @@ -7836,7 +7843,7 @@ func TestGCSignaturesAndUntaggedManifestsWithMetaDB(t *testing.T) { So(err, ShouldBeNil) // make sure both signatures are stored in repodb - repoMeta, err := ctlr.MetaDB.GetRepoMeta(repoName) + repoMeta, err := ctlr.MetaDB.GetRepoMeta(ctx, repoName) So(err, ShouldBeNil) sigMeta := repoMeta.Signatures[digest.String()] @@ -7897,7 +7904,7 @@ func TestGCSignaturesAndUntaggedManifestsWithMetaDB(t *testing.T) { So(err, ShouldBeNil) // make sure repoDB reference was added - repoMeta, err := ctlr.MetaDB.GetRepoMeta(repoName) + repoMeta, err := ctlr.MetaDB.GetRepoMeta(ctx, repoName) So(err, ShouldBeNil) _, ok := repoMeta.Referrers[untaggedManifestDigest.String()] @@ -7927,7 +7934,7 @@ func TestGCSignaturesAndUntaggedManifestsWithMetaDB(t *testing.T) { So(err, ShouldBeNil) // make sure both signatures are removed from repodb and repo reference for untagged is removed - repoMeta, err = ctlr.MetaDB.GetRepoMeta(repoName) + repoMeta, err = ctlr.MetaDB.GetRepoMeta(ctx, repoName) So(err, ShouldBeNil) sigMeta := repoMeta.Signatures[digest.String()] diff --git a/pkg/api/routes.go b/pkg/api/routes.go index ec93ffdc59..ef6516ecc3 100644 --- a/pkg/api/routes.go +++ b/pkg/api/routes.go @@ -516,7 +516,7 @@ func (rh *RouteHandler) GetManifest(response http.ResponseWriter, request *http. } if rh.c.MetaDB != nil { - err := meta.OnGetManifest(name, reference, content, rh.c.StoreController, rh.c.MetaDB, rh.c.Log) + err := meta.OnGetManifest(name, reference, mediaType, content, rh.c.StoreController, rh.c.MetaDB, rh.c.Log) if err != nil { response.WriteHeader(http.StatusInternalServerError) diff --git a/pkg/cli/client/cve_cmd_test.go b/pkg/cli/client/cve_cmd_test.go index ebb0c817cf..17f6cffb9c 100644 --- a/pkg/cli/client/cve_cmd_test.go +++ b/pkg/cli/client/cve_cmd_test.go @@ -6,7 +6,6 @@ package client_test import ( "bytes" "context" - "encoding/json" "errors" "fmt" "io" @@ -733,7 +732,7 @@ func getMockCveScanner(metaDB mTypes.MetaDB) cveinfo.Scanner { imageDir := repo inputTag := reference - repoMeta, err := metaDB.GetRepoMeta(imageDir) + repoMeta, err := metaDB.GetRepoMeta(context.Background(), imageDir) if err != nil { return false, err } @@ -756,19 +755,12 @@ func getMockCveScanner(metaDB mTypes.MetaDB) cveinfo.Scanner { return false, err } - manifestData, err := metaDB.GetManifestData(manifestDigest) + manifestData, err := metaDB.GetImageData(manifestDigest) if err != nil { return false, err } - var manifestContent ispec.Manifest - - err = json.Unmarshal(manifestData.ManifestBlob, &manifestContent) - if err != nil { - return false, zerr.ErrScanNotSupported - } - - for _, imageLayer := range manifestContent.Layers { + for _, imageLayer := range manifestData.Manifests[0].Manifest.Layers { switch imageLayer.MediaType { case ispec.MediaTypeImageLayerGzip, ispec.MediaTypeImageLayer, string(regTypes.DockerLayer): diff --git a/pkg/common/common.go b/pkg/common/common.go index 6081ab7dd2..e374e21fcc 100644 --- a/pkg/common/common.go +++ b/pkg/common/common.go @@ -29,6 +29,12 @@ const ( ArtifactTypeNotation = "application/vnd.cncf.notary.signature" ) +var cosignTagRule = regexp.MustCompile(`sha256\-.+\.sig`) + +func IsCosignTag(tag string) bool { + return cosignTagRule.MatchString(tag) +} + func Contains[T comparable](elems []T, v T) bool { for _, s := range elems { if v == s { diff --git a/pkg/common/oci.go b/pkg/common/oci.go index 4981aac457..f4baf303cf 100644 --- a/pkg/common/oci.go +++ b/pkg/common/oci.go @@ -6,10 +6,8 @@ import ( "github.com/opencontainers/go-digest" ispec "github.com/opencontainers/image-spec/specs-go/v1" - "google.golang.org/protobuf/types/known/timestamppb" zerr "zotregistry.io/zot/errors" - "zotregistry.io/zot/pkg/meta/proto_go" ) func GetImageDirAndTag(imageName string) (string, string) { @@ -86,24 +84,6 @@ func GetImageLastUpdated(imageInfo ispec.Image) time.Time { return *timeStamp } -func GetProtoImageLastUpdated(configData *proto_go.ConfigData) time.Time { - timeStamp := configData.Created - - if timeStamp != nil && !timeStamp.AsTime().IsZero() { - return timeStamp.AsTime() - } - - if len(configData.History) > 0 { - timeStamp = configData.History[len(configData.History)-1].Created - } - - if timeStamp == nil { - timeStamp = timestamppb.New(time.Time{}) - } - - return timeStamp.AsTime() -} - // GetRepoReference returns the components of a repoName:tag or repoName@digest string. If the format is wrong // an error is returned. // The returned values have the following meaning: diff --git a/pkg/extensions/extension_image_trust_test.go b/pkg/extensions/extension_image_trust_test.go index af3b24fca6..9d89929ed2 100644 --- a/pkg/extensions/extension_image_trust_test.go +++ b/pkg/extensions/extension_image_trust_test.go @@ -131,23 +131,23 @@ func TestSignatureUploadAndVerificationAWS(t *testing.T) { cacheTablename := "BlobTable" + uuid.String() repoMetaTablename := "RepoMetadataTable" + uuid.String() - manifestDataTablename := "ManifestDataTable" + uuid.String() versionTablename := "Version" + uuid.String() - indexDataTablename := "IndexDataTable" + uuid.String() userDataTablename := "UserDataTable" + uuid.String() apiKeyTablename := "ApiKeyTable" + uuid.String() + imageDataTablename := "imageDataTable" + uuid.String() + repoBlobsInfoTablename := "repoBlobsInfoTable" + uuid.String() cacheDriverParams := map[string]interface{}{ - "name": "dynamoDB", - "endpoint": os.Getenv("DYNAMODBMOCK_ENDPOINT"), - "region": "us-east-2", - "cacheTablename": cacheTablename, - "repoMetaTablename": repoMetaTablename, - "manifestDataTablename": manifestDataTablename, - "indexDataTablename": indexDataTablename, - "userDataTablename": userDataTablename, - "apiKeyTablename": apiKeyTablename, - "versionTablename": versionTablename, + "name": "dynamoDB", + "endpoint": os.Getenv("DYNAMODBMOCK_ENDPOINT"), + "region": "us-east-2", + "cacheTablename": cacheTablename, + "repoMetaTablename": repoMetaTablename, + "imageDataTablename": imageDataTablename, + "repoBlobsInfoTablename": repoBlobsInfoTablename, + "userDataTablename": userDataTablename, + "apiKeyTablename": apiKeyTablename, + "versionTablename": versionTablename, } t.Logf("using dynamo driver options: %v", cacheDriverParams) diff --git a/pkg/extensions/imagetrust/image_trust.go b/pkg/extensions/imagetrust/image_trust.go index b29208ae3a..e197d1f42e 100644 --- a/pkg/extensions/imagetrust/image_trust.go +++ b/pkg/extensions/imagetrust/image_trust.go @@ -5,7 +5,6 @@ package imagetrust import ( "context" - "encoding/json" "time" "github.com/aws/aws-sdk-go-v2/aws" @@ -157,8 +156,8 @@ func IsResourceExistsException(err error) bool { return false } -func (imgTrustStore *ImageTrustStore) ProtoVerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, imageData mTypes.ImageData2, +func (imgTrustStore *ImageTrustStore) VerifySignature( + signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, imageData mTypes.ImageData, repo string, ) (string, time.Time, bool, error) { desc := ispec.Descriptor{ @@ -183,42 +182,9 @@ func (imgTrustStore *ImageTrustStore) ProtoVerifySignature( } } -func (imgTrustStore *ImageTrustStore) VerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, blob []byte, - repo string, -) (string, time.Time, bool, error) { - var manifest ispec.Manifest - - err := json.Unmarshal(blob, &manifest) - if err != nil { - return "", time.Time{}, false, err - } - - desc := ispec.Descriptor{ - MediaType: manifest.MediaType, - Digest: manifestDigest, - Size: int64(len(blob)), - } - - if manifestDigest.String() == "" { - return "", time.Time{}, false, zerr.ErrBadManifestDigest - } - - switch signatureType { - case zcommon.CosignSignature: - author, isValid, err := VerifyCosignSignature(imgTrustStore.CosignStorage, repo, manifestDigest, sigKey, rawSignature) - - return author, time.Time{}, isValid, err - case zcommon.NotationSignature: - return VerifyNotationSignature(imgTrustStore.NotationStorage, desc, manifestDigest.String(), rawSignature, sigKey) - default: - return "", time.Time{}, false, zerr.ErrInvalidSignatureType - } -} - func NewTaskGenerator(metaDB mTypes.MetaDB, log log.Logger) scheduler.TaskGenerator { return &sigValidityTaskGenerator{ - repos: []mTypes.RepoMetadata{}, + repos: []mTypes.RepoMeta{}, metaDB: metaDB, repoIndex: -1, log: log, @@ -226,7 +192,7 @@ func NewTaskGenerator(metaDB mTypes.MetaDB, log log.Logger) scheduler.TaskGenera } type sigValidityTaskGenerator struct { - repos []mTypes.RepoMetadata + repos []mTypes.RepoMeta metaDB mTypes.MetaDB repoIndex int done bool @@ -237,7 +203,7 @@ func (gen *sigValidityTaskGenerator) Next() (scheduler.Task, error) { if len(gen.repos) == 0 { ctx := context.Background() - repos, err := gen.metaDB.GetMultipleRepoMeta(ctx, func(repoMeta mTypes.RepoMetadata) bool { + repos, err := gen.metaDB.GetMultipleRepoMeta(ctx, func(repoMeta mTypes.RepoMeta) bool { return true }) if err != nil { @@ -271,18 +237,18 @@ func (gen *sigValidityTaskGenerator) IsReady() bool { func (gen *sigValidityTaskGenerator) Reset() { gen.done = false gen.repoIndex = -1 - gen.repos = []mTypes.RepoMetadata{} + gen.repos = []mTypes.RepoMeta{} gen.log.Info().Msg("finished resetting task generator for updating signatures validity") } type validityTask struct { metaDB mTypes.MetaDB - repo mTypes.RepoMetadata + repo mTypes.RepoMeta log log.Logger } -func NewValidityTask(metaDB mTypes.MetaDB, repo mTypes.RepoMetadata, log log.Logger) *validityTask { +func NewValidityTask(metaDB mTypes.MetaDB, repo mTypes.RepoMeta, log log.Logger) *validityTask { return &validityTask{metaDB, repo, log} } diff --git a/pkg/extensions/imagetrust/image_trust_disabled.go b/pkg/extensions/imagetrust/image_trust_disabled.go index b2390970ee..cd555d2011 100644 --- a/pkg/extensions/imagetrust/image_trust_disabled.go +++ b/pkg/extensions/imagetrust/image_trust_disabled.go @@ -22,14 +22,7 @@ func NewAWSImageTrustStore(region, endpoint string) (*imageTrustDisabled, error) type imageTrustDisabled struct{} func (imgTrustStore *imageTrustDisabled) VerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, manifestContent []byte, - repo string, -) (string, time.Time, bool, error) { - return "", time.Time{}, false, nil -} - -func (imgTrustStore *imageTrustDisabled) ProtoVerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, imageData mTypes.ImageData2, + signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, imageData mTypes.ImageData, repo string, ) (string, time.Time, bool, error) { return "", time.Time{}, false, nil diff --git a/pkg/extensions/imagetrust/image_trust_disabled_test.go b/pkg/extensions/imagetrust/image_trust_disabled_test.go index 1140f7cc70..6abb28a010 100644 --- a/pkg/extensions/imagetrust/image_trust_disabled_test.go +++ b/pkg/extensions/imagetrust/image_trust_disabled_test.go @@ -27,15 +27,13 @@ func TestImageTrust(t *testing.T) { repo := "repo" - image := CreateRandomImage() //nolint:staticcheck - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest + image := CreateRandomImage() localImgTrustStore, err := imagetrust.NewLocalImageTrustStore(rootDir) So(err, ShouldBeNil) author, expTime, ok, err := localImgTrustStore.VerifySignature("cosign", - []byte(""), "", manifestDigest, manifestContent, repo, + []byte(""), "", image.Digest(), image.AsImageData(), repo, ) So(author, ShouldBeEmpty) So(expTime, ShouldBeZeroValue) @@ -54,7 +52,7 @@ func TestImageTrust(t *testing.T) { So(err, ShouldBeNil) author, expTime, ok, err = cloudImgTrustStore.VerifySignature("cosign", - []byte(""), "", manifestDigest, manifestContent, repo, + []byte(""), "", image.Digest(), image.AsImageData(), repo, ) So(author, ShouldBeEmpty) So(expTime, ShouldBeZeroValue) diff --git a/pkg/extensions/imagetrust/image_trust_test.go b/pkg/extensions/imagetrust/image_trust_test.go index c3db0b1983..c6a1d32e17 100644 --- a/pkg/extensions/imagetrust/image_trust_test.go +++ b/pkg/extensions/imagetrust/image_trust_test.go @@ -152,21 +152,19 @@ func TestInitCosignAndNotationDirs(t *testing.T) { func TestVerifySignatures(t *testing.T) { Convey("empty manifest digest", t, func() { image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data imgTrustStore := &imagetrust.ImageTrustStore{} - _, _, _, err := imgTrustStore.VerifySignature("", []byte(""), "", "", manifestContent, "repo") + _, _, _, err := imgTrustStore.VerifySignature("", []byte(""), "", "", image.AsImageData(), "repo") So(err, ShouldNotBeNil) So(err, ShouldEqual, zerr.ErrBadManifestDigest) }) Convey("wrong signature type", t, func() { image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest imgTrustStore := &imagetrust.ImageTrustStore{} - _, _, _, err := imgTrustStore.VerifySignature("wrongType", []byte(""), "", manifestDigest, manifestContent, "repo") + _, _, _, err := imgTrustStore.VerifySignature("wrongType", []byte(""), "", image.Digest(), image.AsImageData(), + "repo") So(err, ShouldNotBeNil) So(err, ShouldEqual, zerr.ErrInvalidSignatureType) }) @@ -176,15 +174,13 @@ func TestVerifySignatures(t *testing.T) { tag := "test" //nolint:goconst image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest Convey("cosignDir is not set", func() { imgTrustStore := &imagetrust.ImageTrustStore{ CosignStorage: &imagetrust.PublicKeyLocalStorage{}, } - _, _, _, err := imgTrustStore.VerifySignature("cosign", []byte(""), "", manifestDigest, manifestContent, repo) + _, _, _, err := imgTrustStore.VerifySignature("cosign", []byte(""), "", image.Digest(), image.AsImageData(), repo) So(err, ShouldNotBeNil) So(err, ShouldEqual, zerr.ErrSignConfigDirNotSet) }) @@ -204,7 +200,7 @@ func TestVerifySignatures(t *testing.T) { CosignStorage: pubKeyStorage, } - _, _, _, err = imgTrustStore.VerifySignature("cosign", []byte(""), "", manifestDigest, manifestContent, repo) + _, _, _, err = imgTrustStore.VerifySignature("cosign", []byte(""), "", image.Digest(), image.AsImageData(), repo) So(err, ShouldNotBeNil) }) @@ -224,8 +220,8 @@ func TestVerifySignatures(t *testing.T) { CosignStorage: pubKeyStorage, } - _, _, isTrusted, err := imgTrustStore.VerifySignature("cosign", []byte(""), "", manifestDigest, - manifestContent, repo) + _, _, isTrusted, err := imgTrustStore.VerifySignature("cosign", []byte(""), "", image.Digest(), image.AsImageData(), + repo) So(err, ShouldBeNil) So(isTrusted, ShouldBeFalse) }) @@ -274,7 +270,7 @@ func TestVerifySignatures(t *testing.T) { AnnotationOptions: options.AnnotationOptions{Annotations: []string{fmt.Sprintf("tag=%s", tag)}}, Upload: true, }, - []string{fmt.Sprintf("localhost:%s/%s@%s", port, repo, manifestDigest.String())}) + []string{fmt.Sprintf("localhost:%s/%s@%s", port, repo, image.DigestStr())}) So(err, ShouldBeNil) err = os.Remove(path.Join(cosignDir, "cosign.key")) @@ -291,7 +287,7 @@ func TestVerifySignatures(t *testing.T) { var sigKey string for _, manifest := range index.Manifests { - if manifest.Digest != manifestDigest { + if manifest.Digest != image.Digest() { blobContent, err := ctlr.StoreController.DefaultStore.GetBlobContent(repo, manifest.Digest) So(err, ShouldBeNil) @@ -312,8 +308,8 @@ func TestVerifySignatures(t *testing.T) { } // signature is trusted - author, _, isTrusted, err := imgTrustStore.VerifySignature("cosign", rawSignature, sigKey, manifestDigest, - manifestContent, repo) + author, _, isTrusted, err := imgTrustStore.VerifySignature("cosign", rawSignature, sigKey, image.Digest(), + image.AsImageData(), repo) So(err, ShouldBeNil) So(isTrusted, ShouldBeTrue) So(author, ShouldNotBeEmpty) @@ -324,16 +320,14 @@ func TestVerifySignatures(t *testing.T) { repo := "repo" //nolint:goconst tag := "test" //nolint:goconst image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest Convey("notationDir is not set", func() { imgTrustStore := &imagetrust.ImageTrustStore{ NotationStorage: &imagetrust.CertificateLocalStorage{}, } - _, _, _, err := imgTrustStore.VerifySignature("notation", []byte("signature"), "", manifestDigest, - manifestContent, repo) + _, _, _, err := imgTrustStore.VerifySignature("notation", []byte("signature"), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) So(err, ShouldEqual, zerr.ErrSignConfigDirNotSet) }) @@ -348,8 +342,8 @@ func TestVerifySignatures(t *testing.T) { NotationStorage: certStorage, } - _, _, isTrusted, err := imgTrustStore.VerifySignature("notation", []byte(""), "", manifestDigest, - manifestContent, repo) + _, _, isTrusted, err := imgTrustStore.VerifySignature("notation", []byte(""), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) So(isTrusted, ShouldBeFalse) }) @@ -369,8 +363,8 @@ func TestVerifySignatures(t *testing.T) { NotationStorage: certStorage, } - _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", manifestDigest, - manifestContent, repo) + _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) }) @@ -391,8 +385,8 @@ func TestVerifySignatures(t *testing.T) { NotationStorage: certStorage, } - _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", manifestDigest, manifestContent, - repo) + _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) }) @@ -429,10 +423,10 @@ func TestVerifySignatures(t *testing.T) { err = signature.GenerateNotationCerts(notationDir, "notation-sign-test") So(err, ShouldBeNil) - // sign the image - image := fmt.Sprintf("localhost:%s/%s", port, fmt.Sprintf("%s:%s", repo, tag)) + // sign the imageURL + imageURL := fmt.Sprintf("localhost:%s/%s", port, fmt.Sprintf("%s:%s", repo, tag)) - err = signature.SignWithNotation("notation-sign-test", image, notationDir, true) + err = signature.SignWithNotation("notation-sign-test", imageURL, notationDir, true) So(err, ShouldBeNil) err = test.CopyFiles(path.Join(notationDir, "notation", "truststore"), path.Join(notationDir, "truststore")) @@ -473,7 +467,7 @@ func TestVerifySignatures(t *testing.T) { var sigKey string for _, manifest := range index.Manifests { - if manifest.Digest != manifestDigest { + if manifest.Digest != image.Digest() { blobContent, err := ctlr.StoreController.DefaultStore.GetBlobContent(repo, manifest.Digest) So(err, ShouldBeNil) @@ -494,8 +488,8 @@ func TestVerifySignatures(t *testing.T) { } // signature is trusted - author, _, isTrusted, err := imgTrustStore.VerifySignature("notation", rawSignature, sigKey, manifestDigest, - manifestContent, repo) + author, _, isTrusted, err := imgTrustStore.VerifySignature("notation", rawSignature, sigKey, image.Digest(), + image.AsImageData(), repo) So(err, ShouldBeNil) So(isTrusted, ShouldBeTrue) So(author, ShouldNotBeEmpty) @@ -504,8 +498,8 @@ func TestVerifySignatures(t *testing.T) { So(err, ShouldBeNil) // signature is not trusted - author, _, isTrusted, err = imgTrustStore.VerifySignature("notation", rawSignature, sigKey, manifestDigest, - manifestContent, repo) + author, _, isTrusted, err = imgTrustStore.VerifySignature("notation", rawSignature, sigKey, image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) So(isTrusted, ShouldBeFalse) So(author, ShouldNotBeEmpty) @@ -969,9 +963,6 @@ func TestAWSTrustStore(t *testing.T) { repo := "repo" image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest - secretsManagerMock := mocks.SecretsManagerMock{ CreateSecretFn: func(ctx context.Context, params *secretsmanager.CreateSecretInput, optFns ...func(*secretsmanager.Options), @@ -993,8 +984,8 @@ func TestAWSTrustStore(t *testing.T) { NotationStorage: notationStorage, } - _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", manifestDigest, - manifestContent, repo) + _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) }) @@ -1002,9 +993,6 @@ func TestAWSTrustStore(t *testing.T) { repo := "repo" image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest - secretsManagerMock := mocks.SecretsManagerMock{ CreateSecretFn: func(ctx context.Context, params *secretsmanager.CreateSecretInput, optFns ...func(*secretsmanager.Options), @@ -1026,8 +1014,8 @@ func TestAWSTrustStore(t *testing.T) { NotationStorage: notationStorage, } - _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", manifestDigest, - manifestContent, repo) + _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) secretsManagerCacheMock = mocks.SecretsManagerCacheMock{ @@ -1043,8 +1031,8 @@ func TestAWSTrustStore(t *testing.T) { NotationStorage: notationStorage, } - _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", manifestDigest, - manifestContent, repo) + _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) secretsManagerCacheMock = mocks.SecretsManagerCacheMock{ @@ -1060,8 +1048,8 @@ func TestAWSTrustStore(t *testing.T) { NotationStorage: notationStorage, } - _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", manifestDigest, - manifestContent, repo) + _, _, _, err = imgTrustStore.VerifySignature("notation", []byte("signature"), "", image.Digest(), + image.AsImageData(), repo) So(err, ShouldNotBeNil) }) @@ -1072,22 +1060,22 @@ func TestAWSTrustStore(t *testing.T) { } repoMetaTablename := "RepoMetadataTable" + uuid.String() - manifestDataTablename := "ManifestDataTable" + uuid.String() versionTablename := "Version" + uuid.String() - indexDataTablename := "IndexDataTable" + uuid.String() userDataTablename := "UserDataTable" + uuid.String() apiKeyTablename := "ApiKeyTable" + uuid.String() + imageDataTablename := "imageDataTable" + uuid.String() + repoBlobsInfoTablename := "repoBlobsInfoTable" + uuid.String() dynamoDBDriverParams := map[string]interface{}{ - "name": "dynamoDB", - "endpoint": os.Getenv("DYNAMODBMOCK_ENDPOINT"), - "region": "us-east-2", - "repometatablename": repoMetaTablename, - "manifestdatatablename": manifestDataTablename, - "indexdatatablename": indexDataTablename, - "userdatatablename": userDataTablename, - "apikeytablename": apiKeyTablename, - "versiontablename": versionTablename, + "name": "dynamoDB", + "endpoint": os.Getenv("DYNAMODBMOCK_ENDPOINT"), + "region": "us-east-2", + "repometatablename": repoMetaTablename, + "imagedatatablename": imageDataTablename, + "repoblobsinfotablename": repoBlobsInfoTablename, + "userdatatablename": userDataTablename, + "apikeytablename": apiKeyTablename, + "versiontablename": versionTablename, } t.Logf("using dynamo driver options: %v", dynamoDBDriverParams) @@ -1229,8 +1217,6 @@ func RunVerificationTests(t *testing.T, dbDriverParams map[string]interface{}) { Convey("verify cosign signature is trusted", func() { image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest err = UploadImage(image, baseURL, repo, tag) So(err, ShouldBeNil) @@ -1256,7 +1242,7 @@ func RunVerificationTests(t *testing.T, dbDriverParams map[string]interface{}) { AnnotationOptions: options.AnnotationOptions{Annotations: []string{fmt.Sprintf("tag=%s", tag)}}, Upload: true, }, - []string{fmt.Sprintf("localhost:%s/%s@%s", port, repo, manifestDigest.String())}) + []string{fmt.Sprintf("localhost:%s/%s@%s", port, repo, image.DigestStr())}) So(err, ShouldBeNil) indexContent, err := ctlr.StoreController.DefaultStore.GetIndexContent(repo) @@ -1270,7 +1256,7 @@ func RunVerificationTests(t *testing.T, dbDriverParams map[string]interface{}) { var sigKey string for _, manifest := range index.Manifests { - if manifest.Digest != manifestDigest { + if manifest.Digest != image.Digest() { blobContent, err := ctlr.StoreController.DefaultStore.GetBlobContent(repo, manifest.Digest) So(err, ShouldBeNil) @@ -1300,8 +1286,8 @@ func RunVerificationTests(t *testing.T, dbDriverParams map[string]interface{}) { imageTrustStore := ctlr.MetaDB.ImageTrustStore() // signature is trusted - author, _, isTrusted, err := imageTrustStore.VerifySignature("cosign", rawSignature, sigKey, manifestDigest, - manifestContent, repo) + author, _, isTrusted, err := imageTrustStore.VerifySignature("cosign", rawSignature, sigKey, image.Digest(), + image.AsImageData(), repo) So(err, ShouldBeNil) So(isTrusted, ShouldBeTrue) So(author, ShouldNotBeEmpty) @@ -1309,8 +1295,6 @@ func RunVerificationTests(t *testing.T, dbDriverParams map[string]interface{}) { Convey("verify notation signature is trusted", func() { image := CreateRandomImage() - manifestContent := image.ManifestDescriptor.Data - manifestDigest := image.ManifestDescriptor.Digest err = UploadImage(image, baseURL, repo, tag) So(err, ShouldBeNil) @@ -1358,7 +1342,7 @@ func RunVerificationTests(t *testing.T, dbDriverParams map[string]interface{}) { t.Logf("Processing manifest %v", notationSig) if notationSig.Config.MediaType != notreg.ArtifactTypeNotation || - notationSig.Subject.Digest != manifestDigest { + notationSig.Subject.Digest != image.Digest() { continue } @@ -1392,8 +1376,8 @@ func RunVerificationTests(t *testing.T, dbDriverParams map[string]interface{}) { imageTrustStore := ctlr.MetaDB.ImageTrustStore() // signature is trusted - author, _, isTrusted, err := imageTrustStore.VerifySignature("notation", rawSignature, sigKey, manifestDigest, - manifestContent, repo) + author, _, isTrusted, err := imageTrustStore.VerifySignature("notation", rawSignature, sigKey, image.Digest(), + image.AsImageData(), repo) So(err, ShouldBeNil) So(isTrusted, ShouldBeTrue) So(author, ShouldEqual, "CN=cert,O=Notary,L=Seattle,ST=WA,C=US") diff --git a/pkg/extensions/search/convert/convert_internal_test.go b/pkg/extensions/search/convert/convert_internal_test.go index fe99ae68c7..9dfb691514 100644 --- a/pkg/extensions/search/convert/convert_internal_test.go +++ b/pkg/extensions/search/convert/convert_internal_test.go @@ -4,12 +4,10 @@ package convert import ( "context" - "encoding/json" "errors" "testing" "github.com/99designs/gqlgen/graphql" - godigest "github.com/opencontainers/go-digest" ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" @@ -17,7 +15,7 @@ import ( "zotregistry.io/zot/pkg/extensions/search/gql_generated" "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta/boltdb" - mTypes "zotregistry.io/zot/pkg/meta/types" + . "zotregistry.io/zot/pkg/test/image-utils" "zotregistry.io/zot/pkg/test/mocks" ) @@ -34,33 +32,20 @@ func TestCVEConvert(t *testing.T) { metaDB, err := boltdb.New(boltDB, log.NewLogger("debug", "")) So(err, ShouldBeNil) - configBlob, err := json.Marshal(ispec.Image{}) - So(err, ShouldBeNil) + image := CreateImageWith(). + Layers([]Layer{{ + MediaType: ispec.MediaTypeImageLayerGzip, + Digest: ispec.MediaTypeEmptyJSON, + Blob: ispec.DescriptorEmptyJSON.Data, + }}).DefaultConfig().Build() - manifestBlob, err := json.Marshal(ispec.Manifest{ - Layers: []ispec.Descriptor{ - { - MediaType: ispec.MediaTypeImageLayerGzip, - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), - }, - }, - }) + err = metaDB.SetRepoReference("repo1", "0.1.0", image.AsImageData()) So(err, ShouldBeNil) - repoMeta11 := mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - } - - digest11 := godigest.FromString("abc1") - err = metaDB.SetManifestMeta("repo1", digest11, repoMeta11) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo1", "0.1.0", digest11, ispec.MediaTypeImageManifest) + repoMetaList, err := metaDB.SearchRepos(context.Background(), "") So(err, ShouldBeNil) - reposMeta, manifestMetaMap, _, err := metaDB.SearchRepos(context.Background(), "") - So(err, ShouldBeNil) + imageData, err := metaDB.FilterImageData(context.Background(), []string{image.DigestStr()}) ctx := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) @@ -86,8 +71,8 @@ func TestCVEConvert(t *testing.T) { So(imageSummary, ShouldBeNil) So(graphql.GetErrors(ctx), ShouldBeNil) - imageSummary, _, err = ImageManifest2ImageSummary(ctx, "repo1", "0.1.0", digest11, reposMeta[0], - manifestMetaMap[digest11.String()]) + imageSummary, _, err = ImageManifest2ImageSummary(ctx, GetImageMeta("0.1.0", repoMetaList[0], + imageData[image.DigestStr()])) So(err, ShouldBeNil) So(imageSummary, ShouldNotBeNil) @@ -177,8 +162,8 @@ func TestCVEConvert(t *testing.T) { So(repoSummary, ShouldBeNil) So(graphql.GetErrors(ctx), ShouldBeNil) - imageSummary, _, err := ImageManifest2ImageSummary(ctx, "repo1", "0.1.0", digest11, reposMeta[0], - manifestMetaMap[digest11.String()]) + imageSummary, _, err := ImageManifest2ImageSummary(ctx, GetImageMeta("0.1.0", repoMetaList[0], + imageData[image.DigestStr()])) So(err, ShouldBeNil) So(imageSummary, ShouldNotBeNil) @@ -235,8 +220,8 @@ func TestCVEConvert(t *testing.T) { So(manifestSummary, ShouldBeNil) So(graphql.GetErrors(ctx), ShouldBeNil) - imageSummary, _, err := ImageManifest2ImageSummary(ctx, "repo1", "0.1.0", digest11, reposMeta[0], - manifestMetaMap[digest11.String()]) + imageSummary, _, err := ImageManifest2ImageSummary(ctx, GetImageMeta("0.1.0", repoMetaList[0], + imageData[image.DigestStr()])) So(err, ShouldBeNil) manifestSummary = imageSummary.Manifests[0] diff --git a/pkg/extensions/search/convert/convert_test.go b/pkg/extensions/search/convert/convert_test.go index cc52fcfcbe..3bd7dc24a9 100644 --- a/pkg/extensions/search/convert/convert_test.go +++ b/pkg/extensions/search/convert/convert_test.go @@ -4,21 +4,19 @@ package convert_test import ( "context" - "encoding/json" "errors" "testing" "time" - "github.com/99designs/gqlgen/graphql" godigest "github.com/opencontainers/go-digest" ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" "zotregistry.io/zot/pkg/extensions/search/convert" - cvemodel "zotregistry.io/zot/pkg/extensions/search/cve/model" "zotregistry.io/zot/pkg/extensions/search/gql_generated" "zotregistry.io/zot/pkg/extensions/search/pagination" "zotregistry.io/zot/pkg/log" + "zotregistry.io/zot/pkg/meta/boltdb" mTypes "zotregistry.io/zot/pkg/meta/types" . "zotregistry.io/zot/pkg/test/image-utils" "zotregistry.io/zot/pkg/test/mocks" @@ -27,187 +25,6 @@ import ( var ErrTestError = errors.New("TestError") -func TestConvertErrors(t *testing.T) { - Convey("ImageIndex2ImageSummary errors", t, func() { - ctx := graphql.WithResponseContext(context.Background(), - graphql.DefaultErrorPresenter, graphql.DefaultRecover) - - _, _, err := convert.ImageIndex2ImageSummary( - ctx, - "repo", - "tag", - godigest.FromString("indexDigest"), - mTypes.RepoMetadata{}, - mTypes.IndexData{ - IndexBlob: []byte("bad json"), - }, - map[string]mTypes.ManifestMetadata{}, - ) - So(err, ShouldNotBeNil) - }) - - Convey("ImageIndex2ImageSummary cve scanning", t, func() { - ctx := graphql.WithResponseContext(context.Background(), - graphql.DefaultErrorPresenter, graphql.DefaultRecover) - - _, _, err := convert.ImageIndex2ImageSummary( - ctx, - "repo", - "tag", - godigest.FromString("indexDigest"), - mTypes.RepoMetadata{}, - mTypes.IndexData{ - IndexBlob: []byte("{}"), - }, - map[string]mTypes.ManifestMetadata{}, - ) - So(err, ShouldBeNil) - }) - - Convey("ImageManifest2ImageSummary", t, func() { - ctx := graphql.WithResponseContext(context.Background(), - graphql.DefaultErrorPresenter, graphql.DefaultRecover) - configBlob, err := json.Marshal(ispec.Image{ - Platform: ispec.Platform{ - OS: "os", - Architecture: "arch", - Variant: "var", - }, - }) - So(err, ShouldBeNil) - - _, _, err = convert.ImageManifest2ImageSummary( - ctx, - "repo", - "tag", - godigest.FromString("manifestDigest"), - mTypes.RepoMetadata{}, - mTypes.ManifestMetadata{ - ManifestBlob: []byte("{}"), - ConfigBlob: configBlob, - }, - ) - So(err, ShouldBeNil) - }) - - Convey("ImageManifest2ManifestSummary", t, func() { - ctx := graphql.WithResponseContext(context.Background(), - graphql.DefaultErrorPresenter, graphql.DefaultRecover) - - // with bad config json, shouldn't error when unmarshaling - _, _, _, err := convert.ImageManifest2ManifestSummary( - ctx, - "repo", - "tag", - ispec.Descriptor{ - Digest: "dig", - MediaType: ispec.MediaTypeImageManifest, - }, - mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - }, - mTypes.ManifestMetadata{ - ManifestBlob: []byte(`{}`), - ConfigBlob: []byte("bad json"), - }, - nil, - ) - So(err, ShouldBeNil) - - // CVE scan using platform - configBlob, err := json.Marshal(ispec.Image{ - Platform: ispec.Platform{ - OS: "os", - Architecture: "arch", - Variant: "var", - }, - }) - So(err, ShouldBeNil) - - _, _, _, err = convert.ImageManifest2ManifestSummary( - ctx, - "repo", - "tag", - ispec.Descriptor{ - Digest: "dig", - MediaType: ispec.MediaTypeImageManifest, - }, - mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{"dig": {"cosine": []mTypes.SignatureInfo{{}}}}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - }, - mTypes.ManifestMetadata{ - ManifestBlob: []byte("{}"), - ConfigBlob: configBlob, - }, - nil, - ) - So(err, ShouldBeNil) - }) - - Convey("RepoMeta2ExpandedRepoInfo", t, func() { - ctx := graphql.WithResponseContext(context.Background(), - graphql.DefaultErrorPresenter, graphql.DefaultRecover) - - // with bad config json, error while unmarshaling - _, imageSummaries := convert.RepoMeta2ExpandedRepoInfo( - ctx, - mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag1": {Digest: "dig", MediaType: ispec.MediaTypeImageManifest}, - }, - }, - map[string]mTypes.ManifestMetadata{ - "dig": { - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("bad json"), - }, - }, - map[string]mTypes.IndexData{}, - convert.SkipQGLField{ - Vulnerabilities: false, - }, - mocks.CveInfoMock{ - GetCVESummaryForImageMediaFn: func(repo, digest, mediaType string) (cvemodel.ImageCVESummary, error) { - return cvemodel.ImageCVESummary{}, ErrTestError - }, - }, log.NewLogger("debug", ""), - ) - So(len(imageSummaries), ShouldEqual, 1) - - // cveInfo present no error - _, imageSummaries = convert.RepoMeta2ExpandedRepoInfo( - ctx, - mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag1": {Digest: "dig", MediaType: ispec.MediaTypeImageManifest}, - }, - }, - map[string]mTypes.ManifestMetadata{ - "dig": { - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }, - }, - map[string]mTypes.IndexData{}, - convert.SkipQGLField{ - Vulnerabilities: false, - }, - mocks.CveInfoMock{ - GetCVESummaryForImageMediaFn: func(repo, digest, mediaType string) (cvemodel.ImageCVESummary, error) { - return cvemodel.ImageCVESummary{}, ErrTestError - }, - }, log.NewLogger("debug", ""), - ) - So(len(imageSummaries), ShouldEqual, 1) - }) -} - func TestUpdateLastUpdatedTimestamp(t *testing.T) { Convey("Image summary is the first image checked for the repo", t, func() { before := time.Time{} @@ -300,14 +117,25 @@ func TestLabels(t *testing.T) { func TestGetSignaturesInfo(t *testing.T) { Convey("Test get signatures info - cosign", t, func() { - indexDigest := godigest.FromString("123") - repoMeta := mTypes.RepoMetadata{ - Signatures: map[string]mTypes.ManifestSignatures{string(indexDigest): {"cosign": []mTypes.SignatureInfo{{ - LayersInfo: []mTypes.LayerInfo{{LayerContent: []byte{}, LayerDigest: "", SignatureKey: "", Signer: "author"}}, - }}}}, + digest := godigest.FromString("dig") + signatures := map[string]mTypes.ManifestSignatures{ + digest.String(): { + "cosign": []mTypes.SignatureInfo{ + { + LayersInfo: []mTypes.LayerInfo{ + { + LayerContent: []byte{}, + LayerDigest: "", + SignatureKey: "", + Signer: "author", + }, + }, + }, + }, + }, } - signaturesSummary := convert.GetSignaturesInfo(true, repoMeta, indexDigest) + signaturesSummary := convert.GetSignaturesInfo(true, signatures[digest.String()]) So(signaturesSummary, ShouldNotBeEmpty) So(*signaturesSummary[0].Author, ShouldEqual, "author") So(*signaturesSummary[0].IsTrusted, ShouldEqual, true) @@ -315,22 +143,26 @@ func TestGetSignaturesInfo(t *testing.T) { }) Convey("Test get signatures info - notation", t, func() { - indexDigest := godigest.FromString("123") - repoMeta := mTypes.RepoMetadata{ - Signatures: map[string]mTypes.ManifestSignatures{string(indexDigest): {"notation": []mTypes.SignatureInfo{{ - LayersInfo: []mTypes.LayerInfo{ + digest := godigest.FromString("dig") + signatures := map[string]mTypes.ManifestSignatures{ + digest.String(): { + "notation": []mTypes.SignatureInfo{ { - LayerContent: []byte{}, - LayerDigest: "", - SignatureKey: "", - Signer: "author", - Date: time.Now().AddDate(0, 0, -1), + LayersInfo: []mTypes.LayerInfo{ + { + LayerContent: []byte{}, + LayerDigest: "", + SignatureKey: "", + Signer: "author", + Date: time.Now().AddDate(0, 0, -1), + }, + }, }, }, - }}}}, + }, } - signaturesSummary := convert.GetSignaturesInfo(true, repoMeta, indexDigest) + signaturesSummary := convert.GetSignaturesInfo(true, signatures[digest.String()]) So(signaturesSummary, ShouldNotBeEmpty) So(*signaturesSummary[0].Author, ShouldEqual, "author") So(*signaturesSummary[0].IsTrusted, ShouldEqual, false) @@ -422,6 +254,18 @@ func ref[T any](val T) *T { func TestPaginatedConvert(t *testing.T) { ctx := context.Background() + tempDir := t.TempDir() + + driver, err := boltdb.GetBoltDriver(boltdb.DBParameters{RootDir: tempDir}) + if err != nil { + t.FailNow() + } + + metaDB, err := boltdb.New(driver, log.NewLogger("debug", "")) + if err != nil { + t.FailNow() + } + var ( badBothImage = CreateImageWith().DefaultLayers().ImageConfig( ispec.Image{Platform: ispec.Platform{OS: "bad-os", Architecture: "bad-arch"}}).Build() @@ -432,8 +276,9 @@ func TestPaginatedConvert(t *testing.T) { goodImage = CreateImageWith().DefaultLayers().ImageConfig( ispec.Image{Platform: ispec.Platform{OS: "good-os", Architecture: "good-arch"}}).Build() - randomImage1 = CreateRandomImage() - randomImage2 = CreateRandomImage() + randomImage1 = CreateRandomImage() + randomImage2 = CreateRandomImage() + signatureDigest = godigest.FromString("signature") badMultiArch = CreateMultiarchWith().Images( []Image{badBothImage, badOsImage, badArchImage, randomImage1}).Build() @@ -441,14 +286,14 @@ func TestPaginatedConvert(t *testing.T) { []Image{badOsImage, badArchImage, randomImage2, goodImage}).Build() ) - reposMeta, manifestMetaMap, indexDataMap := ociutils.GetMetadataForRepos( + ctx, err = ociutils.InitializeTestMetaDB(ctx, metaDB, ociutils.Repo{ Name: "repo1-only-images", Images: []ociutils.RepoImage{ - {Image: goodImage, Tag: "goodImage"}, - {Image: badOsImage, Tag: "badOsImage"}, - {Image: badArchImage, Tag: "badArchImage"}, - {Image: badBothImage, Tag: "badBothImage"}, + {Image: goodImage, Reference: "goodImage"}, + {Image: badOsImage, Reference: "badOsImage"}, + {Image: badArchImage, Reference: "badArchImage"}, + {Image: badBothImage, Reference: "badBothImage"}, }, IsBookmarked: true, IsStarred: true, @@ -456,9 +301,9 @@ func TestPaginatedConvert(t *testing.T) { ociutils.Repo{ Name: "repo2-only-bad-images", Images: []ociutils.RepoImage{ - {Image: randomImage1, Tag: "randomImage1"}, - {Image: randomImage2, Tag: "randomImage2"}, - {Image: badBothImage, Tag: "badBothImage"}, + {Image: randomImage1, Reference: "randomImage1"}, + {Image: randomImage2, Reference: "randomImage2"}, + {Image: badBothImage, Reference: "badBothImage"}, }, IsBookmarked: true, IsStarred: true, @@ -466,8 +311,8 @@ func TestPaginatedConvert(t *testing.T) { ociutils.Repo{ Name: "repo3-only-multiarch", MultiArchImages: []ociutils.RepoMultiArchImage{ - {MultiarchImage: badMultiArch, Tag: "badMultiArch"}, - {MultiarchImage: goodMultiArch, Tag: "goodMultiArch"}, + {MultiarchImage: badMultiArch, Reference: "badMultiArch"}, + {MultiarchImage: goodMultiArch, Reference: "goodMultiArch"}, }, IsBookmarked: true, IsStarred: true, @@ -475,44 +320,56 @@ func TestPaginatedConvert(t *testing.T) { ociutils.Repo{ Name: "repo4-not-bookmarked-or-starred", Images: []ociutils.RepoImage{ - {Image: goodImage, Tag: "goodImage"}, + {Image: goodImage, Reference: "goodImage"}, }, MultiArchImages: []ociutils.RepoMultiArchImage{ - {MultiarchImage: goodMultiArch, Tag: "goodMultiArch"}, + {MultiarchImage: goodMultiArch, Reference: "goodMultiArch"}, }, }, ociutils.Repo{ Name: "repo5-signed", Images: []ociutils.RepoImage{ - {Image: goodImage, Tag: "goodImage"}, // is fake signed by the image below - {Image: CreateFakeTestSignature(goodImage.DescriptorRef())}, + {Image: goodImage, Reference: "goodImage"}, // is fake signed by the image below + }, + Signatures: map[string]mTypes.ManifestSignatures{ + goodImage.DigestStr(): ociutils.GetFakeSignatureInfo(signatureDigest.String()), }, }, ) + if err != nil { + t.FailNow() + } skipCVE := convert.SkipQGLField{Vulnerabilities: true} Convey("PaginatedRepoMeta2RepoSummaries filtering and sorting", t, func() { // Test different combinations of the filter + repoMetaList, err := metaDB.FilterRepos(ctx, mTypes.AcceptAllRepoNames, mTypes.AcceptAllRepoMeta) + So(err, ShouldBeNil) + imageData, err := metaDB.FilterImageData(ctx, mTypes.GetLatestImageDigests(repoMetaList)) + So(err, ShouldBeNil) reposSum, pageInfo, err := convert.PaginatedRepoMeta2RepoSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + ctx, repoMetaList, imageData, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, IsBookmarked: ref(true), IsStarred: ref(true), }, - pagination.PageInput{SortBy: pagination.AlphabeticAsc}, + pagination.PageInput{SortBy: pagination.AlphabeticAsc}, mocks.CveInfoMock{}, skipCVE, ) So(err, ShouldBeNil) So(len(reposSum), ShouldEqual, 2) So(*reposSum[0].Name, ShouldResemble, "repo1-only-images") So(*reposSum[1].Name, ShouldResemble, "repo3-only-multiarch") So(pageInfo.ItemCount, ShouldEqual, 2) + So(pageInfo.ItemCount, ShouldEqual, 2) + So(pageInfo.ItemCount, ShouldEqual, 2) + So(pageInfo.ItemCount, ShouldEqual, 2) reposSum, pageInfo, err = convert.PaginatedRepoMeta2RepoSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + ctx, repoMetaList, imageData, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, @@ -520,18 +377,18 @@ func TestPaginatedConvert(t *testing.T) { IsStarred: ref(true), HasToBeSigned: ref(true), }, - pagination.PageInput{SortBy: pagination.AlphabeticAsc}, + pagination.PageInput{SortBy: pagination.AlphabeticAsc}, mocks.CveInfoMock{}, skipCVE, ) So(err, ShouldBeNil) So(len(reposSum), ShouldEqual, 0) So(pageInfo.ItemCount, ShouldEqual, 0) reposSum, pageInfo, err = convert.PaginatedRepoMeta2RepoSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + ctx, repoMetaList, imageData, mTypes.Filter{ HasToBeSigned: ref(true), }, - pagination.PageInput{SortBy: pagination.AlphabeticAsc}, + pagination.PageInput{SortBy: pagination.AlphabeticAsc}, mocks.CveInfoMock{}, skipCVE, ) So(err, ShouldBeNil) So(len(reposSum), ShouldEqual, 1) @@ -540,8 +397,8 @@ func TestPaginatedConvert(t *testing.T) { // no filter reposSum, pageInfo, err = convert.PaginatedRepoMeta2RepoSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, - mTypes.Filter{}, pagination.PageInput{SortBy: pagination.AlphabeticAsc}, + ctx, repoMetaList, imageData, + mTypes.Filter{}, pagination.PageInput{SortBy: pagination.AlphabeticAsc}, mocks.CveInfoMock{}, skipCVE, ) So(err, ShouldBeNil) So(len(reposSum), ShouldEqual, 5) @@ -554,8 +411,8 @@ func TestPaginatedConvert(t *testing.T) { // no filter opposite sorting reposSum, pageInfo, err = convert.PaginatedRepoMeta2RepoSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, - mTypes.Filter{}, pagination.PageInput{SortBy: pagination.AlphabeticDsc}, + ctx, repoMetaList, imageData, + mTypes.Filter{}, pagination.PageInput{SortBy: pagination.AlphabeticDsc}, mocks.CveInfoMock{}, skipCVE, ) So(err, ShouldBeNil) So(len(reposSum), ShouldEqual, 5) @@ -568,14 +425,14 @@ func TestPaginatedConvert(t *testing.T) { // add pagination reposSum, pageInfo, err = convert.PaginatedRepoMeta2RepoSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + ctx, repoMetaList, imageData, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, IsBookmarked: ref(true), IsStarred: ref(true), }, - pagination.PageInput{Limit: 1, Offset: 0, SortBy: pagination.AlphabeticAsc}, + pagination.PageInput{Limit: 1, Offset: 0, SortBy: pagination.AlphabeticAsc}, mocks.CveInfoMock{}, skipCVE, ) So(err, ShouldBeNil) So(len(reposSum), ShouldEqual, 1) @@ -584,14 +441,14 @@ func TestPaginatedConvert(t *testing.T) { So(pageInfo.TotalCount, ShouldEqual, 2) reposSum, pageInfo, err = convert.PaginatedRepoMeta2RepoSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + ctx, repoMetaList, imageData, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, IsBookmarked: ref(true), IsStarred: ref(true), }, - pagination.PageInput{Limit: 1, Offset: 1, SortBy: pagination.AlphabeticAsc}, + pagination.PageInput{Limit: 1, Offset: 1, SortBy: pagination.AlphabeticAsc}, mocks.CveInfoMock{}, skipCVE, ) So(err, ShouldBeNil) So(len(reposSum), ShouldEqual, 1) @@ -601,8 +458,11 @@ func TestPaginatedConvert(t *testing.T) { }) Convey("PaginatedRepoMeta2ImageSummaries filtering and sorting", t, func() { - imgSum, pageInfo, err := convert.PaginatedRepoMeta2ImageSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, mTypes.AcceptAllImageData) + So(err, ShouldBeNil) + + imgSum, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries( + ctx, imageDataList, skipCVE, mocks.CveInfoMock{}, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, @@ -624,8 +484,8 @@ func TestPaginatedConvert(t *testing.T) { So(pageInfo.ItemCount, ShouldEqual, 5) // add page of size 2 - imgSum, pageInfo, err = convert.PaginatedRepoMeta2ImageSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + imgSum, pageInfo, err = convert.PaginatedImageMeta2ImageSummaries( + ctx, imageDataList, skipCVE, mocks.CveInfoMock{}, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, @@ -642,8 +502,8 @@ func TestPaginatedConvert(t *testing.T) { So(pageInfo.TotalCount, ShouldEqual, 5) // next page - imgSum, pageInfo, err = convert.PaginatedRepoMeta2ImageSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + imgSum, pageInfo, err = convert.PaginatedImageMeta2ImageSummaries( + ctx, imageDataList, skipCVE, mocks.CveInfoMock{}, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, @@ -660,8 +520,8 @@ func TestPaginatedConvert(t *testing.T) { So(pageInfo.TotalCount, ShouldEqual, 5) // last page - imgSum, pageInfo, err = convert.PaginatedRepoMeta2ImageSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + imgSum, pageInfo, err = convert.PaginatedImageMeta2ImageSummaries( + ctx, imageDataList, skipCVE, mocks.CveInfoMock{}, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, @@ -676,8 +536,8 @@ func TestPaginatedConvert(t *testing.T) { So(pageInfo.TotalCount, ShouldEqual, 5) // has to be signed - imgSum, pageInfo, err = convert.PaginatedRepoMeta2ImageSummaries( - ctx, reposMeta, manifestMetaMap, indexDataMap, skipCVE, mocks.CveInfoMock{}, + imgSum, pageInfo, err = convert.PaginatedImageMeta2ImageSummaries( + ctx, imageDataList, skipCVE, mocks.CveInfoMock{}, mTypes.Filter{ Os: []*string{ref("good-os")}, Arch: []*string{ref("good-arch")}, @@ -697,6 +557,16 @@ func TestIndexAnnotations(t *testing.T) { Convey("Test ImageIndex2ImageSummary annotations logic", t, func() { ctx := context.Background() + tempDir := t.TempDir() + + driver, err := boltdb.GetBoltDriver(boltdb.DBParameters{RootDir: tempDir}) + if err != nil { + t.FailNow() + } + + metaDB, err := boltdb.New(driver, log.NewLogger("debug", "")) + So(err, ShouldBeNil) + configLabels := map[string]string{ ispec.AnnotationDescription: "ConfigDescription", ispec.AnnotationLicenses: "ConfigLicenses", @@ -746,17 +616,22 @@ func TestIndexAnnotations(t *testing.T) { []Image{imageWithManifestAndConfigAnnotations}, ).Annotations(indexAnnotations).Build() - repoMeta, manifestMetadata, indexData := ociutils.GetMetadataForRepos(ociutils.Repo{ - Name: "repo", - MultiArchImages: []ociutils.RepoMultiArchImage{ - {MultiarchImage: indexWithAnnotations, Tag: "tag"}, - }, - }) + ctx, err = ociutils.InitializeTestMetaDB(ctx, metaDB, + ociutils.Repo{ + Name: "repo", + MultiArchImages: []ociutils.RepoMultiArchImage{ + {MultiarchImage: indexWithAnnotations, Reference: "tag"}, + }, + }) + So(err, ShouldBeNil) - digest := indexWithAnnotations.Digest() + repoMeta, err := metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldBeNil) + imageData, err := metaDB.FilterImageData(ctx, []string{indexWithAnnotations.DigestStr()}) + So(err, ShouldBeNil) - imageSummary, _, err := convert.ImageIndex2ImageSummary(ctx, "repo", "tag", digest, repoMeta[0], - indexData[digest.String()], manifestMetadata) + imageSummary, _, err := convert.ImageIndex2ImageSummary(ctx, convert.GetImageMeta("tag", repoMeta, + imageData[indexWithAnnotations.DigestStr()])) So(err, ShouldBeNil) So(*imageSummary.Description, ShouldResemble, "IndexDescription") So(*imageSummary.Licenses, ShouldResemble, "IndexLicenses") @@ -766,19 +641,30 @@ func TestIndexAnnotations(t *testing.T) { So(*imageSummary.Vendor, ShouldResemble, "IndexVendor") So(*imageSummary.Authors, ShouldResemble, "IndexAuthors") + err = metaDB.ResetDB() + So(err, ShouldBeNil) // -------------------------------------------------------- indexWithManifestAndConfigAnnotations := CreateMultiarchWith().Images( []Image{imageWithManifestAndConfigAnnotations, CreateRandomImage(), CreateRandomImage()}, ).Build() - repoMeta, manifestMetadata, indexData = ociutils.GetMetadataForRepos(ociutils.Repo{ - Name: "repo", - MultiArchImages: []ociutils.RepoMultiArchImage{{MultiarchImage: indexWithManifestAndConfigAnnotations}}, + ctx, err = ociutils.InitializeTestMetaDB(ctx, metaDB, ociutils.Repo{ + Name: "repo", + MultiArchImages: []ociutils.RepoMultiArchImage{ + {MultiarchImage: indexWithManifestAndConfigAnnotations, Reference: "tag"}, + }, }) - digest = indexWithManifestAndConfigAnnotations.Digest() + So(err, ShouldBeNil) + + digest := indexWithManifestAndConfigAnnotations.DigestStr() + + repoMeta, err = metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldBeNil) + imageData, err = metaDB.FilterImageData(ctx, []string{digest}) + So(err, ShouldBeNil) - imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, "repo", "tag", digest, - repoMeta[0], indexData[digest.String()], manifestMetadata) + imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, convert.GetImageMeta("tag", repoMeta, + imageData[digest])) So(err, ShouldBeNil) So(*imageSummary.Description, ShouldResemble, "ManifestDescription") So(*imageSummary.Licenses, ShouldResemble, "ManifestLicenses") @@ -787,19 +673,31 @@ func TestIndexAnnotations(t *testing.T) { So(*imageSummary.Documentation, ShouldResemble, "ManifestDocumentation") So(*imageSummary.Vendor, ShouldResemble, "ManifestVendor") So(*imageSummary.Authors, ShouldResemble, "ManifestAuthors") + + err = metaDB.ResetDB() + So(err, ShouldBeNil) // -------------------------------------------------------- indexWithConfigAnnotations := CreateMultiarchWith().Images( []Image{imageWithConfigAnnotations, CreateRandomImage(), CreateRandomImage()}, ).Build() - repoMeta, manifestMetadata, indexData = ociutils.GetMetadataForRepos(ociutils.Repo{ - Name: "repo", - MultiArchImages: []ociutils.RepoMultiArchImage{{MultiarchImage: indexWithConfigAnnotations, Tag: "tag"}}, + ctx, err = ociutils.InitializeTestMetaDB(ctx, metaDB, ociutils.Repo{ + Name: "repo", + MultiArchImages: []ociutils.RepoMultiArchImage{ + {MultiarchImage: indexWithConfigAnnotations, Reference: "tag"}, + }, }) - digest = indexWithConfigAnnotations.Digest() + So(err, ShouldBeNil) + + digest = indexWithConfigAnnotations.DigestStr() - imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, "repo", "tag", digest, - repoMeta[0], indexData[digest.String()], manifestMetadata) + repoMeta, err = metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldBeNil) + imageData, err = metaDB.FilterImageData(ctx, []string{digest}) + So(err, ShouldBeNil) + + imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, convert.GetImageMeta("tag", repoMeta, + imageData[digest])) So(err, ShouldBeNil) So(*imageSummary.Description, ShouldResemble, "ConfigDescription") So(*imageSummary.Licenses, ShouldResemble, "ConfigLicenses") @@ -808,6 +706,9 @@ func TestIndexAnnotations(t *testing.T) { So(*imageSummary.Documentation, ShouldResemble, "ConfigDocumentation") So(*imageSummary.Vendor, ShouldResemble, "ConfigVendor") So(*imageSummary.Authors, ShouldResemble, "ConfigAuthors") + + err = metaDB.ResetDB() + So(err, ShouldBeNil) //-------------------------------------------------------- indexWithMixAnnotations := CreateMultiarchWith().Images( @@ -834,14 +735,23 @@ func TestIndexAnnotations(t *testing.T) { }, ).Build() - repoMeta, manifestMetadata, indexData = ociutils.GetMetadataForRepos(ociutils.Repo{ - Name: "repo", - MultiArchImages: []ociutils.RepoMultiArchImage{{MultiarchImage: indexWithMixAnnotations, Tag: "tag"}}, + ctx, err = ociutils.InitializeTestMetaDB(ctx, metaDB, ociutils.Repo{ + Name: "repo", + MultiArchImages: []ociutils.RepoMultiArchImage{ + {MultiarchImage: indexWithMixAnnotations, Reference: "tag"}, + }, }) - digest = indexWithMixAnnotations.Digest() + So(err, ShouldBeNil) + + digest = indexWithMixAnnotations.DigestStr() + + repoMeta, err = metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldBeNil) + imageData, err = metaDB.FilterImageData(ctx, []string{digest}) + So(err, ShouldBeNil) - imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, "repo", "tag", digest, - repoMeta[0], indexData[digest.String()], manifestMetadata) + imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, convert.GetImageMeta("tag", repoMeta, + imageData[digest])) So(err, ShouldBeNil) So(*imageSummary.Description, ShouldResemble, "ConfigDescription") So(*imageSummary.Licenses, ShouldResemble, "ConfigLicenses") @@ -851,17 +761,28 @@ func TestIndexAnnotations(t *testing.T) { So(*imageSummary.Documentation, ShouldResemble, "IndexDocumentation") So(*imageSummary.Source, ShouldResemble, "IndexSource") + err = metaDB.ResetDB() + So(err, ShouldBeNil) //-------------------------------------------------------- indexWithNoAnnotations := CreateRandomMultiarch() - repoMeta, manifestMetadata, indexData = ociutils.GetMetadataForRepos(ociutils.Repo{ - Name: "repo", - MultiArchImages: []ociutils.RepoMultiArchImage{{MultiarchImage: indexWithNoAnnotations, Tag: "tag"}}, + ctx, err = ociutils.InitializeTestMetaDB(ctx, metaDB, ociutils.Repo{ + Name: "repo", + MultiArchImages: []ociutils.RepoMultiArchImage{ + {MultiarchImage: indexWithNoAnnotations, Reference: "tag"}, + }, }) - digest = indexWithNoAnnotations.Digest() + So(err, ShouldBeNil) - imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, "repo", "tag", digest, - repoMeta[0], indexData[digest.String()], manifestMetadata) + digest = indexWithNoAnnotations.DigestStr() + + repoMeta, err = metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldBeNil) + imageData, err = metaDB.FilterImageData(ctx, []string{digest}) + So(err, ShouldBeNil) + + imageSummary, _, err = convert.ImageIndex2ImageSummary(ctx, convert.GetImageMeta("tag", repoMeta, + imageData[digest])) So(err, ShouldBeNil) So(*imageSummary.Description, ShouldBeBlank) So(*imageSummary.Licenses, ShouldBeBlank) @@ -870,94 +791,8 @@ func TestIndexAnnotations(t *testing.T) { So(*imageSummary.Title, ShouldBeBlank) So(*imageSummary.Documentation, ShouldBeBlank) So(*imageSummary.Source, ShouldBeBlank) - }) -} - -func TestDownloadCount(t *testing.T) { - Convey("manifest", t, func() { - repoMeta, manifestMetaMap, indexDataMap := ociutils.GetMetadataForRepos( - ociutils.Repo{ - Name: "repo", - Images: []ociutils.RepoImage{ - { - Image: CreateRandomImage(), - Tag: "10-downloads", - Statistics: mTypes.DescriptorStatistics{ - DownloadCount: 10, - }, - }, - }, - }, - ) - - repoSummary := convert.RepoMeta2RepoSummary(context.Background(), repoMeta[0], manifestMetaMap, indexDataMap) - So(*repoSummary.DownloadCount, ShouldEqual, 10) - So(*repoSummary.NewestImage.DownloadCount, ShouldEqual, 10) - }) - - Convey("index", t, func() { - img1, img2, img3 := CreateRandomImage(), CreateRandomImage(), CreateRandomImage() - multiArch := CreateMultiarchWith().Images([]Image{img1, img2, img3}).Build() - - repoMeta, manifestMetaMap, indexDataMap := ociutils.GetMetadataForRepos( - ociutils.Repo{ - Name: "repo", - MultiArchImages: []ociutils.RepoMultiArchImage{ - { - MultiarchImage: multiArch, - Tag: "160-multiarch", - ImageStatistics: map[string]mTypes.DescriptorStatistics{ - img1.DigestStr(): {DownloadCount: 10}, - img2.DigestStr(): {DownloadCount: 20}, - img3.DigestStr(): {DownloadCount: 30}, - multiArch.DigestStr(): {DownloadCount: 100}, - }, - }, - }, - }, - ) - - repoSummary := convert.RepoMeta2RepoSummary(context.Background(), repoMeta[0], manifestMetaMap, indexDataMap) - So(*repoSummary.DownloadCount, ShouldEqual, 100) - So(*repoSummary.NewestImage.DownloadCount, ShouldEqual, 100) - }) - - Convey("index + manifest mixed", t, func() { - img1 := CreateRandomImage() - img2 := CreateRandomImage() - img3 := CreateImageWith().DefaultLayers().ImageConfig( - ispec.Image{Created: DateRef(2020, 1, 1, 1, 1, 1, 0, time.UTC)}, - ).Build() - multiArch := CreateMultiarchWith().Images([]Image{img1, img2, img3}).Build() - - repoMeta, manifestMetaMap, indexDataMap := ociutils.GetMetadataForRepos( - ociutils.Repo{ - Name: "repo", - Images: []ociutils.RepoImage{ - { - Image: CreateRandomImage(), - Tag: "5-downloads", - Statistics: mTypes.DescriptorStatistics{DownloadCount: 5}, - }, - }, - MultiArchImages: []ociutils.RepoMultiArchImage{ - { - MultiarchImage: multiArch, - Tag: "160-multiarch", - ImageStatistics: map[string]mTypes.DescriptorStatistics{ - img1.DigestStr(): {DownloadCount: 10}, - img2.DigestStr(): {DownloadCount: 20}, - img3.DigestStr(): {DownloadCount: 30}, - multiArch.DigestStr(): {DownloadCount: 100}, - }, - }, - }, - }, - ) - - repoSummary := convert.RepoMeta2RepoSummary(context.Background(), repoMeta[0], manifestMetaMap, indexDataMap) - So(*repoSummary.DownloadCount, ShouldEqual, 105) - So(*repoSummary.NewestImage.DownloadCount, ShouldEqual, 100) + err = metaDB.ResetDB() + So(err, ShouldBeNil) }) } diff --git a/pkg/extensions/search/convert/metadb.go b/pkg/extensions/search/convert/metadb.go index 64b7c63135..dea51c631c 100644 --- a/pkg/extensions/search/convert/metadb.go +++ b/pkg/extensions/search/convert/metadb.go @@ -2,15 +2,11 @@ package convert import ( "context" - "encoding/json" - "fmt" "sort" "strconv" - "strings" "time" "github.com/99designs/gqlgen/graphql" - godigest "github.com/opencontainers/go-digest" ispec "github.com/opencontainers/image-spec/specs-go/v1" "github.com/vektah/gqlparser/v2/gqlerror" @@ -20,8 +16,6 @@ import ( "zotregistry.io/zot/pkg/extensions/search/gql_generated" "zotregistry.io/zot/pkg/extensions/search/pagination" "zotregistry.io/zot/pkg/log" - mcommon "zotregistry.io/zot/pkg/meta/common" - "zotregistry.io/zot/pkg/meta/proto_go" mTypes "zotregistry.io/zot/pkg/meta/types" ) @@ -29,209 +23,6 @@ type SkipQGLField struct { Vulnerabilities bool } -func RepoMeta2RepoSummary(ctx context.Context, repoMeta mTypes.RepoMetadata, - manifestMetaMap map[string]mTypes.ManifestMetadata, indexDataMap map[string]mTypes.IndexData, -) *gql_generated.RepoSummary { - var ( - repoName = repoMeta.Name - repoLastUpdatedTimestamp = time.Time{} - repoPlatformsSet = map[string]*gql_generated.Platform{} - repoVendorsSet = map[string]bool{} - lastUpdatedImageSummary *gql_generated.ImageSummary - repoDownloadCount = 0 - repoStarCount = repoMeta.Stars // total number of stars - repoIsUserStarred = repoMeta.IsStarred // value specific to the current user - repoIsUserBookMarked = repoMeta.IsBookmarked // value specific to the current user - - // map used to keep track of all blobs of a repo without duplicates as - // some images may have the same layers - repoBlob2Size = make(map[string]int64, 10) - - // made up of all manifests, configs and image layers - size = int64(0) - ) - - for tag, descriptor := range repoMeta.Tags { - imageSummary, imageBlobsMap, err := Descriptor2ImageSummary(ctx, descriptor, repoMeta.Name, - tag, repoMeta, manifestMetaMap, indexDataMap, - ) - if err != nil { - continue - } - - for blobDigest, blobSize := range imageBlobsMap { - repoBlob2Size[blobDigest] = blobSize - } - - for _, manifestSummary := range imageSummary.Manifests { - if *manifestSummary.Platform.Os != "" || *manifestSummary.Platform.Arch != "" { - opSys, arch := *manifestSummary.Platform.Os, *manifestSummary.Platform.Arch - - platformString := strings.TrimSpace(fmt.Sprintf("%s %s", opSys, arch)) - repoPlatformsSet[platformString] = &gql_generated.Platform{Os: &opSys, Arch: &arch} - } - } - - repoDownloadCount += *imageSummary.DownloadCount - - if *imageSummary.Vendor != "" { - repoVendorsSet[*imageSummary.Vendor] = true - } - - lastUpdatedImageSummary = UpdateLastUpdatedTimestamp(&repoLastUpdatedTimestamp, lastUpdatedImageSummary, imageSummary) - } - - // calculate repo size = sum all manifest, config and layer blobs sizes - for _, blobSize := range repoBlob2Size { - size += blobSize - } - - repoSize := strconv.FormatInt(size, 10) - - repoPlatforms := make([]*gql_generated.Platform, 0, len(repoPlatformsSet)) - - for _, platform := range repoPlatformsSet { - repoPlatforms = append(repoPlatforms, platform) - } - - repoVendors := make([]*string, 0, len(repoVendorsSet)) - - for vendor := range repoVendorsSet { - vendor := vendor - repoVendors = append(repoVendors, &vendor) - } - - return &gql_generated.RepoSummary{ - Name: &repoName, - LastUpdated: &repoLastUpdatedTimestamp, - Size: &repoSize, - Platforms: repoPlatforms, - Vendors: repoVendors, - NewestImage: lastUpdatedImageSummary, - DownloadCount: &repoDownloadCount, - StarCount: &repoStarCount, - IsBookmarked: &repoIsUserBookMarked, - IsStarred: &repoIsUserStarred, - Rank: &repoMeta.Rank, - } -} - -func ProtoImageData2RepoSummary(ctx context.Context, repo string, imageDataList []*proto_go.ImageData, -) *gql_generated.RepoSummary { - var ( - repoName = repo - repoLastUpdatedTimestamp = time.Time{} - repoPlatformsSet = map[string]*gql_generated.Platform{} - repoVendorsSet = map[string]bool{} - lastUpdatedImageSummary *gql_generated.ImageSummary - repoDownloadCount = 0 - // repoStarCount = repoMeta.Stars // total number of stars - // repoIsUserStarred = repoMeta.IsStarred // value specific to the current user - // repoIsUserBookMarked = repoMeta.IsBookmarked // value specific to the current user - - // map used to keep track of all blobs of a repo without duplicates as - // some images may have the same layers - repoBlob2Size = make(map[string]int64, 10) - - // made up of all manifests, configs and image layers - size = int64(0) - ) - - for _, imageData := range imageDataList { - imageSummary, imageBlobsMap, err := ProtoImageData2ImageSummary(ctx, imageData) - if err != nil { - continue - } - - for blobDigest, blobSize := range imageBlobsMap { - repoBlob2Size[blobDigest] = blobSize - } - - for _, manifestSummary := range imageSummary.Manifests { - if *manifestSummary.Platform.Os != "" || *manifestSummary.Platform.Arch != "" { - opSys, arch := *manifestSummary.Platform.Os, *manifestSummary.Platform.Arch - - platformString := strings.TrimSpace(fmt.Sprintf("%s %s", opSys, arch)) - repoPlatformsSet[platformString] = &gql_generated.Platform{Os: &opSys, Arch: &arch} - } - } - - repoDownloadCount += *imageSummary.DownloadCount - - if *imageSummary.Vendor != "" { - repoVendorsSet[*imageSummary.Vendor] = true - } - - lastUpdatedImageSummary = UpdateLastUpdatedTimestamp(&repoLastUpdatedTimestamp, lastUpdatedImageSummary, imageSummary) - } - - // calculate repo size = sum all manifest, config and layer blobs sizes - for _, blobSize := range repoBlob2Size { - size += blobSize - } - - repoSize := strconv.FormatInt(size, 10) - - repoPlatforms := make([]*gql_generated.Platform, 0, len(repoPlatformsSet)) - - for _, platform := range repoPlatformsSet { - repoPlatforms = append(repoPlatforms, platform) - } - - repoVendors := make([]*string, 0, len(repoVendorsSet)) - - for vendor := range repoVendorsSet { - vendor := vendor - repoVendors = append(repoVendors, &vendor) - } - - return &gql_generated.RepoSummary{ - Name: &repoName, - LastUpdated: &repoLastUpdatedTimestamp, - Size: &repoSize, - Platforms: repoPlatforms, - Vendors: repoVendors, - NewestImage: lastUpdatedImageSummary, - DownloadCount: &repoDownloadCount, - // StarCount: &repoStarCount, - // IsBookmarked: &repoIsUserBookMarked, - // IsStarred: &repoIsUserStarred, - // Rank: &repoMeta.Rank, - } -} - -func ProtoImageData2ImageSummary(ctx context.Context, imageData *proto_go.ImageData, -) (*gql_generated.ImageSummary, map[string]int64, error) { - panic("unimplemented") -} - -func PaginatedRepoMeta2RepoSummaries(ctx context.Context, reposMeta []mTypes.RepoMetadata, - manifestMetaMap map[string]mTypes.ManifestMetadata, indexDataMap map[string]mTypes.IndexData, - skip SkipQGLField, cveInfo cveinfo.CveInfo, filter mTypes.Filter, pageInput pagination.PageInput, -) ([]*gql_generated.RepoSummary, zcommon.PageInfo, error) { - reposPageFinder, err := pagination.NewRepoSumPageFinder(pageInput.Limit, pageInput.Offset, pageInput.SortBy) - if err != nil { - return []*gql_generated.RepoSummary{}, zcommon.PageInfo{}, err - } - - for _, repoMeta := range reposMeta { - repoSummary := RepoMeta2RepoSummary(ctx, repoMeta, manifestMetaMap, indexDataMap) - - if RepoSumAcceptedByFilter(repoSummary, filter) { - reposPageFinder.Add(repoSummary) - } - } - - page, pageInfo := reposPageFinder.Page() - - // CVE scanning is expensive, only scan for the current page - for _, repoSummary := range page { - updateRepoSummaryVulnerabilities(ctx, repoSummary, skip, cveInfo) - } - - return page, pageInfo, nil -} - func UpdateLastUpdatedTimestamp(repoLastUpdatedTimestamp *time.Time, lastUpdatedImageSummary *gql_generated.ImageSummary, imageSummary *gql_generated.ImageSummary, ) *gql_generated.ImageSummary { @@ -249,223 +40,6 @@ func UpdateLastUpdatedTimestamp(repoLastUpdatedTimestamp *time.Time, return newLastUpdatedImageSummary } -func Descriptor2ImageSummary(ctx context.Context, descriptor mTypes.Descriptor, repo, tag string, - repoMeta mTypes.RepoMetadata, manifestMetaMap map[string]mTypes.ManifestMetadata, - indexDataMap map[string]mTypes.IndexData, -) (*gql_generated.ImageSummary, map[string]int64, error) { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - return ImageManifest2ImageSummary(ctx, repo, tag, godigest.Digest(descriptor.Digest), - repoMeta, manifestMetaMap[descriptor.Digest]) - case ispec.MediaTypeImageIndex: - return ImageIndex2ImageSummary(ctx, repo, tag, godigest.Digest(descriptor.Digest), - repoMeta, indexDataMap[descriptor.Digest], manifestMetaMap) - default: - return &gql_generated.ImageSummary{}, map[string]int64{}, zerr.ErrMediaTypeNotSupported - } -} - -func ImageIndex2ImageSummary(ctx context.Context, repo, tag string, indexDigest godigest.Digest, - repoMeta mTypes.RepoMetadata, indexData mTypes.IndexData, manifestMetaMap map[string]mTypes.ManifestMetadata, -) (*gql_generated.ImageSummary, map[string]int64, error) { - var indexContent ispec.Index - - err := json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return &gql_generated.ImageSummary{}, map[string]int64{}, err - } - - var ( - indexLastUpdated time.Time - isSigned bool - totalIndexSize int64 - indexSize string - totalDownloadCount int - manifestAnnotations *ImageAnnotations - manifestSummaries = make([]*gql_generated.ManifestSummary, 0, len(indexContent.Manifests)) - indexBlobs = make(map[string]int64, 0) - - indexDigestStr = indexDigest.String() - indexMediaType = ispec.MediaTypeImageIndex - ) - - for _, descriptor := range indexContent.Manifests { - manifestSummary, manifestBlobs, annotations, err := ImageManifest2ManifestSummary(ctx, repo, tag, descriptor, - repoMeta, manifestMetaMap[descriptor.Digest.String()], repoMeta.Referrers[descriptor.Digest.String()]) - if err != nil { - return &gql_generated.ImageSummary{}, map[string]int64{}, err - } - - manifestSize := int64(0) - - for digest, size := range manifestBlobs { - indexBlobs[digest] = size - manifestSize += size - } - - if indexLastUpdated.Before(*manifestSummary.LastUpdated) { - indexLastUpdated = *manifestSummary.LastUpdated - } - - if manifestAnnotations == nil { - manifestAnnotations = annotations - } - - totalIndexSize += manifestSize - - manifestSummaries = append(manifestSummaries, manifestSummary) - } - - totalDownloadCount += repoMeta.Statistics[indexDigestStr].DownloadCount - - for _, signatures := range repoMeta.Signatures[indexDigest.String()] { - if len(signatures) > 0 { - isSigned = true - } - } - - indexSize = strconv.FormatInt(totalIndexSize, 10) - - signaturesInfo := GetSignaturesInfo(isSigned, repoMeta, indexDigest) - - if manifestAnnotations == nil { - // The index doesn't have manifests - manifestAnnotations = &ImageAnnotations{} - } - - annotations := GetIndexAnnotations(indexContent.Annotations, manifestAnnotations) - - indexSummary := gql_generated.ImageSummary{ - RepoName: &repo, - Tag: &tag, - Digest: &indexDigestStr, - MediaType: &indexMediaType, - Manifests: manifestSummaries, - LastUpdated: &indexLastUpdated, - IsSigned: &isSigned, - SignatureInfo: signaturesInfo, - Size: &indexSize, - DownloadCount: &totalDownloadCount, - Description: &annotations.Description, - Title: &annotations.Title, - Documentation: &annotations.Documentation, - Licenses: &annotations.Licenses, - Labels: &annotations.Labels, - Source: &annotations.Source, - Vendor: &annotations.Vendor, - Authors: &annotations.Authors, - Referrers: getReferrers(repoMeta.Referrers[indexDigest.String()]), - } - - return &indexSummary, indexBlobs, nil -} - -func ImageManifest2ImageSummary(ctx context.Context, repo, tag string, digest godigest.Digest, - repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata, -) (*gql_generated.ImageSummary, map[string]int64, error) { - var ( - manifestContent ispec.Manifest - manifestDigest = digest.String() - mediaType = ispec.MediaTypeImageManifest - ) - - err := json.Unmarshal(manifestMeta.ManifestBlob, &manifestContent) - if err != nil { - graphql.AddError(ctx, gqlerror.Errorf("can't unmarshal manifest blob for image: %s:%s, manifest digest: %s, "+ - "error: %s", repo, tag, manifestDigest, err.Error())) - - return &gql_generated.ImageSummary{}, map[string]int64{}, err - } - - configContent := mcommon.InitializeImageConfig(manifestMeta.ConfigBlob) - - var ( - repoName = repo - configDigest = manifestContent.Config.Digest.String() - configSize = manifestContent.Config.Size - artifactType = zcommon.GetManifestArtifactType(manifestContent) - imageLastUpdated = zcommon.GetImageLastUpdated(configContent) - downloadCount = repoMeta.Statistics[digest.String()].DownloadCount - isSigned = false - ) - - opSys := configContent.OS - arch := configContent.Architecture - variant := configContent.Variant - - if variant != "" { - arch = arch + "/" + variant - } - - platform := gql_generated.Platform{Os: &opSys, Arch: &arch} - - for _, signatures := range repoMeta.Signatures[digest.String()] { - if len(signatures) > 0 { - isSigned = true - } - } - - size, imageBlobsMap := getImageBlobsInfo( - manifestDigest, int64(len(manifestMeta.ManifestBlob)), - configDigest, configSize, - manifestContent.Layers) - imageSize := strconv.FormatInt(size, 10) - - annotations := GetAnnotations(manifestContent.Annotations, configContent.Config.Labels) - - authors := annotations.Authors - if authors == "" { - authors = configContent.Author - } - - historyEntries, err := getAllHistory(manifestContent, configContent) - if err != nil { - graphql.AddError(ctx, gqlerror.Errorf("error generating history on tag %s in repo %s: "+ - "manifest digest: %s, error: %s", tag, repo, manifestDigest, err.Error())) - } - - signaturesInfo := GetSignaturesInfo(isSigned, repoMeta, digest) - - manifestSummary := gql_generated.ManifestSummary{ - Digest: &manifestDigest, - ConfigDigest: &configDigest, - LastUpdated: &imageLastUpdated, - Size: &imageSize, - IsSigned: &isSigned, - SignatureInfo: signaturesInfo, - Platform: &platform, - DownloadCount: &downloadCount, - Layers: getLayersSummaries(manifestContent), - History: historyEntries, - Referrers: getReferrers(repoMeta.Referrers[manifestDigest]), - ArtifactType: &artifactType, - } - - imageSummary := gql_generated.ImageSummary{ - RepoName: &repoName, - Tag: &tag, - Digest: &manifestDigest, - MediaType: &mediaType, - Manifests: []*gql_generated.ManifestSummary{&manifestSummary}, - LastUpdated: &imageLastUpdated, - IsSigned: &isSigned, - SignatureInfo: signaturesInfo, - Size: &imageSize, - DownloadCount: &downloadCount, - Description: &annotations.Description, - Title: &annotations.Title, - Documentation: &annotations.Documentation, - Licenses: &annotations.Licenses, - Labels: &annotations.Labels, - Source: &annotations.Source, - Vendor: &annotations.Vendor, - Authors: &authors, - Referrers: getReferrers(repoMeta.Referrers[manifestDigest]), - } - - return &imageSummary, imageBlobsMap, nil -} - func getReferrers(referrersInfo []mTypes.ReferrerInfo) []*gql_generated.Referrer { referrers := make([]*gql_generated.Referrer, 0, len(referrersInfo)) @@ -487,95 +61,17 @@ func getReferrers(referrersInfo []mTypes.ReferrerInfo) []*gql_generated.Referrer func getAnnotationsFromMap(annotationsMap map[string]string) []*gql_generated.Annotation { annotations := make([]*gql_generated.Annotation, 0, len(annotationsMap)) - for key, value := range annotationsMap { - key := key - value := value - - annotations = append(annotations, &gql_generated.Annotation{ - Key: &key, - Value: &value, - }) - } - - return annotations -} - -func ImageManifest2ManifestSummary(ctx context.Context, repo, tag string, descriptor ispec.Descriptor, - repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata, - referrersInfo []mTypes.ReferrerInfo, -) (*gql_generated.ManifestSummary, map[string]int64, *ImageAnnotations, error) { - var ( - manifestContent ispec.Manifest - digest = descriptor.Digest - ) - - err := json.Unmarshal(manifestMeta.ManifestBlob, &manifestContent) - if err != nil { - graphql.AddError(ctx, gqlerror.Errorf("can't unmarshal manifest blob for image: %s:%s, manifest digest: %s, "+ - "error: %s", repo, tag, digest, err.Error())) - - return &gql_generated.ManifestSummary{}, map[string]int64{}, &ImageAnnotations{}, err - } - - configContent := mcommon.InitializeImageConfig(manifestMeta.ConfigBlob) - annotations := GetAnnotations(manifestContent.Annotations, configContent.Config.Labels) - - var ( - manifestDigestStr = digest.String() - configDigest = manifestContent.Config.Digest.String() - configSize = manifestContent.Config.Size - artifactType = zcommon.GetManifestArtifactType(manifestContent) - imageLastUpdated = zcommon.GetImageLastUpdated(configContent) - downloadCount = repoMeta.Statistics[digest.String()].DownloadCount - isSigned = false - ) - - opSys := configContent.OS - arch := configContent.Architecture - variant := configContent.Variant - - if variant != "" { - arch = arch + "/" + variant - } - - platform := gql_generated.Platform{Os: &opSys, Arch: &arch} - - size, imageBlobsMap := getImageBlobsInfo( - manifestDigestStr, int64(len(manifestMeta.ManifestBlob)), - configDigest, configSize, - manifestContent.Layers) - imageSize := strconv.FormatInt(size, 10) - - historyEntries, err := getAllHistory(manifestContent, configContent) - if err != nil { - graphql.AddError(ctx, gqlerror.Errorf("error generating history on tag %s in repo %s: "+ - "manifest digest: %s, error: %s", tag, repo, manifestDigestStr, err.Error())) - } - - for _, signatures := range repoMeta.Signatures[manifestDigestStr] { - if len(signatures) > 0 { - isSigned = true - } - } - - signaturesInfo := GetSignaturesInfo(isSigned, repoMeta, digest) - - manifestSummary := gql_generated.ManifestSummary{ - Digest: &manifestDigestStr, - ConfigDigest: &configDigest, - LastUpdated: &imageLastUpdated, - Size: &imageSize, - Platform: &platform, - DownloadCount: &downloadCount, - Layers: getLayersSummaries(manifestContent), - History: historyEntries, - IsSigned: &isSigned, - SignatureInfo: signaturesInfo, - Referrers: getReferrers(referrersInfo), - ArtifactType: &artifactType, + for key, value := range annotationsMap { + key := key + value := value + + annotations = append(annotations, &gql_generated.Annotation{ + Key: &key, + Value: &value, + }) } - return &manifestSummary, imageBlobsMap, &annotations, nil + return annotations } func getImageBlobsInfo(manifestDigest string, manifestSize int64, configDigest string, configSize int64, @@ -601,43 +97,8 @@ func getImageBlobsInfo(manifestDigest string, manifestSize int64, configDigest s return imageSize, imageBlobsMap } -func RepoMeta2ImageSummaries(ctx context.Context, repoMeta mTypes.RepoMetadata, - manifestMetaMap map[string]mTypes.ManifestMetadata, indexDataMap map[string]mTypes.IndexData, - skip SkipQGLField, cveInfo cveinfo.CveInfo, -) []*gql_generated.ImageSummary { - imageSummaries := make([]*gql_generated.ImageSummary, 0, len(repoMeta.Tags)) - - // Make sure the tags are sorted - // We need to implement a proper fix for this taking into account - // the sorting criteria used in the requested page - tags := make([]string, 0, len(repoMeta.Tags)) - for tag := range repoMeta.Tags { - tags = append(tags, tag) - } - - // Sorting ascending by tag name should do for now - sort.Strings(tags) - - for _, tag := range tags { - descriptor := repoMeta.Tags[tag] - - imageSummary, _, err := Descriptor2ImageSummary(ctx, descriptor, repoMeta.Name, tag, - repoMeta, manifestMetaMap, indexDataMap) - if err != nil { - continue - } - - // CVE scanning is expensive, only scan for final slice of results - updateImageSummaryVulnerabilities(ctx, imageSummary, skip, cveInfo) - - imageSummaries = append(imageSummaries, imageSummary) - } - - return imageSummaries -} - -func ProtoRepoMeta2ImageSummaries(ctx context.Context, repoMeta mTypes.RepoMetadata2, - imageData map[string]mTypes.ImageData2, skip SkipQGLField, cveInfo cveinfo.CveInfo, +func RepoMeta2ImageSummaries(ctx context.Context, repoMeta mTypes.RepoMeta, + imageData map[string]mTypes.ImageData, skip SkipQGLField, cveInfo cveinfo.CveInfo, ) []*gql_generated.ImageSummary { imageSummaries := make([]*gql_generated.ImageSummary, 0, len(repoMeta.Tags)) @@ -655,8 +116,7 @@ func ProtoRepoMeta2ImageSummaries(ctx context.Context, repoMeta mTypes.RepoMetad for _, tag := range tags { descriptor := repoMeta.Tags[tag] - imageSummary, _, err := ProtoDescriptor2ImageSummary(ctx, descriptor, repoMeta.Name, tag, - repoMeta, imageData) + imageSummary, _, err := ImageMeta2ImageSummary(ctx, GetImageMeta(tag, repoMeta, imageData[descriptor.Digest])) if err != nil { continue } @@ -670,69 +130,16 @@ func ProtoRepoMeta2ImageSummaries(ctx context.Context, repoMeta mTypes.RepoMetad return imageSummaries } -func PaginatedRepoMeta2ImageSummaries(ctx context.Context, reposMeta []mTypes.RepoMetadata, - manifestMetaMap map[string]mTypes.ManifestMetadata, indexDataMap map[string]mTypes.IndexData, - skip SkipQGLField, cveInfo cveinfo.CveInfo, filter mTypes.Filter, pageInput pagination.PageInput, -) ([]*gql_generated.ImageSummary, zcommon.PageInfo, error) { - imagePageFinder, err := pagination.NewImgSumPageFinder(pageInput.Limit, pageInput.Offset, pageInput.SortBy) - if err != nil { - return []*gql_generated.ImageSummary{}, zcommon.PageInfo{}, err - } - - for _, repoMeta := range reposMeta { - for tag := range repoMeta.Tags { - descriptor := repoMeta.Tags[tag] - - imageSummary, _, err := Descriptor2ImageSummary(ctx, descriptor, repoMeta.Name, tag, - repoMeta, manifestMetaMap, indexDataMap) - if err != nil { - continue - } - - if ImgSumAcceptedByFilter(imageSummary, filter) { - imagePageFinder.Add(imageSummary) - } - } - } - - page, pageInfo := imagePageFinder.Page() - - for _, imageSummary := range page { - // CVE scanning is expensive, only scan for this page - updateImageSummaryVulnerabilities(ctx, imageSummary, skip, cveInfo) - } - - return page, pageInfo, nil -} - -func RepoMeta2ExpandedRepoInfo(ctx context.Context, repoMeta mTypes.RepoMetadata, - manifestMetaMap map[string]mTypes.ManifestMetadata, indexDataMap map[string]mTypes.IndexData, - skip SkipQGLField, cveInfo cveinfo.CveInfo, log log.Logger, +func RepoMeta2ExpandedRepoInfo(ctx context.Context, repoMeta mTypes.RepoMeta, + imageDataMap map[string]mTypes.ImageData, skip SkipQGLField, cveInfo cveinfo.CveInfo, log log.Logger, ) (*gql_generated.RepoSummary, []*gql_generated.ImageSummary) { - var ( - repoName = repoMeta.Name - repoLastUpdatedTimestamp = time.Time{} - repoPlatformsSet = map[string]*gql_generated.Platform{} - repoVendorsSet = map[string]bool{} - lastUpdatedImageSummary *gql_generated.ImageSummary - repoDownloadCount = 0 - repoStarCount = repoMeta.Stars // total number of stars - isStarred = repoMeta.IsStarred // value specific to the current user - isBookmarked = repoMeta.IsBookmarked // value specific to the current user - - // map used to keep track of all blobs of a repo without duplicates as - // some images may have the same layers - repoBlob2Size = make(map[string]int64, 10) - - // made up of all manifests, configs and image layers - size = int64(0) - - imageSummaries = make([]*gql_generated.ImageSummary, 0, len(repoMeta.Tags)) - ) + repoName := repoMeta.Name + imageSummaries := make([]*gql_generated.ImageSummary, 0, len(repoMeta.Tags)) for tag, descriptor := range repoMeta.Tags { - imageSummary, imageBlobs, err := Descriptor2ImageSummary(ctx, descriptor, repoName, tag, - repoMeta, manifestMetaMap, indexDataMap) + imageData := imageDataMap[descriptor.Digest] + + imageSummary, _, err := ImageMeta2ImageSummary(ctx, GetImageMeta(tag, repoMeta, imageData)) if err != nil { log.Error().Str("repository", repoName).Str("reference", tag). Msg("metadb: error while converting descriptor for image") @@ -740,105 +147,47 @@ func RepoMeta2ExpandedRepoInfo(ctx context.Context, repoMeta mTypes.RepoMetadata continue } - for _, manifestSummary := range imageSummary.Manifests { - opSys, arch := *manifestSummary.Platform.Os, *manifestSummary.Platform.Arch - if opSys != "" || arch != "" { - platformString := strings.TrimSpace(fmt.Sprintf("%s %s", opSys, arch)) - repoPlatformsSet[platformString] = &gql_generated.Platform{Os: &opSys, Arch: &arch} - } - - updateRepoBlobsMap(imageBlobs, repoBlob2Size) - } - - if *imageSummary.Vendor != "" { - repoVendorsSet[*imageSummary.Vendor] = true - } - updateImageSummaryVulnerabilities(ctx, imageSummary, skip, cveInfo) - lastUpdatedImageSummary = UpdateLastUpdatedTimestamp(&repoLastUpdatedTimestamp, lastUpdatedImageSummary, imageSummary) - - repoDownloadCount += *imageSummary.DownloadCount - imageSummaries = append(imageSummaries, imageSummary) } - // calculate repo size = sum all manifest, config and layer blobs sizes - for _, blobSize := range repoBlob2Size { - size += blobSize - } - - repoSize := strconv.FormatInt(size, 10) - - repoPlatforms := make([]*gql_generated.Platform, 0, len(repoPlatformsSet)) + repoSummary := RepoMeta2RepoSummary(ctx, repoMeta, imageDataMap) - for _, platform := range repoPlatformsSet { - repoPlatforms = append(repoPlatforms, platform) - } - - repoVendors := make([]*string, 0, len(repoVendorsSet)) + updateRepoSummaryVulnerabilities(ctx, repoSummary, skip, cveInfo) - for vendor := range repoVendorsSet { - vendor := vendor - repoVendors = append(repoVendors, &vendor) - } + return repoSummary, imageSummaries +} - summary := &gql_generated.RepoSummary{ - Name: &repoName, - LastUpdated: &repoLastUpdatedTimestamp, - Size: &repoSize, - Platforms: repoPlatforms, - Vendors: repoVendors, - NewestImage: lastUpdatedImageSummary, - DownloadCount: &repoDownloadCount, - StarCount: &repoStarCount, - IsBookmarked: &isBookmarked, - IsStarred: &isStarred, +func GetImageMeta(tag string, repoMeta mTypes.RepoMeta, imageData mTypes.ImageData, +) mTypes.ImageMeta { + return mTypes.ImageMeta{ + Repo: repoMeta.Name, + Tag: tag, + MediaType: imageData.MediaType, + Digest: imageData.Digest, + Size: imageData.Size, + Index: imageData.Index, + Manifests: GetManifestMeta(repoMeta, imageData.Manifests), + Referrers: repoMeta.Referrers[imageData.Digest.String()], + Statistics: repoMeta.Statistics[imageData.Digest.String()], + Signatures: repoMeta.Signatures[imageData.Digest.String()], } - - updateRepoSummaryVulnerabilities(ctx, summary, skip, cveInfo) - - return summary, imageSummaries } -func FullRepoMeta2ExpandedRepoInfo(ctx context.Context, repoMeta mTypes.FullRepoMetadata, - imageDataMap map[string]mTypes.ImageData2, skip SkipQGLField, cveInfo cveinfo.CveInfo, log log.Logger, -) (*gql_generated.RepoSummary, []*gql_generated.ImageSummary) { - repoName := repoMeta.Name - imageSummaries := make([]*gql_generated.ImageSummary, 0, len(repoMeta.Tags)) - - for tag, descriptor := range repoMeta.Tags { - imageData := imageDataMap[descriptor.Digest] +func GetManifestMeta(repoMeta mTypes.RepoMeta, manifests []mTypes.ManifestData) []mTypes.ManifestMeta { + results := make([]mTypes.ManifestMeta, 0, len(manifests)) - imageSummary, _, err := FullImageData2ImageSummary(ctx, mTypes.FullImageData{ - Repo: repoName, - Tag: tag, - MediaType: descriptor.MediaType, - Digest: godigest.Digest(descriptor.Digest), - Size: imageData.Size, - Index: imageData.Index, - Manifests: getFullManifestData(repoMeta, imageData.Manifests), - Referrers: repoMeta.Referrers[descriptor.Digest], - Statistics: repoMeta.Statistics[descriptor.Digest], - Signatures: repoMeta.Signatures[descriptor.Digest], + for i := range manifests { + results = append(results, mTypes.ManifestMeta{ + ManifestData: manifests[i], + Referrers: repoMeta.Referrers[manifests[i].Digest.String()], + Statistics: repoMeta.Statistics[manifests[i].Digest.String()], + Signatures: repoMeta.Signatures[manifests[i].Digest.String()], }) - if err != nil { - log.Error().Str("repository", repoName).Str("reference", tag). - Msg("metadb: error while converting descriptor for image") - - continue - } - - updateImageSummaryVulnerabilities(ctx, imageSummary, skip, cveInfo) - - imageSummaries = append(imageSummaries, imageSummary) } - repoSummary := FullRepoMeta2RepoSummary(ctx, repoMeta, imageDataMap) - - updateRepoSummaryVulnerabilities(ctx, repoSummary, skip, cveInfo) - - return repoSummary, imageSummaries + return results } func StringMap2Annotations(strMap map[string]string) []*gql_generated.Annotation { @@ -900,90 +249,7 @@ func GetPreloadString(prefix, name string) string { return name } -func GetSignaturesInfo(isSigned bool, repoMeta mTypes.RepoMetadata, indexDigest godigest.Digest, -) []*gql_generated.SignatureSummary { - signaturesInfo := []*gql_generated.SignatureSummary{} - - if !isSigned { - return signaturesInfo - } - - for sigType, signatures := range repoMeta.Signatures[indexDigest.String()] { - for _, sig := range signatures { - for _, layer := range sig.LayersInfo { - var ( - isTrusted bool - author string - tool string - ) - - if layer.Signer != "" { - author = layer.Signer - - if !layer.Date.IsZero() && time.Now().After(layer.Date) { - isTrusted = false - } else { - isTrusted = true - } - } else { - isTrusted = false - author = "" - } - - tool = sigType - - signaturesInfo = append(signaturesInfo, - &gql_generated.SignatureSummary{Tool: &tool, IsTrusted: &isTrusted, Author: &author}) - } - } - } - - return signaturesInfo -} - -func ProtoGetSignaturesInfo(isSigned bool, repoMeta mTypes.RepoMetadata2, indexDigest godigest.Digest, -) []*gql_generated.SignatureSummary { - signaturesInfo := []*gql_generated.SignatureSummary{} - - if !isSigned { - return signaturesInfo - } - - for sigType, signatures := range repoMeta.Signatures[indexDigest.String()] { - for _, sig := range signatures { - for _, layer := range sig.LayersInfo { - var ( - isTrusted bool - author string - tool string - ) - - if layer.Signer != "" { - author = layer.Signer - - if !layer.Date.IsZero() && time.Now().After(layer.Date) { - isTrusted = false - } else { - isTrusted = true - } - } else { - isTrusted = false - author = "" - } - - tool = sigType - - signaturesInfo = append(signaturesInfo, - &gql_generated.SignatureSummary{Tool: &tool, IsTrusted: &isTrusted, Author: &author}) - } - } - } - - return signaturesInfo -} - -func FullGetSignaturesInfo(isSigned bool, signatures mTypes.ManifestSignatures, indexDigest godigest.Digest, -) []*gql_generated.SignatureSummary { +func GetSignaturesInfo(isSigned bool, signatures mTypes.ManifestSignatures) []*gql_generated.SignatureSummary { signaturesInfo := []*gql_generated.SignatureSummary{} if !isSigned { @@ -1023,36 +289,9 @@ func FullGetSignaturesInfo(isSigned bool, signatures mTypes.ManifestSignatures, return signaturesInfo } -func PaginatedProtoRepoMeta2RepoSummaries(ctx context.Context, repoMetaList []mTypes.RepoMetadata2, - imageDataMap map[string]mTypes.ImageData2, filter mTypes.Filter, pageInput pagination.PageInput, cveInfo cveinfo.CveInfo, - skip SkipQGLField, -) ([]*gql_generated.RepoSummary, zcommon.PageInfo, error) { - reposPageFinder, err := pagination.NewRepoSumPageFinder(pageInput.Limit, pageInput.Offset, pageInput.SortBy) - if err != nil { - return []*gql_generated.RepoSummary{}, zcommon.PageInfo{}, err - } - - for _, repoMeta := range repoMetaList { - repoSummary := ProtoRepoMeta2RepoSummary(ctx, repoMeta, imageDataMap) - - if RepoSumAcceptedByFilter(repoSummary, filter) { - reposPageFinder.Add(repoSummary) - } - } - - page, pageInfo := reposPageFinder.Page() - - // CVE scanning is expensive, only scan for the current page - for _, repoSummary := range page { - updateRepoSummaryVulnerabilities(ctx, repoSummary, skip, cveInfo) - } - - return page, pageInfo, nil -} - -func PaginatedFullRepoMeta2RepoSummaries(ctx context.Context, repoMetaList []mTypes.FullRepoMetadata, - imageDataMap map[string]mTypes.ImageData2, filter mTypes.Filter, pageInput pagination.PageInput, cveInfo cveinfo.CveInfo, - skip SkipQGLField, +func PaginatedRepoMeta2RepoSummaries(ctx context.Context, repoMetaList []mTypes.RepoMeta, + imageDataMap map[string]mTypes.ImageData, filter mTypes.Filter, pageInput pagination.PageInput, + cveInfo cveinfo.CveInfo, skip SkipQGLField, ) ([]*gql_generated.RepoSummary, zcommon.PageInfo, error) { reposPageFinder, err := pagination.NewRepoSumPageFinder(pageInput.Limit, pageInput.Offset, pageInput.SortBy) if err != nil { @@ -1060,114 +299,29 @@ func PaginatedFullRepoMeta2RepoSummaries(ctx context.Context, repoMetaList []mTy } for _, repoMeta := range repoMetaList { - repoSummary := FullRepoMeta2RepoSummary(ctx, repoMeta, imageDataMap) + repoSummary := RepoMeta2RepoSummary(ctx, repoMeta, imageDataMap) if RepoSumAcceptedByFilter(repoSummary, filter) { reposPageFinder.Add(repoSummary) - } - } - - page, pageInfo := reposPageFinder.Page() - - // CVE scanning is expensive, only scan for the current page - for _, repoSummary := range page { - updateRepoSummaryVulnerabilities(ctx, repoSummary, skip, cveInfo) - } - - return page, pageInfo, nil -} - -func ProtoRepoMeta2RepoSummary(ctx context.Context, repoMeta mTypes.RepoMetadata2, imageDataMap map[string]mTypes.ImageData2, -) *gql_generated.RepoSummary { - var ( - repoName = repoMeta.Name - repoLastUpdatedTimestamp = time.Time{} - repoPlatformsSet = map[string]*gql_generated.Platform{} - repoVendorsSet = map[string]bool{} - lastUpdatedImageSummary *gql_generated.ImageSummary - repoDownloadCount = 0 - repoStarCount = int(repoMeta.Stars) // total number of stars - repoIsUserStarred = repoMeta.IsStarred // value specific to the current user - repoIsUserBookMarked = repoMeta.IsBookmarked // value specific to the current user - - // map used to keep track of all blobs of a repo without duplicates as - // some images may have the same layers - repoBlob2Size = make(map[string]int64, 10) - - // made up of all manifests, configs and image layers - size = int64(0) - ) - - for tag, descriptor := range repoMeta.Tags { - imageSummary, imageBlobsMap, err := ProtoDescriptor2ImageSummary(ctx, descriptor, repoMeta.Name, - tag, repoMeta, imageDataMap, - ) - if err != nil { - continue - } - - for blobDigest, blobSize := range imageBlobsMap { - repoBlob2Size[blobDigest] = blobSize - } - - for _, manifestSummary := range imageSummary.Manifests { - if *manifestSummary.Platform.Os != "" || *manifestSummary.Platform.Arch != "" { - opSys, arch := *manifestSummary.Platform.Os, *manifestSummary.Platform.Arch - - platformString := strings.TrimSpace(fmt.Sprintf("%s %s", opSys, arch)) - repoPlatformsSet[platformString] = &gql_generated.Platform{Os: &opSys, Arch: &arch} - } - } - - repoDownloadCount += *imageSummary.DownloadCount - - if *imageSummary.Vendor != "" { - repoVendorsSet[*imageSummary.Vendor] = true - } - - lastUpdatedImageSummary = UpdateLastUpdatedTimestamp(&repoLastUpdatedTimestamp, lastUpdatedImageSummary, imageSummary) - } - - // calculate repo size = sum all manifest, config and layer blobs sizes - for _, blobSize := range repoBlob2Size { - size += blobSize - } - - repoSize := strconv.FormatInt(size, 10) - - repoPlatforms := make([]*gql_generated.Platform, 0, len(repoPlatformsSet)) - - for _, platform := range repoPlatformsSet { - repoPlatforms = append(repoPlatforms, platform) + } } - repoVendors := make([]*string, 0, len(repoVendorsSet)) + page, pageInfo := reposPageFinder.Page() - for vendor := range repoVendorsSet { - vendor := vendor - repoVendors = append(repoVendors, &vendor) + // CVE scanning is expensive, only scan for the current page + for _, repoSummary := range page { + updateRepoSummaryVulnerabilities(ctx, repoSummary, skip, cveInfo) } - return &gql_generated.RepoSummary{ - Name: &repoName, - LastUpdated: &repoLastUpdatedTimestamp, - Size: &repoSize, - Platforms: repoPlatforms, - Vendors: repoVendors, - NewestImage: lastUpdatedImageSummary, - DownloadCount: &repoDownloadCount, - StarCount: &repoStarCount, - IsBookmarked: &repoIsUserBookMarked, - IsStarred: &repoIsUserStarred, - Rank: ref(int(repoMeta.Rank)), - } + return page, pageInfo, nil } -func FullRepoMeta2RepoSummary(ctx context.Context, repoMeta mTypes.FullRepoMetadata, imageDataMap map[string]mTypes.ImageData2, +func RepoMeta2RepoSummary(ctx context.Context, repoMeta mTypes.RepoMeta, + imageDataMap map[string]mTypes.ImageData, ) *gql_generated.RepoSummary { var ( repoName = repoMeta.Name - repoLastUpdatedTimestamp = repoMeta.LastUpdatedImage.LastUpdated + repoLastUpdatedTimestamp = deref(repoMeta.LastUpdatedImage, mTypes.LastUpdatedImage{}).LastUpdated repoPlatforms = repoMeta.Platforms repoVendors = repoMeta.Vendors repoDownloadCount = repoMeta.DownloadCount @@ -1175,26 +329,15 @@ func FullRepoMeta2RepoSummary(ctx context.Context, repoMeta mTypes.FullRepoMetad repoIsUserStarred = repoMeta.IsStarred // value specific to the current user repoIsUserBookMarked = repoMeta.IsBookmarked // value specific to the current user repoSize = repoMeta.Size + lastUpdatedImageData = imageDataMap[repoMeta.LastUpdatedImage.Digest] + lastUpdatedTag = repoMeta.LastUpdatedImage.Tag ) - lastUpdatedImageData := imageDataMap[repoMeta.LastUpdatedImage.Digest] if repoLastUpdatedTimestamp == nil { repoLastUpdatedTimestamp = &time.Time{} } - imageSummary, _, err := FullImageData2ImageSummary(ctx, mTypes.FullImageData{ - Repo: repoMeta.Name, - Tag: repoMeta.LastUpdatedImage.Tag, - MediaType: repoMeta.LastUpdatedImage.MediaType, - Digest: godigest.Digest(repoMeta.LastUpdatedImage.Digest), - Size: lastUpdatedImageData.Size, - Index: lastUpdatedImageData.Index, - Manifests: getFullManifestData(repoMeta, lastUpdatedImageData.Manifests), - Referrers: repoMeta.Referrers[repoMeta.LastUpdatedImage.Digest], - Statistics: repoMeta.Statistics[repoMeta.LastUpdatedImage.Digest], - Signatures: repoMeta.Signatures[repoMeta.LastUpdatedImage.Digest], - }) - + imageSummary, _, err := ImageMeta2ImageSummary(ctx, GetImageMeta(lastUpdatedTag, repoMeta, lastUpdatedImageData)) _ = err return &gql_generated.RepoSummary{ @@ -1208,23 +351,8 @@ func FullRepoMeta2RepoSummary(ctx context.Context, repoMeta mTypes.FullRepoMetad StarCount: &repoStarCount, IsBookmarked: &repoIsUserBookMarked, IsStarred: &repoIsUserStarred, - Rank: ref(int(repoMeta.Rank)), - } -} - -func getFullManifestData(repoMeta mTypes.FullRepoMetadata, manifests []mTypes.ManifestData2) []mTypes.FullManifestData { - results := make([]mTypes.FullManifestData, 0, len(manifests)) - - for i := range manifests { - results = append(results, mTypes.FullManifestData{ - ManifestData2: manifests[i], - Referrers: repoMeta.Referrers[manifests[i].Digest.String()], - Statistics: repoMeta.Statistics[manifests[i].Digest.String()], - Signatures: repoMeta.Signatures[manifests[i].Digest.String()], - }) + Rank: ref(repoMeta.Rank), } - - return results } func getGqlVendors(repoVendors []string) []*string { @@ -1255,141 +383,42 @@ type ( BlobDigest = string ) -func ProtoDescriptor2ImageSummary(ctx context.Context, descriptor mTypes.Descriptor, repo, tag string, - repoMeta mTypes.RepoMetadata2, imageDataMap map[ManifestDigest]mTypes.ImageData2, -) (*gql_generated.ImageSummary, map[BlobDigest]int64, error) { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - return ProtoImageManifest2ImageSummary(ctx, repo, tag, repoMeta, imageDataMap[descriptor.Digest]) - case ispec.MediaTypeImageIndex: - return ProtoImageIndex2ImageSummary(ctx, repo, tag, descriptor.Digest, repoMeta, imageDataMap) - default: - // TODO: - return nil, nil, nil - } -} - -func FullImageData2ImageSummary(ctx context.Context, imageData mTypes.FullImageData, +func ImageMeta2ImageSummary(ctx context.Context, imageMeta mTypes.ImageMeta, ) (*gql_generated.ImageSummary, map[BlobDigest]int64, error) { - switch imageData.MediaType { + switch imageMeta.MediaType { case ispec.MediaTypeImageManifest: - return FullImageManifest2ImageSummary(ctx, imageData) + return ImageManifest2ImageSummary(ctx, imageMeta) case ispec.MediaTypeImageIndex: - return FullImageIndex2ImageSummary(ctx, imageData) + return ImageIndex2ImageSummary(ctx, imageMeta) default: - // TODO: - return nil, nil, nil - } -} - -func ProtoImageIndex2ImageSummary(ctx context.Context, repo, tag, digest string, repoMeta mTypes.RepoMetadata2, - imageDataMap map[ManifestDigest]mTypes.ImageData2, -) (*gql_generated.ImageSummary, map[BlobDigest]int64, error) { - imageIndex := imageDataMap[digest] - - var ( - indexLastUpdated time.Time - isSigned = isImageSigned(repoMeta.Signatures[digest]) - indexSize int64 - manifestAnnotations *ImageAnnotations - manifestSummaries = make([]*gql_generated.ManifestSummary, 0, len(imageIndex.Manifests)) - indexBlobs = make(map[string]int64, 0) - - indexDigestStr = digest - indexMediaType = ispec.MediaTypeImageIndex - ) - - for _, imageManifest := range imageIndex.Manifests { - imageManifestSummary, manifestBlobs, err := ProtoImageManifest2ImageSummary(ctx, repo, tag, repoMeta, mTypes.ImageData2{ - MediaType: ispec.MediaTypeImageManifest, - Digest: imageManifest.Digest, - Size: imageIndex.Size, - Manifests: []mTypes.ManifestData2{imageManifest}, - }) - if err != nil { - return &gql_generated.ImageSummary{}, map[string]int64{}, err - } - - manifestSize := int64(0) - - for digest, size := range manifestBlobs { - indexBlobs[digest] = size - manifestSize += size - } - - if indexLastUpdated.Before(*imageManifestSummary.LastUpdated) { - indexLastUpdated = *imageManifestSummary.LastUpdated - } - - annotations := GetAnnotations(imageManifest.Annotations, imageManifest.ConfigContent.Config.Labels) - - if manifestAnnotations == nil { - manifestAnnotations = &annotations - } - - indexSize += manifestSize - - manifestSummaries = append(manifestSummaries, imageManifestSummary.Manifests[0]) - } - - signaturesInfo := ProtoGetSignaturesInfo(isSigned, repoMeta, godigest.Digest(digest)) - - if manifestAnnotations == nil { - // The index doesn't have manifests - manifestAnnotations = &ImageAnnotations{} - } - - annotations := GetIndexAnnotations(imageIndex.Index.Annotations, manifestAnnotations) - - indexSummary := gql_generated.ImageSummary{ - RepoName: &repo, - Tag: &tag, - Digest: &indexDigestStr, - MediaType: &indexMediaType, - Manifests: manifestSummaries, - LastUpdated: &indexLastUpdated, - IsSigned: &isSigned, - SignatureInfo: signaturesInfo, - Size: ref(strconv.FormatInt(indexSize, 10)), - DownloadCount: ref(int(repoMeta.Statistics[indexDigestStr].DownloadCount)), - Description: &annotations.Description, - Title: &annotations.Title, - Documentation: &annotations.Documentation, - Licenses: &annotations.Licenses, - Labels: &annotations.Labels, - Source: &annotations.Source, - Vendor: &annotations.Vendor, - Authors: &annotations.Authors, - Referrers: getReferrers(repoMeta.Referrers[digest]), + return nil, nil, zerr.ErrMediaTypeNotSupported } - - return &indexSummary, indexBlobs, nil } -func FullImageIndex2ImageSummary(ctx context.Context, imageData mTypes.FullImageData, +func ImageIndex2ImageSummary(ctx context.Context, imageMeta mTypes.ImageMeta, ) (*gql_generated.ImageSummary, map[BlobDigest]int64, error) { var ( - repo = imageData.Repo - tag = imageData.Tag + repo = imageMeta.Repo + tag = imageMeta.Tag indexLastUpdated time.Time - isSigned = isImageSigned(imageData.Signatures) + isSigned = isImageSigned(imageMeta.Signatures) indexSize = int64(0) manifestAnnotations *ImageAnnotations - manifestSummaries = make([]*gql_generated.ManifestSummary, 0, len(imageData.Manifests)) + manifestSummaries = make([]*gql_generated.ManifestSummary, 0, len(imageMeta.Manifests)) indexBlobs = map[string]int64{} - indexDigestStr = imageData.Digest.String() + indexDigestStr = imageMeta.Digest.String() indexMediaType = ispec.MediaTypeImageIndex ) - for _, imageManifest := range imageData.Manifests { - imageManifestSummary, manifestBlobs, err := FullImageManifest2ImageSummary(ctx, mTypes.FullImageData{ - Repo: imageData.Repo, - Tag: imageData.Tag, + for _, imageManifest := range imageMeta.Manifests { + imageManifestSummary, manifestBlobs, err := ImageManifest2ImageSummary(ctx, mTypes.ImageMeta{ + Repo: imageMeta.Repo, + Tag: imageMeta.Tag, MediaType: ispec.MediaTypeImageManifest, Digest: imageManifest.Digest, Size: imageManifest.Size, - Manifests: []mTypes.FullManifestData{imageManifest}, + Manifests: []mTypes.ManifestMeta{imageManifest}, Referrers: imageManifest.Referrers, Statistics: imageManifest.Statistics, Signatures: imageManifest.Signatures, @@ -1399,6 +428,7 @@ func FullImageIndex2ImageSummary(ctx context.Context, imageData mTypes.FullImage } manifestSize := int64(0) + for digest, size := range manifestBlobs { indexBlobs[digest] = size manifestSize += size @@ -1408,7 +438,7 @@ func FullImageIndex2ImageSummary(ctx context.Context, imageData mTypes.FullImage indexLastUpdated = *imageManifestSummary.LastUpdated } - annotations := GetAnnotations(imageManifest.Annotations, imageManifest.ConfigContent.Config.Labels) + annotations := GetAnnotations(imageManifest.Manifest.Annotations, imageManifest.Config.Config.Labels) if manifestAnnotations == nil { manifestAnnotations = &annotations } @@ -1418,13 +448,13 @@ func FullImageIndex2ImageSummary(ctx context.Context, imageData mTypes.FullImage manifestSummaries = append(manifestSummaries, imageManifestSummary.Manifests[0]) } - signaturesInfo := FullGetSignaturesInfo(isSigned, imageData.Signatures, godigest.Digest(imageData.Digest)) + signaturesInfo := GetSignaturesInfo(isSigned, imageMeta.Signatures) if manifestAnnotations == nil { manifestAnnotations = &ImageAnnotations{} } - annotations := GetIndexAnnotations(imageData.Index.Annotations, manifestAnnotations) + annotations := GetIndexAnnotations(imageMeta.Index.Annotations, manifestAnnotations) indexSummary := gql_generated.ImageSummary{ RepoName: &repo, @@ -1436,7 +466,7 @@ func FullImageIndex2ImageSummary(ctx context.Context, imageData mTypes.FullImage IsSigned: &isSigned, SignatureInfo: signaturesInfo, Size: ref(strconv.FormatInt(indexSize, 10)), - DownloadCount: ref(int(imageData.Statistics.DownloadCount)), + DownloadCount: ref(imageMeta.Statistics.DownloadCount), Description: &annotations.Description, Title: &annotations.Title, Documentation: &annotations.Documentation, @@ -1445,123 +475,48 @@ func FullImageIndex2ImageSummary(ctx context.Context, imageData mTypes.FullImage Source: &annotations.Source, Vendor: &annotations.Vendor, Authors: &annotations.Authors, - Referrers: getReferrers(imageData.Referrers), + Referrers: getReferrers(imageMeta.Referrers), } return &indexSummary, indexBlobs, nil } -func ProtoImageManifest2ImageSummary(ctx context.Context, repo, tag string, repoMeta mTypes.RepoMetadata2, - imageData mTypes.ImageData2, -) (*gql_generated.ImageSummary, map[BlobDigest]int64, error) { - manifest := imageData.Manifests[0] - - var ( - repoName = repo - configDigest = manifest.Config.Digest.String() - configSize = manifest.Config.Size - manifestDigest = manifest.Digest.String() - manifestSize = int64(manifest.Size) - mediaType = manifest.MediaType - artifactType = zcommon.GetManifestArtifactType(imageData.Manifests[0].Manifest) - platform = getPlatform(manifest.ConfigContent.Platform) - imageLastUpdated = zcommon.GetImageLastUpdated(manifest.ConfigContent) // TODO: we can cache this - downloadCount = int(repoMeta.Statistics[manifest.Digest.String()].DownloadCount) - isSigned = isImageSigned(repoMeta.Signatures[manifest.Digest.String()]) - ) - - imageSize, imageBlobsMap := getImageBlobsInfo(manifestDigest, manifestSize, configDigest, configSize, manifest.Layers) - imageSizeStr := strconv.FormatInt(imageSize, 10) - annotations := GetAnnotations(manifest.Annotations, manifest.ConfigContent.Config.Labels) // TODO: This can be cached - - authors := annotations.Authors - if authors == "" { - authors = manifest.ConfigContent.Author - } - - historyEntries, err := getAllHistory(manifest.Manifest, manifest.ConfigContent) // TODO: this can be cached - if err != nil { - graphql.AddError(ctx, gqlerror.Errorf("error generating history on tag %s in repo %s: "+ - "manifest digest: %s, error: %s", tag, repo, manifest.Digest, err.Error())) - } - - signaturesInfo := ProtoGetSignaturesInfo(isSigned, repoMeta, manifest.Digest) - - manifestSummary := gql_generated.ManifestSummary{ - Digest: &manifestDigest, - ConfigDigest: &configDigest, - LastUpdated: &imageLastUpdated, - Size: &imageSizeStr, - IsSigned: &isSigned, - SignatureInfo: signaturesInfo, - Platform: &platform, - DownloadCount: &downloadCount, - Layers: getLayersSummaries(manifest.Manifest), - History: historyEntries, - Referrers: getReferrers(repoMeta.Referrers[manifestDigest]), - ArtifactType: &artifactType, - } - - imageSummary := gql_generated.ImageSummary{ - RepoName: &repoName, - Tag: &tag, - Digest: &manifestDigest, - MediaType: &mediaType, - Manifests: []*gql_generated.ManifestSummary{&manifestSummary}, - LastUpdated: &imageLastUpdated, - IsSigned: &isSigned, - SignatureInfo: signaturesInfo, - Size: &imageSizeStr, - DownloadCount: &downloadCount, - Description: &annotations.Description, - Title: &annotations.Title, - Documentation: &annotations.Documentation, - Licenses: &annotations.Licenses, - Labels: &annotations.Labels, - Source: &annotations.Source, - Vendor: &annotations.Vendor, - Authors: &authors, - Referrers: manifestSummary.Referrers, - } - - return &imageSummary, imageBlobsMap, nil -} - -func FullImageManifest2ImageSummary(ctx context.Context, imageData mTypes.FullImageData, +func ImageManifest2ImageSummary(ctx context.Context, imageMeta mTypes.ImageMeta, ) (*gql_generated.ImageSummary, map[BlobDigest]int64, error) { - manifest := imageData.Manifests[0] + manifest := imageMeta.Manifests[0] var ( - repoName = imageData.Repo - tag = imageData.Tag - configDigest = manifest.Config.Digest.String() - configSize = manifest.Config.Size + repoName = imageMeta.Repo + tag = imageMeta.Tag + configDigest = manifest.Manifest.Config.Digest.String() + configSize = manifest.Manifest.Config.Size manifestDigest = manifest.Digest.String() - manifestSize = int64(manifest.Size) - mediaType = manifest.MediaType - artifactType = zcommon.GetManifestArtifactType(imageData.Manifests[0].Manifest) - platform = getPlatform(manifest.ConfigContent.Platform) - imageLastUpdated = zcommon.GetImageLastUpdated(manifest.ConfigContent) // TODO: we can cache this - downloadCount = imageData.Statistics.DownloadCount - isSigned = isImageSigned(imageData.Signatures) + manifestSize = manifest.Size + mediaType = manifest.Manifest.MediaType + artifactType = zcommon.GetManifestArtifactType(imageMeta.Manifests[0].Manifest) + platform = getPlatform(manifest.Config.Platform) + imageLastUpdated = zcommon.GetImageLastUpdated(manifest.Config) + downloadCount = imageMeta.Statistics.DownloadCount + isSigned = isImageSigned(imageMeta.Signatures) ) - imageSize, imageBlobsMap := getImageBlobsInfo(manifestDigest, manifestSize, configDigest, configSize, manifest.Layers) + imageSize, imageBlobsMap := getImageBlobsInfo(manifestDigest, manifestSize, configDigest, configSize, + manifest.Manifest.Layers) imageSizeStr := strconv.FormatInt(imageSize, 10) - annotations := GetAnnotations(manifest.Annotations, manifest.ConfigContent.Config.Labels) // TODO: This can be cached + annotations := GetAnnotations(manifest.Manifest.Annotations, manifest.Config.Config.Labels) authors := annotations.Authors if authors == "" { - authors = manifest.ConfigContent.Author + authors = manifest.Config.Author } - historyEntries, err := getAllHistory(manifest.Manifest, manifest.ConfigContent) // TODO: this can be cached + historyEntries, err := getAllHistory(manifest.Manifest, manifest.Config) if err != nil { graphql.AddError(ctx, gqlerror.Errorf("error generating history on tag %s in repo %s: "+ "manifest digest: %s, error: %s", tag, repoName, manifest.Digest, err.Error())) } - signaturesInfo := FullGetSignaturesInfo(isSigned, imageData.Signatures, manifest.Digest) + signaturesInfo := GetSignaturesInfo(isSigned, imageMeta.Signatures) manifestSummary := gql_generated.ManifestSummary{ Digest: &manifestDigest, @@ -1574,7 +529,7 @@ func FullImageManifest2ImageSummary(ctx context.Context, imageData mTypes.FullIm DownloadCount: &downloadCount, Layers: getLayersSummaries(manifest.Manifest), History: historyEntries, - Referrers: getReferrers(imageData.Referrers), + Referrers: getReferrers(imageMeta.Referrers), ArtifactType: &artifactType, } @@ -1603,109 +558,6 @@ func FullImageManifest2ImageSummary(ctx context.Context, imageData mTypes.FullIm return &imageSummary, imageBlobsMap, nil } -func getProtoReferrers(referrersInfo *proto_go.ReferrersInfo) []*gql_generated.Referrer { - referrers := make([]*gql_generated.Referrer, 0, len(referrersInfo.List)) - - for _, referrerInfo := range referrersInfo.List { - referrerInfo := referrerInfo - - referrers = append(referrers, &gql_generated.Referrer{ - MediaType: &referrerInfo.MediaType, - ArtifactType: &referrerInfo.ArtifactType, - Size: ref(int(referrerInfo.Size)), - Digest: &referrerInfo.Digest, - Annotations: getAnnotationsFromMap(referrerInfo.Annotations), - }) - } - - return referrers -} - -func getAllProtoHistory(manifest *proto_go.ManifestData) ([]*gql_generated.LayerHistory, error) { - allHistory := []*gql_generated.LayerHistory{} - layerSummaries := getProtoLayersSummaries(manifest) - - history := manifest.Config.History - if len(history) == 0 { - // We don't have any image history metadata - // let's make due with just the layer metadata - for _, layer := range layerSummaries { - allHistory = append(allHistory, &gql_generated.LayerHistory{ - Layer: layer, - HistoryDescription: &gql_generated.HistoryDescription{}, - }) - } - - return allHistory, nil - } - - // Iterator over manifest layers - var layersIterator int - // Since we are appending pointers, it is important to iterate with an index over slice - for i := range history { - allHistory = append(allHistory, &gql_generated.LayerHistory{ - HistoryDescription: &gql_generated.HistoryDescription{ - Created: ref(history[i].GetCreated().AsTime()), - CreatedBy: history[i].Createdby, - Author: history[i].Author, - Comment: history[i].Comment, - EmptyLayer: history[i].Emptylayer, - }, - }) - - if history[i].Emptylayer != nil && *history[i].Emptylayer { - continue - } - - if layersIterator+1 > len(manifest.Layers) { - return allHistory, zerr.ErrBadLayerCount - } - - allHistory[i].Layer = layerSummaries[layersIterator] - - layersIterator++ - } - - return allHistory, nil -} - -func getProtoLayersSummaries(manifest *proto_go.ManifestData) []*gql_generated.LayerSummary { - layers := make([]*gql_generated.LayerSummary, 0, len(manifest.Layers)) - - for _, layer := range manifest.Layers { - size := strconv.FormatInt(layer.Size, 10) - digest := layer.Digest - - layers = append(layers, &gql_generated.LayerSummary{ - Size: &size, - Digest: &digest, - }) - } - - return layers -} - -func getProtoImageBlobsInfo(manifest *proto_go.ManifestData) (string, map[string]int64) { - imageBlobsMap := map[string]int64{} - imageSize := int64(0) - - // add config size - imageSize += manifest.Config.Size - imageBlobsMap[manifest.Config.Digest] = manifest.Config.Size - - // add manifest size - imageSize += manifest.Size - imageBlobsMap[manifest.Digest] = manifest.Size - - // add layers size - for _, layer := range manifest.Layers { - imageBlobsMap[layer.Digest] = layer.Size - imageSize += layer.Size - } - - return strconv.FormatInt(imageSize, 10), imageBlobsMap -} - func isImageSigned(manifestSignatures mTypes.ManifestSignatures) bool { for _, signatures := range manifestSignatures { if len(signatures) > 0 { @@ -1737,83 +589,15 @@ func ref[T any](val T) *T { return &ref } -func GetProtoSignaturesInfo(isSigned bool, repoMeta *proto_go.ProtoRepoMeta, digest string, -) []*gql_generated.SignatureSummary { - signaturesInfo := []*gql_generated.SignatureSummary{} - - if !isSigned { - return signaturesInfo - } - - for sigType, signatures := range repoMeta.Signatures[digest].Map { - for _, sig := range signatures.List { - for _, layer := range sig.LayersInfo { - var ( - isTrusted bool - author string - tool string - ) - - if layer.Signer != "" { - author = layer.Signer - - if !layer.Date.AsTime().IsZero() && time.Now().After(layer.Date.AsTime()) { - isTrusted = false - } else { - isTrusted = true - } - } else { - isTrusted = false - author = "" - } - - tool = sigType - - signaturesInfo = append(signaturesInfo, - &gql_generated.SignatureSummary{Tool: &tool, IsTrusted: &isTrusted, Author: &author}) - } - } - } - - return signaturesInfo -} - -func PaginatedProtoRepoMeta2ImageSummaries(ctx context.Context, reposMeta []mTypes.RepoMetadata2, - imageData map[string]mTypes.ImageData2, skip SkipQGLField, cveInfo cveinfo.CveInfo, filter mTypes.Filter, - pageInput pagination.PageInput, -) ([]*gql_generated.ImageSummary, zcommon.PageInfo, error) { - imagePageFinder, err := pagination.NewImgSumPageFinder(pageInput.Limit, pageInput.Offset, pageInput.SortBy) - if err != nil { - return []*gql_generated.ImageSummary{}, zcommon.PageInfo{}, err - } - - for _, repoMeta := range reposMeta { - for tag := range repoMeta.Tags { - descriptor := repoMeta.Tags[tag] - - imageSummary, _, err := ProtoDescriptor2ImageSummary(ctx, descriptor, repoMeta.Name, tag, - repoMeta, imageData) - if err != nil { - continue - } - - if ImgSumAcceptedByFilter(imageSummary, filter) { - imagePageFinder.Add(imageSummary) - } - } - } - - page, pageInfo := imagePageFinder.Page() - - for _, imageSummary := range page { - // CVE scanning is expensive, only scan for this page - updateImageSummaryVulnerabilities(ctx, imageSummary, skip, cveInfo) +func deref[T any](pointer *T, defaultVal T) T { + if pointer != nil { + return *pointer } - return page, pageInfo, nil + return defaultVal } -func PaginatedFullImageData2ImageSummaries(ctx context.Context, imageDataList []mTypes.FullImageData, skip SkipQGLField, +func PaginatedImageMeta2ImageSummaries(ctx context.Context, imageMetaList []mTypes.ImageMeta, skip SkipQGLField, cveInfo cveinfo.CveInfo, filter mTypes.Filter, pageInput pagination.PageInput, ) ([]*gql_generated.ImageSummary, zcommon.PageInfo, error) { imagePageFinder, err := pagination.NewImgSumPageFinder(pageInput.Limit, pageInput.Offset, pageInput.SortBy) @@ -1821,8 +605,8 @@ func PaginatedFullImageData2ImageSummaries(ctx context.Context, imageDataList [] return []*gql_generated.ImageSummary{}, zcommon.PageInfo{}, err } - for _, imageData := range imageDataList { - imageSummary, _, err := FullImageData2ImageSummary(ctx, imageData) + for _, imageData := range imageMetaList { + imageSummary, _, err := ImageMeta2ImageSummary(ctx, imageData) if err != nil { continue } diff --git a/pkg/extensions/search/convert/oci.go b/pkg/extensions/search/convert/oci.go index a4f7a37b48..1df370f466 100644 --- a/pkg/extensions/search/convert/oci.go +++ b/pkg/extensions/search/convert/oci.go @@ -9,19 +9,6 @@ import ( "zotregistry.io/zot/pkg/extensions/search/gql_generated" ) -// updateRepoBlobsMap adds all the image blobs and their respective size to the repo blobs map -// and returnes the total size of the image. -func updateRepoBlobsMap(imageBlobs map[string]int64, repoBlob2Size map[string]int64) int64 { - imgSize := int64(0) - - for digest, size := range imageBlobs { - repoBlob2Size[digest] = size - imgSize += size - } - - return imgSize -} - func getLayersSummaries(manifestContent ispec.Manifest) []*gql_generated.LayerSummary { layers := make([]*gql_generated.LayerSummary, 0, len(manifestContent.Layers)) diff --git a/pkg/extensions/search/cve/cve.go b/pkg/extensions/search/cve/cve.go index f4747cff7c..ee6c38eda3 100644 --- a/pkg/extensions/search/cve/cve.go +++ b/pkg/extensions/search/cve/cve.go @@ -1,7 +1,7 @@ package cveinfo import ( - "encoding/json" + "context" "sort" "strings" "time" @@ -9,6 +9,7 @@ import ( godigest "github.com/opencontainers/go-digest" ispec "github.com/opencontainers/image-spec/specs-go/v1" + zerr "zotregistry.io/zot/errors" zcommon "zotregistry.io/zot/pkg/common" cvemodel "zotregistry.io/zot/pkg/extensions/search/cve/model" "zotregistry.io/zot/pkg/extensions/search/cve/trivy" @@ -57,7 +58,7 @@ func NewCVEInfo(scanner Scanner, metaDB mTypes.MetaDB, log log.Logger) *BaseCveI func (cveinfo BaseCveInfo) GetImageListForCVE(repo, cveID string) ([]cvemodel.TagInfo, error) { imgList := make([]cvemodel.TagInfo, 0) - repoMeta, err := cveinfo.MetaDB.GetRepoMeta(repo) + repoMeta, err := cveinfo.MetaDB.GetRepoMeta(context.Background(), repo) if err != nil { cveinfo.Log.Error().Err(err).Str("repository", repo).Str("cve-id", cveID). Msg("unable to get list of tags from repo") @@ -105,7 +106,7 @@ func (cveinfo BaseCveInfo) GetImageListForCVE(repo, cveID string) ([]cvemodel.Ta } func (cveinfo BaseCveInfo) GetImageListWithCVEFixed(repo, cveID string) ([]cvemodel.TagInfo, error) { - repoMeta, err := cveinfo.MetaDB.GetRepoMeta(repo) + repoMeta, err := cveinfo.MetaDB.GetRepoMeta(context.Background(), repo) if err != nil { cveinfo.Log.Error().Err(err).Str("repository", repo).Str("cve-id", cveID). Msg("unable to get list of tags from repo") @@ -287,19 +288,16 @@ func getIndexContent(metaDB mTypes.MetaDB, indexDigestStr string) (ispec.Index, return ispec.Index{}, err } - indexData, err := metaDB.GetIndexData(indexDigest) + indexData, err := metaDB.GetImageData(indexDigest) if err != nil { return ispec.Index{}, err } - var indexContent ispec.Index - - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return ispec.Index{}, err + if indexData.Index == nil { + return ispec.Index{}, zerr.ErrUnexpectedMediaType } - return indexContent, nil + return *indexData.Index, nil } func getConfigAndDigest(metaDB mTypes.MetaDB, manifestDigestStr string) (ispec.Image, godigest.Digest, error) { @@ -308,17 +306,17 @@ func getConfigAndDigest(metaDB mTypes.MetaDB, manifestDigestStr string) (ispec.I return ispec.Image{}, "", err } - manifestData, err := metaDB.GetManifestData(manifestDigest) + manifestData, err := metaDB.GetImageData(manifestDigest) if err != nil { return ispec.Image{}, "", err } - var configContent ispec.Image - - // we'll fail the execution if the config is not compatibe with ispec.Image because we can't scan this type of images. - err = json.Unmarshal(manifestData.ConfigBlob, &configContent) + // we'll fail the execution if the config is not compatible with ispec.Image because we can't scan this type of images. + if manifestData.Manifests[0].Manifest.Config.MediaType != ispec.MediaTypeImageConfig { + return ispec.Image{}, "", zerr.ErrUnexpectedMediaType + } - return configContent, manifestDigest, err + return manifestData.Manifests[0].Config, manifestDigest, err } func filterCVEList(cveMap map[string]cvemodel.CVE, searchedCVE string, pageFinder *CvePageFinder) { diff --git a/pkg/extensions/search/cve/cve_test.go b/pkg/extensions/search/cve/cve_test.go index 32d9059150..31364b7835 100644 --- a/pkg/extensions/search/cve/cve_test.go +++ b/pkg/extensions/search/cve/cve_test.go @@ -5,6 +5,7 @@ package cveinfo_test import ( + "context" "encoding/json" "fmt" "io" @@ -390,8 +391,8 @@ func TestImageFormat(t *testing.T) { log := log.NewLogger("debug", "") metaDB := &mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Tags: map[string]mTypes.Descriptor{ "tag": { MediaType: ispec.MediaTypeImageIndex, @@ -400,8 +401,12 @@ func TestImageFormat(t *testing.T) { }, }, nil }, - GetIndexDataFn: func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{IndexBlob: []byte(`{}`)}, nil + GetImageDataFn: func(digest godigest.Digest) (mTypes.ImageData, error) { + return mTypes.ImageData{ + MediaType: ispec.MediaTypeImageIndex, + Digest: godigest.FromString("digest"), + Index: &ispec.Index{}, + }, nil }, } storeController := storage.StoreController{ @@ -764,73 +769,32 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo image11 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2008, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta11 := mTypes.ManifestMetadata{ - ManifestBlob: image11.ManifestDescriptor.Data, - ConfigBlob: image11.ConfigDescriptor.Data, - DownloadCount: 0, - Signatures: mTypes.ManifestSignatures{}, - } - - err = metaDB.SetManifestMeta(repo1, image11.ManifestDescriptor.Digest, repoMeta11) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "0.1.0", image11.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "0.1.0", image11.AsImageData()) So(err, ShouldBeNil) image12 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2009, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta12 := mTypes.ManifestMetadata{ - ManifestBlob: image12.ManifestDescriptor.Data, - ConfigBlob: image12.ConfigDescriptor.Data, - DownloadCount: 0, - Signatures: mTypes.ManifestSignatures{}, - } - - err = metaDB.SetManifestMeta(repo1, image12.ManifestDescriptor.Digest, repoMeta12) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "1.0.0", image12.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "1.0.0", image12.AsImageData()) So(err, ShouldBeNil) image13 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2010, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta13 := mTypes.ManifestMetadata{ - ManifestBlob: image13.ManifestDescriptor.Data, - ConfigBlob: image13.ConfigDescriptor.Data, - DownloadCount: 0, - Signatures: mTypes.ManifestSignatures{}, - } - - err = metaDB.SetManifestMeta(repo1, image13.ManifestDescriptor.Digest, repoMeta13) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "1.1.0", image13.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "1.1.0", image13.AsImageData()) So(err, ShouldBeNil) image14 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2011, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta14 := mTypes.ManifestMetadata{ - ManifestBlob: image14.ManifestDescriptor.Data, - ConfigBlob: image14.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta(repo1, image14.ManifestDescriptor.Digest, repoMeta14) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "1.0.1", image14.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "1.0.1", image14.AsImageData()) So(err, ShouldBeNil) // Create metadb data for scannable image with no vulnerabilities image61 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2016, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta61 := mTypes.ManifestMetadata{ - ManifestBlob: image61.ManifestDescriptor.Data, - ConfigBlob: image61.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta(repo6, image61.ManifestDescriptor.Digest, repoMeta61) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo6, "1.0.0", image61.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo6, "1.0.0", image61.AsImageData()) So(err, ShouldBeNil) // Create metadb data for image not supporting scanning @@ -840,106 +804,59 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo Digest: godigest.FromBytes([]byte{10, 10, 10}), }}).ImageConfig(ispec.Image{Created: DateRef(2009, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta21 := mTypes.ManifestMetadata{ - ManifestBlob: image21.ManifestDescriptor.Data, - ConfigBlob: image21.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta(repo2, image21.ManifestDescriptor.Digest, repoMeta21) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, "1.0.0", image21.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo2, "1.0.0", image21.AsImageData()) So(err, ShouldBeNil) // Create metadb data for invalid images/negative tests - manifestBlob31 := []byte("invalid manifest blob") - So(err, ShouldBeNil) - - repoMeta31 := mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob31, - } - - digest31 := godigest.FromBytes(manifestBlob31) - err = metaDB.SetManifestMeta(repo3, digest31, repoMeta31) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo3, "invalid-manifest", digest31, ispec.MediaTypeImageManifest) + image := CreateRandomImage() + err = metaDB.SetRepoReference(repo3, "invalid-manifest", image.AsImageData()) So(err, ShouldBeNil) image41 := CreateImageWith().DefaultLayers(). CustomConfigBlob([]byte("invalid config blob"), ispec.MediaTypeImageConfig).Build() - repoMeta41 := mTypes.ManifestMetadata{ - ManifestBlob: image41.ManifestDescriptor.Data, - ConfigBlob: image41.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta(repo4, image41.ManifestDescriptor.Digest, repoMeta41) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo4, "invalid-config", image41.ManifestDescriptor.Digest, - ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo4, "invalid-config", image41.AsImageData()) So(err, ShouldBeNil) digest51 := godigest.FromString("abc8") - err = metaDB.SetRepoReference(repo5, "nonexitent-manifest", digest51, ispec.MediaTypeImageManifest) + randomImgData := CreateRandomImage().AsImageData() + randomImgData.Digest = digest51 + randomImgData.Manifests[0].Digest = digest51 + err = metaDB.SetRepoReference(repo5, "nonexitent-manifest", randomImgData) So(err, ShouldBeNil) // Create metadb data for scannable image which errors during scan image71 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2000, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta71 := mTypes.ManifestMetadata{ - ManifestBlob: image71.ManifestDescriptor.Data, - ConfigBlob: image71.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta(repo7, image71.ManifestDescriptor.Digest, repoMeta71) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo7, "1.0.0", image71.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo7, "1.0.0", image71.AsImageData()) So(err, ShouldBeNil) // create multiarch image with vulnerabilities multiarchImage := CreateRandomMultiarch() - err = metaDB.SetIndexData( - multiarchImage.IndexDescriptor.Digest, - mTypes.IndexData{IndexBlob: multiarchImage.IndexDescriptor.Data}, - ) + err = metaDB.SetRepoReference(repoMultiarch, multiarchImage.Images[0].DigestStr(), + multiarchImage.Images[0].AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData( - multiarchImage.Images[0].ManifestDescriptor.Digest, - mTypes.ManifestData{ - ManifestBlob: multiarchImage.Images[0].ManifestDescriptor.Data, - ConfigBlob: multiarchImage.Images[0].ConfigDescriptor.Data, - }, - ) + err = metaDB.SetRepoReference(repoMultiarch, multiarchImage.Images[1].DigestStr(), + multiarchImage.Images[1].AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData( - multiarchImage.Images[1].ManifestDescriptor.Digest, - mTypes.ManifestData{ - ManifestBlob: multiarchImage.Images[1].ManifestDescriptor.Data, - ConfigBlob: multiarchImage.Images[1].ConfigDescriptor.Data, - }, - ) + err = metaDB.SetRepoReference(repoMultiarch, multiarchImage.Images[2].DigestStr(), + multiarchImage.Images[2].AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData( - multiarchImage.Images[2].ManifestDescriptor.Digest, - mTypes.ManifestData{ - ManifestBlob: multiarchImage.Images[2].ManifestDescriptor.Data, - ConfigBlob: multiarchImage.Images[2].ConfigDescriptor.Data, - }, - ) + err = metaDB.SetRepoReference(repoMultiarch, "tagIndex", multiarchImage.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference( - repoMultiarch, - "tagIndex", - multiarchImage.IndexDescriptor.Digest, - ispec.MediaTypeImageIndex, - ) + err = metaDB.SetRepoMeta("repo-with-bad-tag-digest", mTypes.RepoMeta{ + Name: "repo-with-bad-tag-digest", + Tags: map[string]mTypes.Descriptor{ + "tag": {MediaType: ispec.MediaTypeImageManifest, Digest: godigest.FromString("1").String()}, + }, + }) So(err, ShouldBeNil) - // Keep a record of all the image references / digest pairings // This is normally done in MetaDB, but we want to verify // the whole flow, including MetaDB @@ -965,18 +882,6 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo image21Media := image21.ManifestDescriptor.MediaType image21Name := repo2 + ":1.0.0" imageMap[image21Name] = image21Digest - image31Digest := digest31.String() - image31Media := ispec.MediaTypeImageManifest - image31Name := repo3 + ":invalid-manifest" - imageMap[image31Name] = image31Digest - image41Digest := image41.ManifestDescriptor.Digest.String() - image41Media := image41.ManifestDescriptor.MediaType - image41Name := repo4 + ":invalid-config" - imageMap[image41Name] = image41Digest - image51Digest := digest51.String() - image51Media := ispec.MediaTypeImageManifest - image51Name := repo5 + ":nonexitent-manifest" - imageMap[image51Name] = digest51.String() image61Digest := image61.ManifestDescriptor.Digest.String() image61Media := image61.ManifestDescriptor.MediaType image61Name := repo6 + ":1.0.0" @@ -1150,7 +1055,7 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo // Almost same logic compared to actual Trivy specific implementation imageDir, inputTag := repo, reference - repoMeta, err := metaDB.GetRepoMeta(imageDir) + repoMeta, err := metaDB.GetRepoMeta(context.Background(), imageDir) if err != nil { return false, err } @@ -1173,19 +1078,12 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo return false, err } - manifestData, err := metaDB.GetManifestData(manifestDigest) + manifestData, err := metaDB.GetImageData(manifestDigest) if err != nil { return false, err } - var manifestContent ispec.Manifest - - err = json.Unmarshal(manifestData.ManifestBlob, &manifestContent) - if err != nil { - return false, zerr.ErrScanNotSupported - } - - for _, imageLayer := range manifestContent.Layers { + for _, imageLayer := range manifestData.Manifests[0].Manifest.Layers { switch imageLayer.MediaType { case ispec.MediaTypeImageLayerGzip, ispec.MediaTypeImageLayer, string(regTypes.DockerLayer): @@ -1202,12 +1100,6 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo if repo == repo2 && digest == image21Digest { return false, zerr.ErrScanNotSupported } - if repo == repo3 && digest == image31Digest { - return false, zerr.ErrTagMetaNotFound - } - if repo == repo5 && digest == image51Digest { - return false, zerr.ErrManifestDataNotFound - } if repo == repo100 { return false, zerr.ErrRepoMetaNotFound } @@ -1294,23 +1186,16 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo So(pageInfo.ItemCount, ShouldEqual, 0) So(pageInfo.TotalCount, ShouldEqual, 0) - // Config not valid - cveList, pageInfo, err = cveInfo.GetCVEListForImage(repo4, "invalid-config", "", pageInput) - So(err, ShouldBeNil) - So(len(cveList), ShouldEqual, 0) - So(pageInfo.ItemCount, ShouldEqual, 0) - So(pageInfo.TotalCount, ShouldEqual, 0) - - // Manifest is not found - cveList, pageInfo, err = cveInfo.GetCVEListForImage(repo5, "nonexitent-manifest", "", pageInput) - So(err, ShouldEqual, zerr.ErrManifestDataNotFound) + // Scan failed + cveList, pageInfo, err = cveInfo.GetCVEListForImage(repo7, "1.0.0", "", pageInput) + So(err, ShouldEqual, ErrFailedScan) So(len(cveList), ShouldEqual, 0) So(pageInfo.ItemCount, ShouldEqual, 0) So(pageInfo.TotalCount, ShouldEqual, 0) - // Scan failed - cveList, pageInfo, err = cveInfo.GetCVEListForImage(repo7, "1.0.0", "", pageInput) - So(err, ShouldEqual, ErrFailedScan) + // Tag is not found + cveList, pageInfo, err = cveInfo.GetCVEListForImage("repo-with-bad-tag-digest", "tag", "", pageInput) + So(err, ShouldEqual, zerr.ErrImageDataNotFound) So(len(cveList), ShouldEqual, 0) So(pageInfo.ItemCount, ShouldEqual, 0) So(pageInfo.TotalCount, ShouldEqual, 0) @@ -1363,24 +1248,6 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo So(cveSummary.Count, ShouldEqual, 0) So(cveSummary.MaxSeverity, ShouldEqual, "") - // Tag is not found - cveSummary, err = cveInfo.GetCVESummaryForImageMedia(repo3, image31Digest, image31Media) - So(err, ShouldEqual, zerr.ErrTagMetaNotFound) - So(cveSummary.Count, ShouldEqual, 0) - So(cveSummary.MaxSeverity, ShouldEqual, "") - - // Config not valid - cveSummary, err = cveInfo.GetCVESummaryForImageMedia(repo4, image41Digest, image41Media) - So(err, ShouldBeNil) - So(cveSummary.Count, ShouldEqual, 0) - So(cveSummary.MaxSeverity, ShouldEqual, "NONE") - - // Manifest is not found - cveSummary, err = cveInfo.GetCVESummaryForImageMedia(repo5, image51Digest, image51Media) - So(err, ShouldEqual, zerr.ErrManifestDataNotFound) - So(cveSummary.Count, ShouldEqual, 0) - So(cveSummary.MaxSeverity, ShouldEqual, "") - // Scan failed cveSummary, err = cveInfo.GetCVESummaryForImageMedia(repo5, image71Digest, image71Media) So(err, ShouldBeNil) @@ -1429,16 +1296,6 @@ func TestCVEStruct(t *testing.T) { //nolint:gocyclo So(err, ShouldBeNil) So(len(tagList), ShouldEqual, 0) - // Tag is not found, but we should not error - tagList, err = cveInfo.GetImageListWithCVEFixed(repo3, "CVE101") - So(err, ShouldBeNil) - So(len(tagList), ShouldEqual, 0) - - // Manifest is not found, we just consider exclude it from the fixed list - tagList, err = cveInfo.GetImageListWithCVEFixed(repo5, "CVE101") - So(err, ShouldBeNil) - So(len(tagList), ShouldEqual, 0) - // Repo is not found, there could potentially be unaffected tags in the repo // but we can't access their data tagList, err = cveInfo.GetImageListWithCVEFixed(repo100, "CVE100") @@ -1718,158 +1575,6 @@ func TestFixedTagsWithIndex(t *testing.T) { }) } -func TestImageListWithCVEFixedErrors(t *testing.T) { - indexDigest := godigest.FromString("index") - manifestDigest := "sha256:1111111111111111111111111111111111111111111111111111111111111111" - - Convey("Errors", t, func() { - storeController := storage.StoreController{} - storeController.DefaultStore = mocks.MockedImageStore{} - - metaDB := mocks.MetaDBMock{} - log := log.NewLogger("debug", "") - - Convey("getIndexContent errors", func() { - metaDB.GetRepoMetaFn = func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: indexDigest.String(), - MediaType: ispec.MediaTypeImageIndex, - }, - }, - }, nil - } - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{}, zerr.ErrIndexDataNotFount - } - - scanner := cveinfo.NewScanner(storeController, metaDB, "", "", log) - cveInfo := cveinfo.NewCVEInfo(scanner, metaDB, log) - - _, err := cveInfo.GetImageListWithCVEFixed("repo", Vulnerability1ID) - So(err, ShouldBeNil) - }) - - Convey("getIndexContent bad indexDigest", func() { - metaDB.GetRepoMetaFn = func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: "bad digest", - MediaType: ispec.MediaTypeImageIndex, - }, - }, - }, nil - } - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{}, zerr.ErrIndexDataNotFount - } - - scanner := cveinfo.NewScanner(storeController, metaDB, "", "", log) - cveInfo := cveinfo.NewCVEInfo(scanner, metaDB, log) - - _, err := cveInfo.GetImageListWithCVEFixed("repo", Vulnerability1ID) - So(err, ShouldBeNil) - }) - - Convey("getIndexContent bad index content", func() { - metaDB.GetRepoMetaFn = func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: indexDigest.String(), - MediaType: ispec.MediaTypeImageIndex, - }, - }, - }, nil - } - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{IndexBlob: []byte(`bad index`)}, nil - } - - scanner := cveinfo.NewScanner(storeController, metaDB, "", "", log) - cveInfo := cveinfo.NewCVEInfo(scanner, metaDB, log) - - _, err := cveInfo.GetImageListWithCVEFixed("repo", Vulnerability1ID) - So(err, ShouldBeNil) - }) - - Convey("getTagInfoForManifest bad manifest digest", func() { - metaDB.GetRepoMetaFn = func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: "bad digest", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - }, nil - } - - scanner := cveinfo.NewScanner(storeController, metaDB, "", "", log) - cveInfo := cveinfo.NewCVEInfo(scanner, metaDB, log) - - _, err := cveInfo.GetImageListWithCVEFixed("repo", Vulnerability1ID) - So(err, ShouldBeNil) - }) - - Convey("getTagInfoForManifest fails for index", func() { - metaDB.GetRepoMetaFn = func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: indexDigest.String(), - MediaType: ispec.MediaTypeImageIndex, - }, - }, - }, nil - } - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{ - IndexBlob: []byte(fmt.Sprintf(`{ - "manifests": [ - { - "digest": "%s", - "mediaType": "application/vnd.oci.image.manifest.v1+json" - } - ]}`, manifestDigest)), - }, nil - } - metaDB.GetManifestDataFn = func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, zerr.ErrManifestDataNotFound - } - - scanner := cveinfo.NewScanner(storeController, metaDB, "", "", log) - cveInfo := cveinfo.NewCVEInfo(scanner, metaDB, log) - - tagsInfo, err := cveInfo.GetImageListWithCVEFixed("repo", Vulnerability1ID) - So(err, ShouldBeNil) - So(tagsInfo, ShouldBeEmpty) - }) - - Convey("media type not supported", func() { - metaDB.GetRepoMetaFn = func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: godigest.FromString("media type").String(), - MediaType: "bad media type", - }, - }, - }, nil - } - - scanner := cveinfo.NewScanner(storeController, metaDB, "", "", log) - cveInfo := cveinfo.NewCVEInfo(scanner, metaDB, log) - - tagsInfo, err := cveInfo.GetImageListWithCVEFixed("repo", Vulnerability1ID) - So(err, ShouldBeNil) - So(tagsInfo, ShouldBeEmpty) - }) - }) -} - func TestGetCVESummaryForImageMediaErrors(t *testing.T) { Convey("Errors", t, func() { storeController := storage.StoreController{} diff --git a/pkg/extensions/search/cve/pagination_test.go b/pkg/extensions/search/cve/pagination_test.go index ab518c2cd0..df8f1d1b9e 100644 --- a/pkg/extensions/search/cve/pagination_test.go +++ b/pkg/extensions/search/cve/pagination_test.go @@ -4,13 +4,11 @@ package cveinfo_test import ( - "encoding/json" "fmt" "sort" "testing" "time" - godigest "github.com/opencontainers/go-digest" ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" @@ -18,7 +16,7 @@ import ( cvemodel "zotregistry.io/zot/pkg/extensions/search/cve/model" "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta/boltdb" - mTypes "zotregistry.io/zot/pkg/meta/types" + . "zotregistry.io/zot/pkg/test/image-utils" "zotregistry.io/zot/pkg/test/mocks" ) @@ -36,70 +34,26 @@ func TestCVEPagination(t *testing.T) { // Create metadb data for scannable image with vulnerabilities timeStamp11 := time.Date(2008, 1, 1, 12, 0, 0, 0, time.UTC) - configBlob11, err := json.Marshal(ispec.Image{ - Created: &timeStamp11, - }) - So(err, ShouldBeNil) + image := CreateImageWith(). + Layers([]Layer{{ + MediaType: ispec.MediaTypeImageLayerGzip, + Digest: ispec.DescriptorEmptyJSON.Digest, + Blob: ispec.DescriptorEmptyJSON.Data, + }}).ImageConfig(ispec.Image{Created: &timeStamp11}).Build() - manifestBlob11, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, - Size: 0, - Digest: godigest.FromBytes(configBlob11), - }, - Layers: []ispec.Descriptor{ - { - MediaType: ispec.MediaTypeImageLayerGzip, - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), - }, - }, - }) - So(err, ShouldBeNil) - - repoMeta11 := mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob11, - ConfigBlob: configBlob11, - } - - digest11 := godigest.FromBytes(manifestBlob11) - err = metaDB.SetManifestMeta("repo1", digest11, repoMeta11) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo1", "0.1.0", digest11, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo1", "0.1.0", image.AsImageData()) So(err, ShouldBeNil) timeStamp12 := time.Date(2009, 1, 1, 12, 0, 0, 0, time.UTC) - configBlob12, err := json.Marshal(ispec.Image{ - Created: &timeStamp12, - }) - So(err, ShouldBeNil) + image2 := CreateImageWith(). + Layers([]Layer{{ + MediaType: ispec.MediaTypeImageLayerGzip, + Digest: ispec.DescriptorEmptyJSON.Digest, + Blob: ispec.DescriptorEmptyJSON.Data, + }}).ImageConfig(ispec.Image{Created: &timeStamp12}).Build() - manifestBlob12, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, - Size: 0, - Digest: godigest.FromBytes(configBlob12), - }, - Layers: []ispec.Descriptor{ - { - MediaType: ispec.MediaTypeImageLayerGzip, - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), - }, - }, - }) - So(err, ShouldBeNil) - - repoMeta12 := mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob12, - ConfigBlob: configBlob12, - } - - digest12 := godigest.FromBytes(manifestBlob12) - err = metaDB.SetManifestMeta("repo1", digest12, repoMeta12) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo1", "1.0.0", digest12, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo1", "1.0.0", image2.AsImageData()) So(err, ShouldBeNil) // MetaDB loaded with initial data, mock the scanner diff --git a/pkg/extensions/search/cve/scan.go b/pkg/extensions/search/cve/scan.go index cacf084ad7..e0a67bc8ca 100644 --- a/pkg/extensions/search/cve/scan.go +++ b/pkg/extensions/search/cve/scan.go @@ -4,8 +4,6 @@ import ( "context" "sync" - godigest "github.com/opencontainers/go-digest" - "zotregistry.io/zot/pkg/log" mTypes "zotregistry.io/zot/pkg/meta/types" reqCtx "zotregistry.io/zot/pkg/requestcontext" @@ -43,13 +41,13 @@ type scanTaskGenerator struct { } func (gen *scanTaskGenerator) getMatcherFunc() mTypes.FilterFunc { - return func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { + return func(repoMeta mTypes.RepoMeta, imageData mTypes.ImageData) bool { // Note this matcher will return information based on scan status of manifests // An index scan aggregates results of manifest scans // If at least one of its manifests can be scanned, // the index and its tag will be returned by the caller function too repoName := repoMeta.Name - manifestDigest := godigest.FromBytes(manifestMeta.ManifestBlob).String() + manifestDigest := imageData.Digest.String() if gen.isScheduled(manifestDigest) { // We skip this manifest as it has already scheduled @@ -121,18 +119,18 @@ func (gen *scanTaskGenerator) Next() (scheduler.Task, error) { userAc.SetIsAdmin(true) ctx := userAc.DeriveContext(context.Background()) - // Obtain a list of repos with unscanned scannable manifests + // Obtain a list of repos with un-scanned scannable manifests // We may implement a method to return just 1 match at some point - reposMeta, _, _, err := gen.metaDB.FilterTags(ctx, gen.getMatcherFunc()) + imageData, err := gen.metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, gen.getMatcherFunc()) if err != nil { - // Do not crash the generator for potential repodb inconistencies + // Do not crash the generator for potential metadb inconsistencies // as there may be scannable images not yet scanned gen.log.Warn().Err(err).Msg("Scheduled CVE scan: error while obtaining repo metadata") } - // no reposMeta are returned, all results are in already in cache + // no imageData are returned, all results are in already in cache // or manifests cannot be scanned - if len(reposMeta) == 0 { + if len(imageData) == 0 { gen.log.Info().Msg("Scheduled CVE scan: finished for available images") gen.done = true @@ -140,23 +138,14 @@ func (gen *scanTaskGenerator) Next() (scheduler.Task, error) { return nil, nil } - // Since reposMeta will always contain just unscanned images we can pick - // any repo and any tag out of the resulting matches - repoMeta := reposMeta[0] - - var digest string - - // Pick any tag - for _, descriptor := range repoMeta.Tags { - digest = descriptor.Digest - - break - } + // Since imageData will always contain just un-scanned images we can pick + // any image out of the resulting matches + digest := imageData[0].Digest.String() // Mark the digest as scheduled so it is skipped on next generator run gen.setScheduled(digest, true) - return newScanTask(gen, repoMeta.Name, digest), nil + return newScanTask(gen, imageData[0].Repo, digest), nil } func (gen *scanTaskGenerator) IsDone() bool { diff --git a/pkg/extensions/search/cve/scan_test.go b/pkg/extensions/search/cve/scan_test.go index ccb45dfc8b..1731f08be2 100644 --- a/pkg/extensions/search/cve/scan_test.go +++ b/pkg/extensions/search/cve/scan_test.go @@ -5,7 +5,6 @@ package cveinfo_test import ( "context" - "encoding/json" "errors" "io" "os" @@ -69,77 +68,38 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo metaDB, err := boltdb.New(boltDriver, log.NewLogger("debug", "")) So(err, ShouldBeNil) + // Refactor Idea: We can use InitializeTestMetaDB + // Create metadb data for scannable image with vulnerabilities image11 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2008, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta11 := mTypes.ManifestMetadata{ - ManifestBlob: image11.ManifestDescriptor.Data, - ConfigBlob: image11.ConfigDescriptor.Data, - DownloadCount: 0, - Signatures: mTypes.ManifestSignatures{}, - } - - err = metaDB.SetManifestMeta("repo1", image11.ManifestDescriptor.Digest, repoMeta11) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo1", "0.1.0", image11.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo1", "0.1.0", image11.AsImageData()) So(err, ShouldBeNil) image12 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2009, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta12 := mTypes.ManifestMetadata{ - ManifestBlob: image12.ManifestDescriptor.Data, - ConfigBlob: image12.ConfigDescriptor.Data, - DownloadCount: 0, - Signatures: mTypes.ManifestSignatures{}, - } - - err = metaDB.SetManifestMeta("repo1", image12.ManifestDescriptor.Digest, repoMeta12) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo1", "1.0.0", image12.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo1", "1.0.0", image12.AsImageData()) So(err, ShouldBeNil) image13 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2010, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta13 := mTypes.ManifestMetadata{ - ManifestBlob: image13.ManifestDescriptor.Data, - ConfigBlob: image13.ConfigDescriptor.Data, - DownloadCount: 0, - Signatures: mTypes.ManifestSignatures{}, - } - - err = metaDB.SetManifestMeta("repo1", image13.ManifestDescriptor.Digest, repoMeta13) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo1", "1.1.0", image13.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo1", "1.1.0", image13.AsImageData()) So(err, ShouldBeNil) image14 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2011, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta14 := mTypes.ManifestMetadata{ - ManifestBlob: image14.ManifestDescriptor.Data, - ConfigBlob: image14.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta("repo1", image14.ManifestDescriptor.Digest, repoMeta14) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo1", "1.0.1", image14.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo1", "1.0.1", image14.AsImageData()) So(err, ShouldBeNil) // Create metadb data for scannable image with no vulnerabilities image61 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2016, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta61 := mTypes.ManifestMetadata{ - ManifestBlob: image61.ManifestDescriptor.Data, - ConfigBlob: image61.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta("repo6", image61.ManifestDescriptor.Digest, repoMeta61) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo6", "1.0.0", image61.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo6", "1.0.0", image61.AsImageData()) So(err, ShouldBeNil) // Create metadb data for image not supporting scanning @@ -149,104 +109,58 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo Digest: godigest.FromBytes([]byte{10, 10, 10}), }}).ImageConfig(ispec.Image{Created: DateRef(2009, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta21 := mTypes.ManifestMetadata{ - ManifestBlob: image21.ManifestDescriptor.Data, - ConfigBlob: image21.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta("repo2", image21.ManifestDescriptor.Digest, repoMeta21) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo2", "1.0.0", image21.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo2", "1.0.0", image21.AsImageData()) So(err, ShouldBeNil) // Create metadb data for invalid images/negative tests - manifestBlob31 := []byte("invalid manifest blob") - So(err, ShouldBeNil) + img := CreateRandomImage() + digest31 := img.Digest() - repoMeta31 := mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob31, - } - - digest31 := godigest.FromBytes(manifestBlob31) - err = metaDB.SetManifestMeta("repo3", digest31, repoMeta31) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo3", "invalid-manifest", digest31, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo3", "invalid-manifest", img.AsImageData()) So(err, ShouldBeNil) image41 := CreateImageWith().DefaultLayers(). CustomConfigBlob([]byte("invalid config blob"), ispec.MediaTypeImageConfig).Build() - repoMeta41 := mTypes.ManifestMetadata{ - ManifestBlob: image41.ManifestDescriptor.Data, - ConfigBlob: image41.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta("repo4", image41.ManifestDescriptor.Digest, repoMeta41) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo4", "invalid-config", image41.ManifestDescriptor.Digest, - ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo4", "invalid-config", image41.AsImageData()) So(err, ShouldBeNil) - digest51 := godigest.FromString("abc8") - err = metaDB.SetRepoReference("repo5", "nonexitent-manifest", digest51, ispec.MediaTypeImageManifest) + image15 := CreateRandomMultiarch() + + digest51 := image15.Digest() + err = metaDB.SetRepoReference("repo5", "nonexitent-manifests-for-multiarch", image15.AsImageData()) So(err, ShouldBeNil) // Create metadb data for scannable image which errors during scan image71 := CreateImageWith().DefaultLayers(). ImageConfig(ispec.Image{Created: DateRef(2000, 1, 1, 12, 0, 0, 0, time.UTC)}).Build() - repoMeta71 := mTypes.ManifestMetadata{ - ManifestBlob: image71.ManifestDescriptor.Data, - ConfigBlob: image71.ConfigDescriptor.Data, - } - - err = metaDB.SetManifestMeta("repo7", image71.ManifestDescriptor.Digest, repoMeta71) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo7", "1.0.0", image71.ManifestDescriptor.Digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo7", "1.0.0", image71.AsImageData()) So(err, ShouldBeNil) // Create multiarch image with vulnerabilities multiarchImage := CreateRandomMultiarch() - err = metaDB.SetIndexData( - multiarchImage.IndexDescriptor.Digest, - mTypes.IndexData{IndexBlob: multiarchImage.IndexDescriptor.Data}, - ) + err = metaDB.SetRepoReference(repoIndex, multiarchImage.Images[0].DigestStr(), + multiarchImage.Images[0].AsImageData()) So(err, ShouldBeNil) - - err = metaDB.SetManifestData( - multiarchImage.Images[0].ManifestDescriptor.Digest, - mTypes.ManifestData{ - ManifestBlob: multiarchImage.Images[0].ManifestDescriptor.Data, - ConfigBlob: multiarchImage.Images[0].ConfigDescriptor.Data, - }, - ) + err = metaDB.SetRepoReference(repoIndex, multiarchImage.Images[1].DigestStr(), + multiarchImage.Images[1].AsImageData()) So(err, ShouldBeNil) - - err = metaDB.SetManifestData( - multiarchImage.Images[1].ManifestDescriptor.Digest, - mTypes.ManifestData{ - ManifestBlob: multiarchImage.Images[1].ManifestDescriptor.Data, - ConfigBlob: multiarchImage.Images[1].ConfigDescriptor.Data, - }, - ) + err = metaDB.SetRepoReference(repoIndex, multiarchImage.Images[2].DigestStr(), + multiarchImage.Images[2].AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData( - multiarchImage.Images[2].ManifestDescriptor.Digest, - mTypes.ManifestData{ - ManifestBlob: multiarchImage.Images[2].ManifestDescriptor.Data, - ConfigBlob: multiarchImage.Images[2].ConfigDescriptor.Data, - }, - ) + err = metaDB.SetRepoReference(repoIndex, "tagIndex", multiarchImage.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference( - repoIndex, - "tagIndex", - multiarchImage.IndexDescriptor.Digest, - ispec.MediaTypeImageIndex, - ) + err = metaDB.SetRepoMeta("repo-with-bad-tag-digest", mTypes.RepoMeta{ + Name: "repo-with-bad-tag-digest", + Tags: map[string]mTypes.Descriptor{ + "tag": {MediaType: ispec.MediaTypeImageManifest, Digest: godigest.FromString("1").String()}, + "tag-multi-arch": {MediaType: ispec.MediaTypeImageIndex, Digest: godigest.FromString("2").String()}, + }, + }) So(err, ShouldBeNil) // Keep a record of all the image references / digest pairings @@ -274,7 +188,7 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo image41Digest := image41.ManifestDescriptor.Digest.String() image41Name := "repo4:invalid-config" imageMap[image41Name] = image41Digest - image51Name := "repo5:nonexitent-manifest" + image51Name := "repo5:nonexitent-manifest-for-multiarch" imageMap[image51Name] = digest51.String() image61Digest := image61.ManifestDescriptor.Digest.String() image61Name := "repo6:1.0.0" @@ -296,7 +210,7 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo imageMap[indexM3Name] = indexM3Digest // Initialize a test CVE cache - cache := cvecache.NewCveCache(10, logger) + cache := cvecache.NewCveCache(20, logger) // MetaDB loaded with initial data, now mock the scanner // Setup test CVE data in mock scanner @@ -440,7 +354,7 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo // Almost same logic compared to actual Trivy specific implementation imageDir, inputTag := repo, reference - repoMeta, err := metaDB.GetRepoMeta(imageDir) + repoMeta, err := metaDB.GetRepoMeta(context.Background(), imageDir) if err != nil { return false, err } @@ -463,19 +377,12 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo return false, err } - manifestData, err := metaDB.GetManifestData(manifestDigest) + manifestData, err := metaDB.GetImageData(manifestDigest) if err != nil { return false, err } - var manifestContent ispec.Manifest - - err = json.Unmarshal(manifestData.ManifestBlob, &manifestContent) - if err != nil { - return false, zerr.ErrScanNotSupported - } - - for _, imageLayer := range manifestContent.Layers { + for _, imageLayer := range manifestData.Manifests[0].Manifest.Layers { switch imageLayer.MediaType { case ispec.MediaTypeImageLayerGzip, ispec.MediaTypeImageLayer, string(regTypes.DockerLayer): @@ -531,7 +438,7 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo // Make sure the scanner generator has completed despite errors found, err := test.ReadLogFileAndSearchString(logPath, - "Scheduled CVE scan: finished for available images", 20*time.Second) + "Scheduled CVE scan: finished for available images", 40*time.Second) So(err, ShouldBeNil) So(found, ShouldBeTrue) @@ -546,13 +453,12 @@ func TestScanGeneratorWithMockedData(t *testing.T) { //nolint: gocyclo t.Log("expecting " + image + " " + digestStr + " to be present in cache") So(scanner.IsResultCached(digestStr), ShouldBeTrue) } else { - // We don't cache results for unscannable manifests + // We don't cache results for un-scannable manifests t.Log("expecting " + image + " " + digestStr + " to be absent from cache") So(scanner.IsResultCached(digestStr), ShouldBeFalse) } } - // Make sure the scanner generator is catching the metadb error for repo5:nonexitent-manifest found, err = test.ReadLogFileAndSearchString(logPath, "Scheduled CVE scan: error while obtaining repo metadata", 20*time.Second) So(err, ShouldBeNil) diff --git a/pkg/extensions/search/cve/trivy/scanner.go b/pkg/extensions/search/cve/trivy/scanner.go index ca70c7423e..49080e8090 100644 --- a/pkg/extensions/search/cve/trivy/scanner.go +++ b/pkg/extensions/search/cve/trivy/scanner.go @@ -2,7 +2,6 @@ package trivy import ( "context" - "encoding/json" "fmt" "os" "path" @@ -26,7 +25,6 @@ import ( cvecache "zotregistry.io/zot/pkg/extensions/search/cve/cache" cvemodel "zotregistry.io/zot/pkg/extensions/search/cve/model" "zotregistry.io/zot/pkg/log" - mcommon "zotregistry.io/zot/pkg/meta/common" mTypes "zotregistry.io/zot/pkg/meta/types" "zotregistry.io/zot/pkg/storage" ) @@ -193,7 +191,7 @@ func (scanner Scanner) IsImageFormatScannable(repo, ref string) (bool, error) { ) if zcommon.IsTag(ref) { - imgDescriptor, err := mcommon.GetImageDescriptor(scanner.metaDB, repo, ref) + imgDescriptor, err := getImageDescriptor(scanner.metaDB, repo, ref) if err != nil { return false, err } @@ -203,7 +201,7 @@ func (scanner Scanner) IsImageFormatScannable(repo, ref string) (bool, error) { } else { var found bool - found, mediaType = mcommon.FindMediaTypeForDigest(scanner.metaDB, godigest.Digest(ref)) + found, mediaType = findMediaTypeForDigest(scanner.metaDB, godigest.Digest(ref)) if !found { return false, zerr.ErrManifestNotFound } @@ -224,7 +222,7 @@ func (scanner Scanner) IsImageMediaScannable(repo, digestStr, mediaType string) return ok, nil case ispec.MediaTypeImageIndex: - ok, err := scanner.isIndexScanable(digestStr) + ok, err := scanner.isIndexScannable(digestStr) if err != nil { return ok, fmt.Errorf("image '%s' %w", image, err) } @@ -240,21 +238,37 @@ func (scanner Scanner) isManifestScanable(digestStr string) (bool, error) { return true, nil } - manifestData, err := scanner.metaDB.GetManifestData(godigest.Digest(digestStr)) + manifestData, err := scanner.metaDB.GetImageData(godigest.Digest(digestStr)) if err != nil { return false, err } - var manifestContent ispec.Manifest + if manifestData.MediaType != ispec.MediaTypeImageManifest { + return false, zerr.ErrUnexpectedMediaType + } - err = json.Unmarshal(manifestData.ManifestBlob, &manifestContent) - if err != nil { - scanner.log.Error().Err(err).Msg("unable to unmashal manifest blob") + for _, imageLayer := range manifestData.Manifests[0].Manifest.Layers { + switch imageLayer.MediaType { + case ispec.MediaTypeImageLayerGzip, ispec.MediaTypeImageLayer, string(regTypes.DockerLayer): + continue + default: + return false, zerr.ErrScanNotSupported + } + } + return true, nil +} + +func (scanner Scanner) isManifestDataScannable(manifestData mTypes.ManifestData) (bool, error) { + if scanner.cache.Get(manifestData.Digest.String()) != nil { + return true, nil + } + + if manifestData.Manifest.MediaType != ispec.MediaTypeImageManifest { return false, zerr.ErrScanNotSupported } - for _, imageLayer := range manifestContent.Layers { + for _, imageLayer := range manifestData.Manifest.Layers { switch imageLayer.MediaType { case ispec.MediaTypeImageLayerGzip, ispec.MediaTypeImageLayer, string(regTypes.DockerLayer): continue @@ -266,34 +280,33 @@ func (scanner Scanner) isManifestScanable(digestStr string) (bool, error) { return true, nil } -func (scanner Scanner) isIndexScanable(digestStr string) (bool, error) { +func (scanner Scanner) isIndexScannable(digestStr string) (bool, error) { if scanner.cache.Get(digestStr) != nil { return true, nil } - indexData, err := scanner.metaDB.GetIndexData(godigest.Digest(digestStr)) + indexData, err := scanner.metaDB.GetImageData(godigest.Digest(digestStr)) if err != nil { return false, err } - var indexContent ispec.Index - - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return false, err + if indexData.MediaType != ispec.MediaTypeImageIndex || indexData.Index == nil { + return false, zerr.ErrUnexpectedMediaType } + indexContent := *indexData.Index + if len(indexContent.Manifests) == 0 { return true, nil } - for _, manifest := range indexContent.Manifests { - isScannable, err := scanner.isManifestScanable(manifest.Digest.String()) + for _, manifest := range indexData.Manifests { + isScannable, err := scanner.isManifestDataScannable(manifest) if err != nil { continue } - // if at least 1 manifest is scanable, the whole index is scanable + // if at least 1 manifest is scannable, the whole index is scannable if isScannable { return true, nil } @@ -323,7 +336,7 @@ func (scanner Scanner) ScanImage(image string) (map[string]cvemodel.CVE, error) digest = ref if isTag { - imgDescriptor, err := mcommon.GetImageDescriptor(scanner.metaDB, repo, ref) + imgDescriptor, err := getImageDescriptor(scanner.metaDB, repo, ref) if err != nil { return map[string]cvemodel.CVE{}, err } @@ -333,7 +346,7 @@ func (scanner Scanner) ScanImage(image string) (map[string]cvemodel.CVE, error) } else { var found bool - found, mediaType = mcommon.FindMediaTypeForDigest(scanner.metaDB, godigest.Digest(ref)) + found, mediaType = findMediaTypeForDigest(scanner.metaDB, godigest.Digest(ref)) if !found { return map[string]cvemodel.CVE{}, zerr.ErrManifestNotFound } @@ -441,21 +454,18 @@ func (scanner Scanner) scanIndex(repo, digest string) (map[string]cvemodel.CVE, return cachedMap, nil } - indexData, err := scanner.metaDB.GetIndexData(godigest.Digest(digest)) + indexData, err := scanner.metaDB.GetImageData(godigest.Digest(digest)) if err != nil { return map[string]cvemodel.CVE{}, err } - var indexContent ispec.Index - - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return map[string]cvemodel.CVE{}, err + if indexData.Index == nil { + return map[string]cvemodel.CVE{}, zerr.ErrUnexpectedMediaType } indexCveIDMap := map[string]cvemodel.CVE{} - for _, manifest := range indexContent.Manifests { + for _, manifest := range indexData.Index.Manifests { if isScannable, err := scanner.isManifestScanable(manifest.Digest.String()); isScannable && err == nil { manifestCveIDMap, err := scanner.scanManifest(repo, manifest.Digest.String()) if err != nil { @@ -567,6 +577,31 @@ func (scanner Scanner) checkDBPresence() error { return nil } +func getImageDescriptor(metaDB mTypes.MetaDB, repo, tag string) (mTypes.Descriptor, error) { + repoMeta, err := metaDB.GetRepoMeta(context.Background(), repo) + if err != nil { + return mTypes.Descriptor{}, err + } + + imageDescriptor, ok := repoMeta.Tags[tag] + if !ok { + return mTypes.Descriptor{}, zerr.ErrTagMetaNotFound + } + + return imageDescriptor, nil +} + +// findMediaTypeForDigest will look into the buckets for a certain digest. Depending on which bucket that +// digest is found the corresponding mediatype is returned. +func findMediaTypeForDigest(metaDB mTypes.MetaDB, digest godigest.Digest) (bool, string) { + imageData, err := metaDB.GetImageData(digest) + if err == nil { + return true, imageData.MediaType + } + + return false, "" +} + func convertSeverity(detectedSeverity string) string { trivySeverity, _ := dbTypes.NewSeverity(detectedSeverity) diff --git a/pkg/extensions/search/cve/trivy/scanner_internal_test.go b/pkg/extensions/search/cve/trivy/scanner_internal_test.go index 4d81ef9778..63dc232c30 100644 --- a/pkg/extensions/search/cve/trivy/scanner_internal_test.go +++ b/pkg/extensions/search/cve/trivy/scanner_internal_test.go @@ -22,7 +22,7 @@ import ( "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta" "zotregistry.io/zot/pkg/meta/boltdb" - mTypes "zotregistry.io/zot/pkg/meta/types" + "zotregistry.io/zot/pkg/meta/types" "zotregistry.io/zot/pkg/storage" "zotregistry.io/zot/pkg/storage/imagestore" "zotregistry.io/zot/pkg/storage/local" @@ -288,114 +288,31 @@ func TestImageScannable(t *testing.T) { // Create metadb data for scannable image timeStamp := time.Date(2008, 1, 1, 12, 0, 0, 0, time.UTC) - validConfigBlob, err := json.Marshal(ispec.Image{ + validConfig := ispec.Image{ Created: &timeStamp, - }) - if err != nil { - panic(err) - } - - validManifestBlob, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, - Size: 0, - Digest: godigest.FromBytes(validConfigBlob), - }, - Layers: []ispec.Descriptor{ - { - MediaType: ispec.MediaTypeImageLayerGzip, - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), - }, - }, - }) - if err != nil { - panic(err) - } - - validRepoMeta := mTypes.ManifestData{ - ManifestBlob: validManifestBlob, - ConfigBlob: validConfigBlob, } - digestValidManifest := godigest.FromBytes(validManifestBlob) - - err = metaDB.SetManifestData(digestValidManifest, validRepoMeta) - if err != nil { - panic(err) - } + validImage := CreateImageWith(). + Layers([]Layer{{ + MediaType: ispec.MediaTypeImageLayerGzip, + Digest: ispec.DescriptorEmptyJSON.Digest, + Blob: ispec.DescriptorEmptyJSON.Data, + }}).ImageConfig(validConfig).Build() - err = metaDB.SetRepoReference("repo1", "valid", digestValidManifest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo1", "valid", validImage.AsImageData()) if err != nil { panic(err) } // Create MetaDB data for manifest with unscannable layers - manifestBlobUnscannableLayer, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, - Size: 0, - Digest: godigest.FromBytes(validConfigBlob), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "unscannable_media_type", - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), - }, - }, - }) - if err != nil { - panic(err) - } - - repoMetaUnscannableLayer := mTypes.ManifestData{ - ManifestBlob: manifestBlobUnscannableLayer, - ConfigBlob: validConfigBlob, - } - - digestManifestUnscannableLayer := godigest.FromBytes(manifestBlobUnscannableLayer) - - err = metaDB.SetManifestData(digestManifestUnscannableLayer, repoMetaUnscannableLayer) - if err != nil { - panic(err) - } - - err = metaDB.SetRepoReference("repo1", "unscannable-layer", digestManifestUnscannableLayer, - ispec.MediaTypeImageManifest) - if err != nil { - panic(err) - } - - // Create MetaDB data for unmarshable manifest - unmarshableManifestBlob := []byte("Some string") - repoMetaUnmarshable := mTypes.ManifestData{ - ManifestBlob: unmarshableManifestBlob, - ConfigBlob: validConfigBlob, - } - - digestUnmarshableManifest := godigest.FromBytes(unmarshableManifestBlob) - - err = metaDB.SetManifestData(digestUnmarshableManifest, repoMetaUnmarshable) - if err != nil { - panic(err) - } - - err = metaDB.SetRepoReference("repo1", "unmarshable", digestUnmarshableManifest, ispec.MediaTypeImageManifest) - if err != nil { - panic(err) - } - - // Manifest meta cannot be found - digestMissingManifest := godigest.FromBytes([]byte("Some other string")) - - err = metaDB.SetRepoReference("repo1", "missing", digestMissingManifest, ispec.MediaTypeImageManifest) - if err != nil { - panic(err) - } - - // RepoMeta contains invalid digest - err = metaDB.SetRepoReference("repo1", "invalid-digest", "invalid", ispec.MediaTypeImageManifest) + imageWithUnscannableLayer := CreateImageWith(). + Layers([]Layer{{ + MediaType: "unscannable_media_type", + Digest: ispec.DescriptorEmptyJSON.Digest, + Blob: ispec.DescriptorEmptyJSON.Data, + }}).ImageConfig(validConfig).Build() + + err = metaDB.SetRepoReference("repo1", "unscannable-layer", imageWithUnscannableLayer.AsImageData()) if err != nil { panic(err) } @@ -423,18 +340,6 @@ func TestImageScannable(t *testing.T) { So(result, ShouldBeFalse) }) - Convey("Image with unmarshable manifests should be unscannable", t, func() { - result, err := scanner.IsImageFormatScannable("repo1", "unmarshable") - So(err, ShouldNotBeNil) - So(result, ShouldBeFalse) - }) - - Convey("Image with missing manifest meta should be unscannable", t, func() { - result, err := scanner.IsImageFormatScannable("repo1", "missing") - So(err, ShouldNotBeNil) - So(result, ShouldBeFalse) - }) - Convey("Image with invalid manifest digest should be unscannable", t, func() { result, err := scanner.IsImageFormatScannable("repo1", "invalid-digest") So(err, ShouldNotBeNil) @@ -523,14 +428,14 @@ func TestIsIndexScanable(t *testing.T) { scanner.cache.Add("digest", make(map[string]model.CVE)) - found, err := scanner.isIndexScanable("digest") + found, err := scanner.isIndexScannable("digest") So(err, ShouldBeNil) So(found, ShouldBeTrue) }) }) } -func TestScanIndexErrors(t *testing.T) { +func TestIsIndexScannableErrors(t *testing.T) { Convey("Errors", t, func() { storeController := storage.StoreController{} storeController.DefaultStore = mocks.MockedImageStore{} @@ -538,107 +443,22 @@ func TestScanIndexErrors(t *testing.T) { metaDB := mocks.MetaDBMock{} log := log.NewLogger("debug", "") - Convey("GetIndexData fails", func() { - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{}, godigest.ErrDigestUnsupported - } - - scanner := NewScanner(storeController, metaDB, "", "", log) - - _, err := scanner.scanIndex("repo", "digest") - So(err, ShouldNotBeNil) - }) - - Convey("Bad Index Blob, Unamrshal fails", func() { - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{ - IndexBlob: []byte(`bad-blob`), - }, nil + Convey("all manifests of a index are not scannable", func() { + unscannableLayer := []Layer{{MediaType: "unscannable-layer-type", Digest: godigest.FromString("123")}} + img1 := CreateImageWith().Layers(unscannableLayer).RandomConfig().Build() + img2 := CreateImageWith().Layers(unscannableLayer).RandomConfig().Build() + multiarch := CreateMultiarchWith().Images([]Image{img1, img2}).Build() + + metaDB.GetImageDataFn = func(digest godigest.Digest) (types.ImageData, error) { + return map[string]types.ImageData{ + img1.DigestStr(): img1.AsImageData(), + img2.DigestStr(): img2.AsImageData(), + multiarch.DigestStr(): multiarch.AsImageData(), + }[digest.String()], nil } scanner := NewScanner(storeController, metaDB, "", "", log) - - _, err := scanner.scanIndex("repo", "digest") - So(err, ShouldNotBeNil) - }) - }) -} - -func TestIsIndexScanableErrors(t *testing.T) { - Convey("Errors", t, func() { - storeController := storage.StoreController{} - storeController.DefaultStore = mocks.MockedImageStore{} - - metaDB := mocks.MetaDBMock{} - log := log.NewLogger("debug", "") - - Convey("GetIndexData errors", func() { - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{}, zerr.ErrManifestDataNotFound - } - scanner := NewScanner(storeController, metaDB, "", "", log) - - _, err := scanner.isIndexScanable("digest") - So(err, ShouldNotBeNil) - }) - - Convey("bad index data, can't unmarshal", func() { - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{IndexBlob: []byte(`bad`)}, nil - } - scanner := NewScanner(storeController, metaDB, "", "", log) - - ok, err := scanner.isIndexScanable("digest") - So(err, ShouldNotBeNil) - So(ok, ShouldBeFalse) - }) - - Convey("is Manifest Scanable errors", func() { - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{IndexBlob: []byte(`{ - "manifests": [{ - "digest": "digest2" - }, - { - "digest": "digest1" - } - ] - }`)}, nil - } - metaDB.GetManifestDataFn = func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - switch manifestDigest { - case "digest1": - return mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - }, nil - case "digest2": - return mTypes.ManifestData{}, zerr.ErrBadBlob - } - - return mTypes.ManifestData{}, nil - } - scanner := NewScanner(storeController, metaDB, "", "", log) - - ok, err := scanner.isIndexScanable("digest") - So(err, ShouldBeNil) - So(ok, ShouldBeTrue) - }) - - Convey("is Manifest Scanable returns false because no manifest is scanable", func() { - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{IndexBlob: []byte(`{ - "manifests": [{ - "digest": "digest2" - } - ] - }`)}, nil - } - metaDB.GetManifestDataFn = func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, zerr.ErrBadBlob - } - scanner := NewScanner(storeController, metaDB, "", "", log) - - ok, err := scanner.isIndexScanable("digest") + ok, err := scanner.isIndexScannable(multiarch.DigestStr()) So(err, ShouldBeNil) So(ok, ShouldBeFalse) }) diff --git a/pkg/extensions/search/cve/trivy/scanner_test.go b/pkg/extensions/search/cve/trivy/scanner_test.go index c03b821816..028549c4b9 100644 --- a/pkg/extensions/search/cve/trivy/scanner_test.go +++ b/pkg/extensions/search/cve/trivy/scanner_test.go @@ -11,7 +11,6 @@ import ( ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" - zerr "zotregistry.io/zot/errors" "zotregistry.io/zot/pkg/api" "zotregistry.io/zot/pkg/api/config" extconf "zotregistry.io/zot/pkg/extensions/config" @@ -20,13 +19,11 @@ import ( "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta" "zotregistry.io/zot/pkg/meta/boltdb" - mTypes "zotregistry.io/zot/pkg/meta/types" "zotregistry.io/zot/pkg/storage" "zotregistry.io/zot/pkg/storage/local" . "zotregistry.io/zot/pkg/test/common" "zotregistry.io/zot/pkg/test/deprecated" . "zotregistry.io/zot/pkg/test/image-utils" - "zotregistry.io/zot/pkg/test/mocks" ) func TestScanBigTestFile(t *testing.T) { @@ -130,31 +127,6 @@ func TestScanningByDigest(t *testing.T) { }) } -func TestScannerErrors(t *testing.T) { - digest := godigest.FromString("dig") - - Convey("Errors", t, func() { - storeController := storage.StoreController{} - storeController.DefaultStore = mocks.MockedImageStore{} - - metaDB := mocks.MetaDBMock{} - log := log.NewLogger("debug", "") - - Convey("IsImageFormatSanable", func() { - metaDB.GetManifestDataFn = func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, zerr.ErrManifestDataNotFound - } - metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{}, zerr.ErrManifestDataNotFound - } - scanner := trivy.NewScanner(storeController, metaDB, "", "", log) - - _, err := scanner.ScanImage("repo@" + digest.String()) - So(err, ShouldNotBeNil) - }) - }) -} - func TestVulnerableLayer(t *testing.T) { Convey("Vulnerable layer", t, func() { vulnerableLayer, err := GetLayerWithVulnerability() diff --git a/pkg/extensions/search/cve/update_test.go b/pkg/extensions/search/cve/update_test.go index e9c0849ea0..8cdf9554a1 100644 --- a/pkg/extensions/search/cve/update_test.go +++ b/pkg/extensions/search/cve/update_test.go @@ -40,8 +40,8 @@ func TestCVEDBGenerator(t *testing.T) { sch := scheduler.NewScheduler(cfg, logger) metaDB := &mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Tags: map[string]mTypes.Descriptor{ "tag": {MediaType: ispec.MediaTypeImageIndex}, }, diff --git a/pkg/extensions/search/resolver.go b/pkg/extensions/search/resolver.go index a322f7ff1c..9c31ece946 100644 --- a/pkg/extensions/search/resolver.go +++ b/pkg/extensions/search/resolver.go @@ -6,7 +6,6 @@ package search import ( "context" - "encoding/json" "errors" "fmt" "sort" @@ -74,45 +73,8 @@ func NewResolver(log log.Logger, storeController storage.StoreController, } func FilterByDigest(digest string) mTypes.FilterFunc { - return func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - lookupDigest := digest - contains := false - - var manifest ispec.Manifest - - err := json.Unmarshal(manifestMeta.ManifestBlob, &manifest) - if err != nil { - return false - } - - manifestDigest := godigest.FromBytes(manifestMeta.ManifestBlob).String() - - // Check the image manifest in index.json matches the search digest - // This is a blob with mediaType application/vnd.oci.image.manifest.v1+json - if strings.Contains(manifestDigest, lookupDigest) { - contains = true - } - - // Check the image config matches the search digest - // This is a blob with mediaType application/vnd.oci.image.config.v1+json - if strings.Contains(manifest.Config.Digest.String(), lookupDigest) { - contains = true - } - - // Check to see if the individual layers in the oci image manifest match the digest - // These are blobs with mediaType application/vnd.oci.image.layer.v1.tar+gzip - for _, layer := range manifest.Layers { - if strings.Contains(layer.Digest.String(), lookupDigest) { - contains = true - } - } - - return contains - } -} - -func ProtoFilterByDigest(digest string) mTypes.FilterProtoFunc { - return func(repoMeta mTypes.RepoMetadata2, imageData mTypes.ImageData2) bool { + // imageData will always contain 1 manifest + return func(repoMeta mTypes.RepoMeta, imageData mTypes.ImageData) bool { lookupDigest := digest contains := false @@ -128,13 +90,13 @@ func ProtoFilterByDigest(digest string) mTypes.FilterProtoFunc { // Check the image config matches the search digest // This is a blob with mediaType application/vnd.oci.image.config.v1+json - if strings.Contains(manifest.Config.Digest.String(), lookupDigest) { + if strings.Contains(manifest.Manifest.Config.Digest.String(), lookupDigest) { contains = true } // Check to see if the individual layers in the oci image manifest match the digest // These are blobs with mediaType application/vnd.oci.image.layer.v1.tar+gzip - for _, layer := range manifest.Layers { + for _, layer := range manifest.Manifest.Layers { if strings.Contains(layer.Digest.String(), lookupDigest) { contains = true } @@ -156,19 +118,19 @@ func getImageListForDigest(ctx context.Context, digest string, metaDB mTypes.Met } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), + deref(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), ), } - imageDataList, err := metaDB.ProtoFilterTags(ctx, mTypes.AcceptAllRepoTag, ProtoFilterByDigest(digest)) + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, FilterByDigest(digest)) if err != nil { return &gql_generated.PaginatedImagesResult{}, err } - imageSummaries, pageInfo, err := convert.PaginatedFullImageData2ImageSummaries(ctx, imageDataList, skip, + imageSummaries, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries(ctx, imageDataList, skip, cveInfo, mTypes.Filter{}, pageInput) if err != nil { return &gql_generated.PaginatedImagesResult{}, err @@ -188,132 +150,7 @@ func getImageSummary(ctx context.Context, repo, tag string, digest *string, skip ) ( *gql_generated.ImageSummary, error, ) { - repoMeta, err := metaDB.GetRepoMeta(repo) - if err != nil { - return nil, err - } - - manifestDescriptor, ok := repoMeta.Tags[tag] - if !ok { - return nil, gqlerror.Errorf("can't find image: %s:%s", repo, tag) - } - - for t := range repoMeta.Tags { - if t != tag { - delete(repoMeta.Tags, t) - } - } - - var ( - manifestMetaMap = map[string]mTypes.ManifestMetadata{} - indexDataMap = map[string]mTypes.IndexData{} - ) - - switch manifestDescriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := manifestDescriptor.Digest - - if digest != nil && *digest != manifestDigest { - return nil, fmt.Errorf("resolver: can't get ManifestData for digest %s for image '%s:%s' %w", - manifestDigest, repo, tag, zerr.ErrManifestDataNotFound) - } - - manifestData, err := metaDB.GetManifestData(godigest.Digest(manifestDigest)) - if err != nil { - return nil, err - } - - manifestMetaMap[manifestDigest] = mTypes.ManifestMetadata{ - ManifestBlob: manifestData.ManifestBlob, - ConfigBlob: manifestData.ConfigBlob, - } - case ispec.MediaTypeImageIndex: - indexDigest := manifestDescriptor.Digest - - indexData, err := metaDB.GetIndexData(godigest.Digest(indexDigest)) - if err != nil { - return nil, err - } - - var indexContent ispec.Index - - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return nil, err - } - - if digest != nil { - manifestDigest := *digest - - digestFound := false - - for _, manifest := range indexContent.Manifests { - if manifest.Digest.String() == manifestDigest { - digestFound = true - - break - } - } - - if !digestFound { - return nil, fmt.Errorf("resolver: can't get ManifestData for digest %s for image '%s:%s' %w", - manifestDigest, repo, tag, zerr.ErrManifestDataNotFound) - } - - manifestData, err := metaDB.GetManifestData(godigest.Digest(manifestDigest)) - if err != nil { - return nil, fmt.Errorf("resolver: can't get ManifestData for digest %s for image '%s:%s' %w", - manifestDigest, repo, tag, err) - } - - manifestMetaMap[manifestDigest] = mTypes.ManifestMetadata{ - ManifestBlob: manifestData.ManifestBlob, - ConfigBlob: manifestData.ConfigBlob, - } - - // We update the tag descriptor to be the manifest descriptor with digest specified in the - // 'digest' parameter. We treat it as a standalone image. - repoMeta.Tags[tag] = mTypes.Descriptor{ - Digest: manifestDigest, - MediaType: ispec.MediaTypeImageManifest, - } - - break - } - - for _, manifest := range indexContent.Manifests { - manifestData, err := metaDB.GetManifestData(manifest.Digest) - if err != nil { - return nil, fmt.Errorf("resolver: can't get ManifestData for digest %s for image '%s:%s' %w", - manifest.Digest, repo, tag, err) - } - - manifestMetaMap[manifest.Digest.String()] = mTypes.ManifestMetadata{ - ManifestBlob: manifestData.ManifestBlob, - ConfigBlob: manifestData.ConfigBlob, - } - } - - indexDataMap[indexDigest] = indexData - default: - log.Error().Str("mediaType", manifestDescriptor.MediaType).Msg("resolver: media type not supported") - } - - imageSummaries := convert.RepoMeta2ImageSummaries(ctx, repoMeta, manifestMetaMap, indexDataMap, skipCVE, cveInfo) - - if len(imageSummaries) == 0 { - return &gql_generated.ImageSummary{}, nil - } - - return imageSummaries[0], nil -} - -func getProtoImageSummary(ctx context.Context, repo, tag string, digest *string, skipCVE convert.SkipQGLField, - metaDB mTypes.MetaDB, cveInfo cveinfo.CveInfo, log log.Logger, //nolint:unparam -) ( - *gql_generated.ImageSummary, error, -) { - repoMeta, err := metaDB.ProtoGetRepoMeta(repo) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo) if err != nil { return nil, err } @@ -328,14 +165,18 @@ func getProtoImageSummary(ctx context.Context, repo, tag string, digest *string, imageDigest := manifestDescriptor.Digest if digest != nil { imageDigest = *digest + repoMeta.Tags[tag] = mTypes.Descriptor{ + Digest: imageDigest, + MediaType: ispec.MediaTypeImageManifest, + } } - imageDataMap, err := metaDB.ProtoFilterImageData(ctx, []string{imageDigest}) + imageDataMap, err := metaDB.FilterImageData(ctx, []string{imageDigest}) if err != nil { return &gql_generated.ImageSummary{}, err } - imageSummaries := convert.ProtoRepoMeta2ImageSummaries(ctx, repoMeta, imageDataMap, skipCVE, cveInfo) + imageSummaries := convert.RepoMeta2ImageSummaries(ctx, repoMeta, imageDataMap, skipCVE, cveInfo) if len(imageSummaries) == 0 { return &gql_generated.ImageSummary{}, nil @@ -357,10 +198,10 @@ func getCVEListForImage( } pageInput := cvemodel.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: cvemodel.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaSeverity), + deref(requestedPage.SortBy, gql_generated.SortCriteriaSeverity), ), } @@ -419,30 +260,7 @@ func getCVEListForImage( } func FilterByTagInfo(tagsInfo []cvemodel.TagInfo) mTypes.FilterFunc { - return func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - manifestDigest := godigest.FromBytes(manifestMeta.ManifestBlob).String() - - for _, tagInfo := range tagsInfo { - switch tagInfo.Descriptor.MediaType { - case ispec.MediaTypeImageManifest: - if tagInfo.Descriptor.Digest.String() == manifestDigest { - return true - } - case ispec.MediaTypeImageIndex: - for _, manifestDesc := range tagInfo.Manifests { - if manifestDesc.Digest.String() == manifestDigest { - return true - } - } - } - } - - return false - } -} - -func ProtoFilterByTagInfo(tagsInfo []cvemodel.TagInfo) mTypes.FilterProtoFunc { - return func(repoMeta mTypes.RepoMetadata2, imageData mTypes.ImageData2) bool { + return func(repoMeta mTypes.RepoMeta, imageData mTypes.ImageData) bool { manifestDigest := imageData.Manifests[0].Digest.String() for _, tagInfo := range tagsInfo { @@ -465,34 +283,7 @@ func ProtoFilterByTagInfo(tagsInfo []cvemodel.TagInfo) mTypes.FilterProtoFunc { } func FilterByRepoAndTagInfo(repo string, tagsInfo []cvemodel.TagInfo) mTypes.FilterFunc { - return func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - if repoMeta.Name != repo { - return false - } - - manifestDigest := godigest.FromBytes(manifestMeta.ManifestBlob).String() - - for _, tagInfo := range tagsInfo { - switch tagInfo.Descriptor.MediaType { - case ispec.MediaTypeImageManifest: - if tagInfo.Descriptor.Digest.String() == manifestDigest { - return true - } - case ispec.MediaTypeImageIndex: - for _, manifestDesc := range tagInfo.Manifests { - if manifestDesc.Digest.String() == manifestDigest { - return true - } - } - } - } - - return false - } -} - -func ProtoFilterByRepoAndTagInfo(repo string, tagsInfo []cvemodel.TagInfo) mTypes.FilterProtoFunc { - return func(repoMeta mTypes.RepoMetadata2, imageData mTypes.ImageData2) bool { + return func(repoMeta mTypes.RepoMeta, imageData mTypes.ImageData) bool { if repoMeta.Name != repo { return false } @@ -531,16 +322,11 @@ func getImageListForCVE( // Infinite page to make sure we scan all repos in advance, before filtering results // The CVE scan logic is called from here, not in the actual filter, // this is because we shouldn't keep the DB locked while we wait on scan results - reposMeta, err := metaDB.ProtoGetMultipleRepoMeta(ctx, func(repoMeta mTypes.RepoMetadata2) bool { return true }) + reposMeta, err := metaDB.GetMultipleRepoMeta(ctx, func(repoMeta mTypes.RepoMeta) bool { return true }) if err != nil { return &gql_generated.PaginatedImagesResult{}, err } - // reposMeta, err := metaDB.GetMultipleRepoMeta(ctx, func(repoMeta mTypes.RepoMetadata) bool { return true }) - // if err != nil { - // return &gql_generated.PaginatedImagesResult{}, err - // } - affectedImages := []cvemodel.TagInfo{} for _, repoMeta := range reposMeta { @@ -579,20 +365,20 @@ func getImageListForCVE( // Actual page requested by user pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), + deref(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), ), } // get all repos - imageDataList, err := metaDB.ProtoFilterTags(ctx, mTypes.AcceptAllRepoTag, ProtoFilterByTagInfo(affectedImages)) + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, FilterByTagInfo(affectedImages)) if err != nil { return &gql_generated.PaginatedImagesResult{}, err } - imageSummaries, pageInfo, err := convert.PaginatedFullImageData2ImageSummaries(ctx, imageDataList, + imageSummaries, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries(ctx, imageDataList, skip, cveInfo, localFilter, pageInput) if err != nil { return &gql_generated.PaginatedImagesResult{}, err @@ -652,20 +438,20 @@ func getImageListWithCVEFixed( // Actual page requested by user pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), + deref(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), ), } // get all repos - imageDataList, err := metaDB.ProtoFilterTags(ctx, mTypes.AcceptAllRepoTag, ProtoFilterByRepoAndTagInfo(repo, tagsInfo)) + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, FilterByRepoAndTagInfo(repo, tagsInfo)) if err != nil { return &gql_generated.PaginatedImagesResult{}, err } - imageSummaries, pageInfo, err := convert.PaginatedFullImageData2ImageSummaries(ctx, imageDataList, + imageSummaries, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries(ctx, imageDataList, skip, cveInfo, localFilter, pageInput) if err != nil { return &gql_generated.PaginatedImagesResult{}, err @@ -698,24 +484,24 @@ func repoListWithNewestImage( } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), + deref(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), ), } - reposMeta, err := metaDB.ProtoSearchRepos(ctx, "") + repoMetaList, err := metaDB.SearchRepos(ctx, "") if err != nil { return &gql_generated.PaginatedReposResult{}, err } - imageDataMap, err := metaDB.ProtoFilterImageData(ctx, []string{}) + imageDataMap, err := metaDB.FilterImageData(ctx, mTypes.GetLatestImageDigests(repoMetaList)) if err != nil { return &gql_generated.PaginatedReposResult{}, err } - repos, pageInfo, err := convert.PaginatedFullRepoMeta2RepoSummaries(ctx, reposMeta, imageDataMap, + repos, pageInfo, err := convert.PaginatedRepoMeta2RepoSummaries(ctx, repoMetaList, imageDataMap, mTypes.Filter{}, pageInput, cveInfo, skip) if err != nil { return &gql_generated.PaginatedReposResult{}, err @@ -743,19 +529,11 @@ func getBookmarkedRepos( return &gql_generated.PaginatedReposResult{}, err } - // filterFn := func(repoMeta mTypes.RepoMetadata) bool { - // return zcommon.Contains(repoNames, repoMeta.Name) - // } - - filterByName := func(repo string) int { - if zcommon.Contains(bookmarkedRepos, repo) { - return 1 - } - - return -1 + filterByName := func(repo string) bool { + return zcommon.Contains(bookmarkedRepos, repo) } - return protoGetFilteredPaginatedRepos(ctx, cveInfo, filterByName, log, requestedPage, metaDB) + return getFilteredPaginatedRepos(ctx, cveInfo, filterByName, log, requestedPage, metaDB) } func getStarredRepos( @@ -770,62 +548,14 @@ func getStarredRepos( return &gql_generated.PaginatedReposResult{}, err } - protoFilterFn := func(repo string) int { - if zcommon.Contains(starredRepos, repo) { - return 1 - } - - return -1 + filterFn := func(repo string) bool { + return zcommon.Contains(starredRepos, repo) } - return protoGetFilteredPaginatedRepos(ctx, cveInfo, protoFilterFn, log, requestedPage, metaDB) + return getFilteredPaginatedRepos(ctx, cveInfo, filterFn, log, requestedPage, metaDB) } func getFilteredPaginatedRepos( - ctx context.Context, - cveInfo cveinfo.CveInfo, - filterFn mTypes.FilterRepoFunc, - log log.Logger, //nolint:unparam // may be used by devs for debugging - requestedPage *gql_generated.PageInput, - metaDB mTypes.MetaDB, -) (*gql_generated.PaginatedReposResult, error) { - if requestedPage == nil { - requestedPage = &gql_generated.PageInput{} - } - - skip := convert.SkipQGLField{ - Vulnerabilities: canSkipField(convert.GetPreloads(ctx), "Results.NewestImage.Vulnerabilities"), - } - - pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), - SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), - ), - } - - reposMeta, manifestMetaMap, indexDataMap, err := metaDB.FilterRepos(ctx, filterFn) - if err != nil { - return &gql_generated.PaginatedReposResult{}, err - } - - repos, pageInfo, err := convert.PaginatedRepoMeta2RepoSummaries(ctx, reposMeta, manifestMetaMap, indexDataMap, - skip, cveInfo, mTypes.Filter{}, pageInput) - if err != nil { - return &gql_generated.PaginatedReposResult{}, err - } - - return &gql_generated.PaginatedReposResult{ - Results: repos, - Page: &gql_generated.PageInfo{ - TotalCount: pageInfo.TotalCount, - ItemCount: pageInfo.ItemCount, - }, - }, nil -} - -func protoGetFilteredPaginatedRepos( ctx context.Context, cveInfo cveinfo.CveInfo, filterFn mTypes.FilterRepoNameFunc, @@ -842,24 +572,24 @@ func protoGetFilteredPaginatedRepos( } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), + deref(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), ), } - repoMetaList, err := metaDB.ProtoFilterRepos(ctx, filterFn, mTypes.AcceptAllRepoMeta) + repoMetaList, err := metaDB.FilterRepos(ctx, filterFn, mTypes.AcceptAllRepoMeta) if err != nil { return &gql_generated.PaginatedReposResult{}, err } - latestImageData, err := metaDB.ProtoFilterImageData(ctx, []string{}) + latestImageData, err := metaDB.FilterImageData(ctx, mTypes.GetLatestImageDigests(repoMetaList)) if err != nil { return &gql_generated.PaginatedReposResult{}, err } - repos, pageInfo, err := convert.PaginatedFullRepoMeta2RepoSummaries(ctx, repoMetaList, latestImageData, + repos, pageInfo, err := convert.PaginatedRepoMeta2RepoSummaries(ctx, repoMetaList, latestImageData, mTypes.Filter{}, pageInput, cveInfo, skip) if err != nil { return &gql_generated.PaginatedReposResult{}, err @@ -904,27 +634,31 @@ func globalSearch(ctx context.Context, query string, metaDB mTypes.MetaDB, filte } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), + deref(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), ), } - repoMetaList, err := metaDB.ProtoSearchRepos(ctx, query) + repoMetaList, err := metaDB.SearchRepos(ctx, query) if err != nil { - return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, []*gql_generated.LayerSummary{}, err + return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, + []*gql_generated.LayerSummary{}, err } - imageDataMap, err := metaDB.ProtoFilterImageData(ctx, getLatestImageDigest(repoMetaList)) + imageDataMap, err := metaDB.FilterImageData(ctx, mTypes.GetLatestImageDigests(repoMetaList)) if err != nil { - return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, []*gql_generated.LayerSummary{}, err + return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, + []*gql_generated.LayerSummary{}, err } - repos, pageInfo, err := convert.PaginatedFullRepoMeta2RepoSummaries(ctx, repoMetaList, imageDataMap, localFilter, pageInput, cveInfo, + repos, pageInfo, err := convert.PaginatedRepoMeta2RepoSummaries(ctx, repoMetaList, imageDataMap, localFilter, + pageInput, cveInfo, skip) if err != nil { - return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, []*gql_generated.LayerSummary{}, err + return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, + []*gql_generated.LayerSummary{}, err } paginatedRepos.Page = &gql_generated.PageInfo{ @@ -939,19 +673,19 @@ func globalSearch(ctx context.Context, query string, metaDB mTypes.MetaDB, filte } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), + deref(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), ), } - imageDataList, err := metaDB.ProtoSearchTags(ctx, query) + imageDataList, err := metaDB.SearchTags(ctx, query) if err != nil { return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, []*gql_generated.LayerSummary{}, err } - imageSummaries, pageInfo, err := convert.PaginatedFullImageData2ImageSummaries(ctx, imageDataList, skip, cveInfo, + imageSummaries, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries(ctx, imageDataList, skip, cveInfo, localFilter, pageInput) if err != nil { return &gql_generated.PaginatedReposResult{}, []*gql_generated.ImageSummary{}, []*gql_generated.LayerSummary{}, err @@ -968,16 +702,6 @@ func globalSearch(ctx context.Context, query string, metaDB mTypes.MetaDB, filte return &paginatedRepos, images, layers, nil } -func getLatestImageDigest(repoMetaList []mTypes.FullRepoMetadata) []string { - digests := make([]string, 0, len(repoMetaList)) - - for i := range repoMetaList { - digests = append(digests, repoMetaList[i].LastUpdatedImage.Digest) - } - - return digests -} - func canSkipField(preloads map[string]bool, s string) bool { fieldIsPresent := preloads[s] @@ -993,10 +717,10 @@ func derivedImageList(ctx context.Context, image string, digest *string, metaDB } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), + deref(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), ), } @@ -1013,7 +737,7 @@ func derivedImageList(ctx context.Context, image string, digest *string, metaDB Vulnerabilities: true, } - searchedImage, err := getProtoImageSummary(ctx, imageRepo, imageTag, digest, skipReferenceImage, metaDB, cveInfo, log) + searchedImage, err := getImageSummary(ctx, imageRepo, imageTag, digest, skipReferenceImage, metaDB, cveInfo, log) if err != nil { if errors.Is(err, zerr.ErrRepoMetaNotFound) { return &gql_generated.PaginatedImagesResult{}, gqlerror.Errorf("repository: not found") @@ -1023,12 +747,12 @@ func derivedImageList(ctx context.Context, image string, digest *string, metaDB } // we need all available tags - imageDataList, err := metaDB.ProtoFilterTags(ctx, mTypes.AcceptAllRepoTag, protoFilterDerivedImages(searchedImage)) + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, filterDerivedImages(searchedImage)) if err != nil { return &gql_generated.PaginatedImagesResult{}, err } - derivedList, pageInfo, err := convert.PaginatedFullImageData2ImageSummaries(ctx, imageDataList, skip, cveInfo, + derivedList, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries(ctx, imageDataList, skip, cveInfo, mTypes.Filter{}, pageInput) if err != nil { return &gql_generated.PaginatedImagesResult{}, err @@ -1044,53 +768,7 @@ func derivedImageList(ctx context.Context, image string, digest *string, metaDB } func filterDerivedImages(image *gql_generated.ImageSummary) mTypes.FilterFunc { - return func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - var addImageToList bool - - var imageManifest ispec.Manifest - - err := json.Unmarshal(manifestMeta.ManifestBlob, &imageManifest) - if err != nil { - return false - } - - for i := range image.Manifests { - manifestDigest := godigest.FromBytes(manifestMeta.ManifestBlob).String() - if manifestDigest == *image.Manifests[i].Digest { - return false - } - imageLayers := image.Manifests[i].Layers - - addImageToList = false - layers := imageManifest.Layers - - sameLayer := 0 - - for _, l := range imageLayers { - for _, k := range layers { - if k.Digest.String() == *l.Digest { - sameLayer++ - } - } - } - - // if all layers are the same - if sameLayer == len(imageLayers) { - // it's a derived image - addImageToList = true - } - - if addImageToList { - return true - } - } - - return false - } -} - -func protoFilterDerivedImages(image *gql_generated.ImageSummary) mTypes.FilterProtoFunc { - return func(repoMeta mTypes.RepoMetadata2, imageData mTypes.ImageData2) bool { + return func(repoMeta mTypes.RepoMeta, imageData mTypes.ImageData) bool { var addImageToList bool imageManifest := imageData.Manifests[0] @@ -1103,7 +781,7 @@ func protoFilterDerivedImages(image *gql_generated.ImageSummary) mTypes.FilterPr imageLayers := image.Manifests[i].Layers addImageToList = false - layers := imageManifest.Layers + layers := imageManifest.Manifest.Layers sameLayer := 0 @@ -1139,10 +817,10 @@ func baseImageList(ctx context.Context, image string, digest *string, metaDB mTy } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), + deref(requestedPage.SortBy, gql_generated.SortCriteriaUpdateTime), ), } @@ -1160,7 +838,7 @@ func baseImageList(ctx context.Context, image string, digest *string, metaDB mTy Vulnerabilities: true, } - searchedImage, err := getProtoImageSummary(ctx, imageRepo, imageTag, digest, skipReferenceImage, metaDB, cveInfo, log) + searchedImage, err := getImageSummary(ctx, imageRepo, imageTag, digest, skipReferenceImage, metaDB, cveInfo, log) if err != nil { if errors.Is(err, zerr.ErrRepoMetaNotFound) { return &gql_generated.PaginatedImagesResult{}, gqlerror.Errorf("repository: not found") @@ -1170,12 +848,12 @@ func baseImageList(ctx context.Context, image string, digest *string, metaDB mTy } // we need all available tags - imageDataList, err := metaDB.ProtoFilterTags(ctx, mTypes.AcceptAllRepoTag, protoFilterBaseImages(searchedImage)) + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, filterBaseImages(searchedImage)) if err != nil { return &gql_generated.PaginatedImagesResult{}, err } - baseList, pageInfo, err := convert.PaginatedFullImageData2ImageSummaries(ctx, imageDataList, + baseList, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries(ctx, imageDataList, skip, cveInfo, mTypes.Filter{}, pageInput) if err != nil { return &gql_generated.PaginatedImagesResult{}, err @@ -1191,53 +869,7 @@ func baseImageList(ctx context.Context, image string, digest *string, metaDB mTy } func filterBaseImages(image *gql_generated.ImageSummary) mTypes.FilterFunc { - return func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - var addImageToList bool - - var manifestContent ispec.Manifest - - err := json.Unmarshal(manifestMeta.ManifestBlob, &manifestContent) - if err != nil { - return false - } - - for i := range image.Manifests { - manifestDigest := godigest.FromBytes(manifestMeta.ManifestBlob).String() - if manifestDigest == *image.Manifests[i].Digest { - return false - } - - addImageToList = true - - for _, l := range manifestContent.Layers { - foundLayer := false - - for _, k := range image.Manifests[i].Layers { - if l.Digest.String() == *k.Digest { - foundLayer = true - - break - } - } - - if !foundLayer { - addImageToList = false - - break - } - } - - if addImageToList { - return true - } - } - - return false - } -} - -func protoFilterBaseImages(image *gql_generated.ImageSummary) mTypes.FilterProtoFunc { - return func(repoMeta mTypes.RepoMetadata2, imageData mTypes.ImageData2) bool { + return func(repoMeta mTypes.RepoMeta, imageData mTypes.ImageData) bool { var addImageToList bool manifest := imageData.Manifests[0] @@ -1250,7 +882,7 @@ func protoFilterBaseImages(image *gql_generated.ImageSummary) mTypes.FilterProto addImageToList = true - for _, l := range manifest.Layers { + for _, l := range manifest.Manifest.Layers { foundLayer := false for _, k := range image.Manifests[i].Layers { @@ -1281,7 +913,7 @@ func validateGlobalSearchInput(query string, filter *gql_generated.Filter, requestedPage *gql_generated.PageInput, ) error { if len(query) > querySizeLimit { - return fmt.Errorf("global-search: max string size limit exeeded for query parameter. max=%d current=%d %w", + return fmt.Errorf("global-search: max string size limit exceeded for query parameter. max=%d current=%d %w", querySizeLimit, len(query), zerr.ErrInvalidRequestParams) } @@ -1305,14 +937,14 @@ func checkFilter(filter *gql_generated.Filter) error { for _, arch := range filter.Arch { if len(*arch) > querySizeLimit { - return fmt.Errorf("global-search: max string size limit exeeded for arch parameter. max=%d current=%d %w", + return fmt.Errorf("global-search: max string size limit exceeded for arch parameter. max=%d current=%d %w", querySizeLimit, len(*arch), zerr.ErrInvalidRequestParams) } } for _, osSys := range filter.Os { if len(*osSys) > querySizeLimit { - return fmt.Errorf("global-search: max string size limit exeeded for os parameter. max=%d current=%d %w", + return fmt.Errorf("global-search: max string size limit exceeded for os parameter. max=%d current=%d %w", querySizeLimit, len(*osSys), zerr.ErrInvalidRequestParams) } } @@ -1404,7 +1036,7 @@ func expandedRepoInfo(ctx context.Context, repo string, metaDB mTypes.MetaDB, cv return &gql_generated.RepoInfo{}, nil //nolint:nilerr // don't give details to a potential attacker } - repoMeta, err := metaDB.ProtoGetFullRepoMeta(ctx, repo) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo) if err != nil { log.Error().Err(err).Str("repository", repo).Msg("resolver: can't retrieve repoMeta for repo") @@ -1412,6 +1044,7 @@ func expandedRepoInfo(ctx context.Context, repo string, metaDB mTypes.MetaDB, cv } tagsDigests := []string{} + for i := range repoMeta.Tags { if i == "" { continue @@ -1420,7 +1053,7 @@ func expandedRepoInfo(ctx context.Context, repo string, metaDB mTypes.MetaDB, cv tagsDigests = append(tagsDigests, repoMeta.Tags[i].Digest) } - imageDataMap, err := metaDB.ProtoFilterImageData(ctx, tagsDigests) + imageDataMap, err := metaDB.FilterImageData(ctx, tagsDigests) if err != nil { log.Error().Err(err).Str("repository", repo).Msg("resolver: can't retrieve imageData for repo") @@ -1432,7 +1065,7 @@ func expandedRepoInfo(ctx context.Context, repo string, metaDB mTypes.MetaDB, cv canSkipField(convert.GetPreloads(ctx), "Images.Vulnerabilities"), } - repoSummary, imageSummaries := convert.FullRepoMeta2ExpandedRepoInfo(ctx, repoMeta, imageDataMap, + repoSummary, imageSummaries := convert.RepoMeta2ExpandedRepoInfo(ctx, repoMeta, imageDataMap, skip, cveInfo, log) dateSortedImages := make(timeSlice, 0, len(imageSummaries)) @@ -1459,7 +1092,7 @@ func (p timeSlice) Swap(i, j int) { p[i], p[j] = p[j], p[i] } -func safeDereferencing[T any](pointer *T, defaultVal T) T { +func deref[T any](pointer *T, defaultVal T) T { if pointer != nil { return *pointer } @@ -1483,21 +1116,27 @@ func getImageList(ctx context.Context, repo string, metaDB mTypes.MetaDB, cveInf } pageInput := pagination.PageInput{ - Limit: safeDereferencing(requestedPage.Limit, 0), - Offset: safeDereferencing(requestedPage.Offset, 0), + Limit: deref(requestedPage.Limit, 0), + Offset: deref(requestedPage.Offset, 0), SortBy: pagination.SortCriteria( - safeDereferencing(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), + deref(requestedPage.SortBy, gql_generated.SortCriteriaRelevance), ), } - matchRepoName := func(repoName, tag string) bool { return repoName == repo } + var matchRepoName mTypes.FilterRepoTagFunc + + if repo == "" { + matchRepoName = mTypes.AcceptAllRepoTag + } else { + matchRepoName = func(repoName, tag string) bool { return repoName == repo } + } - imageData, err := metaDB.ProtoFilterTags(ctx, matchRepoName, mTypes.AcceptAllImageData) + imageData, err := metaDB.FilterTags(ctx, matchRepoName, mTypes.AcceptAllImageData) if err != nil { return &gql_generated.PaginatedImagesResult{}, err } - imageList, pageInfo, err := convert.PaginatedFullImageData2ImageSummaries(ctx, imageData, skip, + imageList, pageInfo, err := convert.PaginatedImageMeta2ImageSummaries(ctx, imageData, skip, cveInfo, mTypes.Filter{}, pageInput) if err != nil { return &gql_generated.PaginatedImagesResult{}, err @@ -1523,7 +1162,7 @@ func getReferrers(metaDB mTypes.MetaDB, repo string, referredDigest string, arti referredDigest, err) } - referrers, err := metaDB.ProtoGetReferrersInfo(repo, refDigest, artifactTypes) + referrers, err := metaDB.GetReferrersInfo(repo, refDigest, artifactTypes) if err != nil { return nil, err } diff --git a/pkg/extensions/search/resolver_test.go b/pkg/extensions/search/resolver_test.go index d5efa57509..ec33ddc0af 100644 --- a/pkg/extensions/search/resolver_test.go +++ b/pkg/extensions/search/resolver_test.go @@ -4,16 +4,13 @@ package search //nolint import ( "context" - "encoding/json" "errors" "fmt" - "strings" "testing" "time" "github.com/99designs/gqlgen/graphql" godigest "github.com/opencontainers/go-digest" - "github.com/opencontainers/image-spec/specs-go" ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" @@ -24,10 +21,13 @@ import ( "zotregistry.io/zot/pkg/extensions/search/gql_generated" "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta/boltdb" + mConvert "zotregistry.io/zot/pkg/meta/convert" mTypes "zotregistry.io/zot/pkg/meta/types" reqCtx "zotregistry.io/zot/pkg/requestcontext" "zotregistry.io/zot/pkg/storage" + . "zotregistry.io/zot/pkg/test/image-utils" "zotregistry.io/zot/pkg/test/mocks" + ociutils "zotregistry.io/zot/pkg/test/oci-utils" ) var ErrTestError = errors.New("TestError") @@ -38,11 +38,8 @@ func TestResolverGlobalSearch(t *testing.T) { Convey("MetaDB SearchRepos error", func() { mockMetaDB := mocks.MetaDBMock{ SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return make([]mTypes.RepoMetadata, 0), make(map[string]mTypes.ManifestMetadata), - map[string]mTypes.IndexData{}, ErrTestError + ) ([]mTypes.RepoMeta, error) { + return []mTypes.RepoMeta{}, ErrTestError }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, @@ -73,249 +70,10 @@ func TestResolverGlobalSearch(t *testing.T) { So(err, ShouldNotBeNil) }) - Convey("MetaDB SearchRepo is successful", func() { - mockMetaDB := mocks.MetaDBMock{ - SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": { - Digest: "digestTag1.0.1", - MediaType: ispec.MediaTypeImageManifest, - }, - "1.0.2": { - Digest: "digestTag1.0.2", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, - }, - }, - }, - Stars: 100, - }, - } - - createTime := time.Now() - configBlob1, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{ - ispec.AnnotationVendor: "TestVendor1", - }, - }, - Created: &createTime, - }) - So(err, ShouldBeNil) - - configBlob2, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{ - ispec.AnnotationVendor: "TestVendor2", - }, - }, - }) - So(err, ShouldBeNil) - - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob1, - }, - "digestTag1.0.2": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob2, - }, - } - - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil - }, - } - - const query = "repo1" - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - mockCve := mocks.CveInfoMock{} - repos, images, layers, err := globalSearch(responseContext, query, mockMetaDB, - &gql_generated.Filter{}, &pageInput, mockCve, log.NewLogger("debug", "")) - So(err, ShouldBeNil) - So(images, ShouldBeEmpty) - So(layers, ShouldBeEmpty) - So(repos.Results, ShouldNotBeEmpty) - So(len(repos.Results[0].Vendors), ShouldEqual, 2) - }) - - Convey("MetaDB SearchRepo Bad manifest referenced", func() { - mockMetaDB := mocks.MetaDBMock{ - SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": { - Digest: "digestTag1.0.1", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, - }, - }, - }, - Stars: 100, - }, - } - - configBlob, err := json.Marshal(ispec.Image{}) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: []byte("bad manifest blob"), - ConfigBlob: configBlob, - }, - } - - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil - }, - } - - query := "repo1" - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - mockCve := mocks.CveInfoMock{} - - repos, images, layers, err := globalSearch(responseContext, query, mockMetaDB, - &gql_generated.Filter{}, &pageInput, mockCve, log.NewLogger("debug", "")) - So(err, ShouldBeNil) - So(images, ShouldBeEmpty) - So(layers, ShouldBeEmpty) - So(repos, ShouldNotBeEmpty) - - query = "repo1:1.0.1" - - responseContext = graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - repos, images, layers, err = globalSearch(responseContext, query, mockMetaDB, - &gql_generated.Filter{}, &pageInput, mockCve, log.NewLogger("debug", "")) - So(err, ShouldBeNil) - So(images, ShouldBeEmpty) - So(layers, ShouldBeEmpty) - So(repos.Results, ShouldBeEmpty) - }) - - Convey("MetaDB SearchRepo good manifest referenced and bad config blob", func() { - mockMetaDB := mocks.MetaDBMock{ - SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": { - Digest: "digestTag1.0.1", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, - }, - }, - }, - Stars: 100, - }, - } - - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: []byte("bad config blob"), - }, - } - - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil - }, - } - - query := "repo1" - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - - mockCve := mocks.CveInfoMock{} - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - repos, images, layers, err := globalSearch(responseContext, query, mockMetaDB, - &gql_generated.Filter{}, &pageInput, mockCve, log.NewLogger("debug", "")) - So(err, ShouldBeNil) - So(images, ShouldBeEmpty) - So(layers, ShouldBeEmpty) - So(repos.Results, ShouldNotBeEmpty) - - query = "repo1:1.0.1" - responseContext = graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - repos, images, layers, err = globalSearch(responseContext, query, mockMetaDB, - &gql_generated.Filter{}, &pageInput, mockCve, log.NewLogger("debug", "")) - So(err, ShouldBeNil) - So(images, ShouldBeEmpty) - So(layers, ShouldBeEmpty) - So(repos.Results, ShouldBeEmpty) - }) - Convey("MetaDB SearchTags gives error", func() { mockMetaDB := mocks.MetaDBMock{ - SearchTagsFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return make([]mTypes.RepoMetadata, 0), make(map[string]mTypes.ManifestMetadata), - map[string]mTypes.IndexData{}, ErrTestError + SearchTagsFn: func(ctx context.Context, searchText string) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, } const query = "repo1:1.0.1" @@ -330,90 +88,6 @@ func TestResolverGlobalSearch(t *testing.T) { So(layers, ShouldBeEmpty) So(repos.Results, ShouldBeEmpty) }) - - Convey("MetaDB SearchTags is successful", func() { - mockMetaDB := mocks.MetaDBMock{ - SearchTagsFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": { - Digest: "digestTag1.0.1", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, - }, - }, - }, - Stars: 100, - }, - } - - configBlob1, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{ - ispec.AnnotationVendor: "TestVendor1", - }, - }, - }) - So(err, ShouldBeNil) - - configBlob2, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{ - ispec.AnnotationVendor: "TestVendor2", - }, - }, - }) - So(err, ShouldBeNil) - - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob1, - }, - "digestTag1.0.2": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob2, - }, - } - - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil - }, - } - - const query = "repo1:1.0.1" - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - - mockCve := mocks.CveInfoMock{} - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - repos, images, layers, err := globalSearch(responseContext, query, mockMetaDB, - &gql_generated.Filter{}, &pageInput, mockCve, log.NewLogger("debug", "")) - So(err, ShouldBeNil) - So(images, ShouldNotBeEmpty) - So(layers, ShouldBeEmpty) - So(repos.Results, ShouldBeEmpty) - }) }) } @@ -421,24 +95,18 @@ func TestRepoListWithNewestImage(t *testing.T) { Convey("RepoListWithNewestImage", t, func() { Convey("MetaDB SearchRepos error", func() { mockMetaDB := mocks.MetaDBMock{ - SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error, - ) { - return make([]mTypes.RepoMetadata, 0), make(map[string]mTypes.ManifestMetadata), - map[string]mTypes.IndexData{}, ErrTestError + SearchReposFn: func(ctx context.Context, searchText string) ([]mTypes.RepoMeta, error) { + return []mTypes.RepoMeta{}, ErrTestError }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) mockCve := mocks.CveInfoMock{} - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaUpdateTime pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(1), + Offset: ref(0), + SortBy: ref(gql_generated.SortCriteriaUpdateTime), } repos, err := repoListWithNewestImage(responseContext, mockCve, log.NewLogger("debug", ""), &pageInput, mockMetaDB) So(err, ShouldNotBeNil) @@ -458,164 +126,90 @@ func TestRepoListWithNewestImage(t *testing.T) { So(err, ShouldNotBeNil) }) - Convey("MetaDB SearchRepo bad manifest referenced", func() { + Convey("Working SearchRepo function", func() { + createTime := time.Now() + createTime2 := createTime.Add(time.Second) + img1 := CreateImageWith().DefaultLayers(). + ImageConfig(ispec.Image{ + Config: ispec.ImageConfig{ + Labels: map[string]string{}, + }, + Created: &createTime, + }).Build() + + img2 := CreateImageWith().DefaultLayers(). + ImageConfig(ispec.Image{ + Config: ispec.ImageConfig{ + Labels: map[string]string{}, + }, + Created: &createTime2, + }).Build() + mockMetaDB := mocks.MetaDBMock{ - SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ + SearchReposFn: func(ctx context.Context, searchText string) ([]mTypes.RepoMeta, error) { + repos := []mTypes.RepoMeta{ { Name: "repo1", Tags: map[string]mTypes.Descriptor{ "1.0.1": { - Digest: "digestTag1.0.1", + Digest: img1.DigestStr(), MediaType: ispec.MediaTypeImageManifest, }, }, Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { + img1.DigestStr(): { "cosign": []mTypes.SignatureInfo{ {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, }, }, }, - Stars: 100, - }, - { - Name: "repo2", - Tags: map[string]mTypes.Descriptor{ - "1.0.2": { - Digest: "digestTag1.0.2", + StarCount: 100, + LastUpdatedImage: &mTypes.LastUpdatedImage{ + Descriptor: mTypes.Descriptor{ + Digest: img1.DigestStr(), MediaType: ispec.MediaTypeImageManifest, }, + Tag: "1.0.1", + LastUpdated: &createTime, }, - Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, - }, - }, - }, - Stars: 100, - }, - } - - configBlob1, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: []byte("bad manifest blob"), - ConfigBlob: configBlob1, }, - "digestTag1.0.2": { - ManifestBlob: []byte("bad manifest blob"), - ConfigBlob: configBlob1, - }, - } - - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil - }, - } - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - mockCve := mocks.CveInfoMock{} - - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaUpdateTime - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - repos, err := repoListWithNewestImage(responseContext, mockCve, log.NewLogger("debug", ""), &pageInput, mockMetaDB) - So(err, ShouldBeNil) - So(repos.Results, ShouldNotBeEmpty) - }) - - Convey("Working SearchRepo function", func() { - createTime := time.Now() - createTime2 := createTime.Add(time.Second) - mockMetaDB := mocks.MetaDBMock{ - SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ { - Name: "repo1", + Name: "repo2", Tags: map[string]mTypes.Descriptor{ - "1.0.1": { - Digest: "digestTag1.0.1", + "1.0.2": { + Digest: img2.DigestStr(), MediaType: ispec.MediaTypeImageManifest, }, }, Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { + img1.DigestStr(): { "cosign": []mTypes.SignatureInfo{ {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, }, }, }, - Stars: 100, - }, - { - Name: "repo2", - Tags: map[string]mTypes.Descriptor{ - "1.0.2": { - Digest: "digestTag1.0.2", + StarCount: 100, + LastUpdatedImage: &mTypes.LastUpdatedImage{ + Descriptor: mTypes.Descriptor{ + Digest: img2.DigestStr(), MediaType: ispec.MediaTypeImageManifest, }, + Tag: "1.0.2", + LastUpdated: &createTime2, }, - Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, - }, - }, - }, - Stars: 100, }, } - configBlob1, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - Created: &createTime, - }) - So(err, ShouldBeNil) - - configBlob2, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - Created: &createTime2, - }) - So(err, ShouldBeNil) - - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob1, - }, - "digestTag1.0.2": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob2, - }, - } - - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil + return repos, nil + }, + FilterImageDataFn: func(ctx context.Context, digests []string, + ) (map[string]mTypes.ImageData, error) { + return map[string]mTypes.ImageData{ + img1.DigestStr(): mConvert.GetImageManifestData(img1.Manifest, img1.Config, + img1.ManifestDescriptor.Size, img1.ManifestDescriptor.Digest), + img2.DigestStr(): mConvert.GetImageManifestData(img2.Manifest, img2.Config, + img2.ManifestDescriptor.Size, img2.ManifestDescriptor.Digest), + }, nil }, } Convey("MetaDB missing requestedPage", func() { @@ -628,13 +222,10 @@ func TestRepoListWithNewestImage(t *testing.T) { }) Convey("MetaDB SearchRepo is successful", func() { - limit := 2 - offset := 0 - sortCriteria := gql_generated.SortCriteriaUpdateTime pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(2), + Offset: ref(0), + SortBy: ref(gql_generated.SortCriteriaUpdateTime), } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, @@ -691,59 +282,42 @@ func TestGetStarredRepos(t *testing.T) { }) } -func TestGetFilteredPaginatedRepos(t *testing.T) { - Convey("getFilteredPaginatedRepos FilterRepos fails", t, func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - _, err := getFilteredPaginatedRepos( - responseContext, - mocks.CveInfoMock{}, - func(repoMeta mTypes.RepoMetadata) bool { return true }, - log.NewLogger("debug", ""), - nil, - mocks.MetaDBMock{ - FilterReposFn: func(ctx context.Context, filter mTypes.FilterRepoFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - ErrTestError - }, - }, - ) - So(err, ShouldNotBeNil) - }) +func getTestRepoMetaWithImages(repo string, images []Image) mTypes.RepoMeta { + tags := map[string]mTypes.Descriptor{"": {}} + statistics := map[string]mTypes.DescriptorStatistics{"": {}} + signatures := map[string]mTypes.ManifestSignatures{"": {}} + referrers := map[string][]mTypes.ReferrerInfo{"": {}} + + for i := range images { + tags[images[i].DigestStr()] = mTypes.Descriptor{} + statistics[images[i].DigestStr()] = mTypes.DescriptorStatistics{} + signatures[images[i].DigestStr()] = mTypes.ManifestSignatures{} + referrers[images[i].DigestStr()] = []mTypes.ReferrerInfo{} + } - Convey("Paginated convert fails", t, func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - _, err := getFilteredPaginatedRepos(responseContext, - mocks.CveInfoMock{}, - func(repoMeta mTypes.RepoMetadata) bool { return true }, - log.NewLogger("debug", ""), - &gql_generated.PageInput{Limit: ref(-1)}, - mocks.MetaDBMock{}, - ) - So(err, ShouldNotBeNil) - }) + return mTypes.RepoMeta{ + Name: repo, + Tags: tags, + Statistics: statistics, + Signatures: signatures, + Referrers: referrers, + } } func TestImageListForDigest(t *testing.T) { Convey("getImageList", t, func() { Convey("no page requested, FilterTagsFn returns error", func() { - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - ErrTestError + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, + filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) - _, err := getImageListForDigest(responseContext, "invalid", mockSearchDB, mocks.CveInfoMock{}, nil) + _, err := getImageListForDigest(responseContext, "invalid", mockMetaDB, mocks.CveInfoMock{}, nil) So(err, ShouldNotBeNil) }) @@ -753,669 +327,120 @@ func TestImageListForDigest(t *testing.T) { _, err := getImageListForDigest(responseContext, "invalid", mocks.MetaDBMock{}, mocks.CveInfoMock{}, &gql_generated.PageInput{Limit: ref(-1)}) So(err, ShouldNotBeNil) - }) - - Convey("invalid manifest blob", func() { - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "test", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: "digestTag1.0.1", MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, - }, - } - - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - manifestBlob := []byte("invalid") - - manifestsMetaData := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 0, - }, - } - - return repos, manifestsMetaData, map[string]mTypes.IndexData{}, nil - }, - } - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - - imageList, err := getImageListForDigest(responseContext, "test", mockSearchDB, mocks.CveInfoMock{}, nil) - So(err, ShouldBeNil) - So(imageList.Results, ShouldBeEmpty) - }) - - Convey("valid imageListForDigest returned for matching manifest digest", func() { - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob).String() - - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "test", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: manifestDigest, MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, - }, - } - - configBlob, err := json.Marshal(ispec.ImageConfig{}) - So(err, ShouldBeNil) - - manifestsMetaData := map[string]mTypes.ManifestMetadata{ - manifestDigest: { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 0, - }, - } - matchedTags := repos[0].Tags - for tag, manifestDescriptor := range repos[0].Tags { - if !filterFunc(repos[0], manifestsMetaData[manifestDescriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMetaData, manifestDescriptor.Digest) - - continue - } - } - - repos[0].Tags = matchedTags - - return repos, manifestsMetaData, map[string]mTypes.IndexData{}, nil - }, - } - - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - imageSummaries, err := getImageListForDigest(responseContext, manifestDigest, - mockSearchDB, mocks.CveInfoMock{}, &pageInput) - So(err, ShouldBeNil) - So(len(imageSummaries.Results), ShouldEqual, 1) - - imageSummaries, err = getImageListForDigest(responseContext, "invalid", - mockSearchDB, mocks.CveInfoMock{}, &pageInput) - So(err, ShouldBeNil) - So(len(imageSummaries.Results), ShouldEqual, 0) - }) - - Convey("valid imageListForDigest returned for matching config digest", func() { - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob).String() - - configBlob, err := json.Marshal(ispec.Image{}) - So(err, ShouldBeNil) - - configDigest := godigest.FromBytes(configBlob) - - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "test", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: manifestDigest, MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, - }, - } - - manifestBlob, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - Digest: configDigest, - }, - }) - So(err, ShouldBeNil) - - manifestsMetaData := map[string]mTypes.ManifestMetadata{ - manifestDigest: { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 0, - }, - } - - matchedTags := repos[0].Tags - for tag, manifestDescriptor := range repos[0].Tags { - if !filterFunc(repos[0], manifestsMetaData[manifestDescriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMetaData, manifestDescriptor.Digest) - - continue - } - } - - repos[0].Tags = matchedTags - - return repos, manifestsMetaData, map[string]mTypes.IndexData{}, nil - }, - } - - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - - imageSummaries, err := getImageListForDigest(responseContext, configDigest.String(), - mockSearchDB, mocks.CveInfoMock{}, &pageInput) - So(err, ShouldBeNil) - So(len(imageSummaries.Results), ShouldEqual, 1) - }) - - Convey("valid imageListForDigest returned for matching layer digest", func() { - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob).String() - - configBlob, err := json.Marshal(ispec.Image{}) - So(err, ShouldBeNil) - - layerDigest := godigest.Digest("validDigest") - - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "test", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: manifestDigest, MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, - }, - } - - manifestBlob, err := json.Marshal(ispec.Manifest{ - Layers: []ispec.Descriptor{ - { - Digest: layerDigest, - }, - }, - }) - So(err, ShouldBeNil) - - manifestsMetaData := map[string]mTypes.ManifestMetadata{ - manifestDigest: { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 0, - }, - } - - matchedTags := repos[0].Tags - for tag, manifestDescriptor := range repos[0].Tags { - if !filterFunc(repos[0], manifestsMetaData[manifestDescriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMetaData, manifestDescriptor.Digest) - - continue - } - } - - repos[0].Tags = matchedTags - - return repos, manifestsMetaData, map[string]mTypes.IndexData{}, nil - }, - } - - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } - - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - - imageSummaries, err := getImageListForDigest(responseContext, layerDigest.String(), - mockSearchDB, mocks.CveInfoMock{}, &pageInput) - So(err, ShouldBeNil) - So(len(imageSummaries.Results), ShouldEqual, 1) - }) - - Convey("valid imageListForDigest, multiple matching tags", func() { - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob).String() - - configBlob, err := json.Marshal(ispec.Image{}) - So(err, ShouldBeNil) - - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "test", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: manifestDigest, MediaType: ispec.MediaTypeImageManifest}, - "1.0.2": {Digest: manifestDigest, MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, - }, - } + }) - manifestsMetaData := map[string]mTypes.ManifestMetadata{ - manifestDigest: { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 0, - }, - } + Convey("valid imageListForDigest returned for matching manifest digest", func() { + img1, img2 := CreateRandomImage(), CreateRandomImage() + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, + filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + imageDataList := []mTypes.ImageData{img1.AsImageData(), img2.AsImageData()} + repoMeta := getTestRepoMetaWithImages("repo", []Image{img1, img2}) + tags := []string{"tag1", "tag2"} - for i, repo := range repos { - matchedTags := repo.Tags + acceptedImages := []mTypes.ImageMeta{} - for tag, manifestDescriptor := range repo.Tags { - if !filterFunc(repo, manifestsMetaData[manifestDescriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMetaData, manifestDescriptor.Digest) + for i := range imageDataList { + if filterFunc(repoMeta, imageDataList[i]) { + acceptedImages = append(acceptedImages, + convert.GetImageMeta(tags[i], repoMeta, imageDataList[i])) - continue - } + continue } - - repos[i].Tags = matchedTags } - return repos, manifestsMetaData, map[string]mTypes.IndexData{}, nil + return acceptedImages, nil }, } - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(1), + Offset: ref(0), + SortBy: ref(gql_generated.SortCriteriaAlphabeticAsc), } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) - - imageSummaries, err := getImageListForDigest(responseContext, manifestDigest, - mockSearchDB, mocks.CveInfoMock{}, &pageInput) + imageSummaries, err := getImageListForDigest(responseContext, img1.DigestStr(), + mockMetaDB, mocks.CveInfoMock{}, &pageInput) So(err, ShouldBeNil) So(len(imageSummaries.Results), ShouldEqual, 1) - }) - Convey("valid imageListForDigest, multiple matching tags limited by pageInput", func() { - manifestBlob, err := json.Marshal(ispec.Manifest{}) + imageSummaries, err = getImageListForDigest(responseContext, "invalid", + mockMetaDB, mocks.CveInfoMock{}, &pageInput) So(err, ShouldBeNil) + So(len(imageSummaries.Results), ShouldEqual, 0) - manifestDigest := godigest.FromBytes(manifestBlob).String() + imageSummaries, err = getImageListForDigest(responseContext, img1.Manifest.Config.Digest.String(), + mockMetaDB, mocks.CveInfoMock{}, &pageInput) + So(err, ShouldBeNil) + So(len(imageSummaries.Results), ShouldEqual, 1) - configBlob, err := json.Marshal(ispec.Image{}) + imageSummaries, err = getImageListForDigest(responseContext, img1.Manifest.Layers[0].Digest.String(), + mockMetaDB, mocks.CveInfoMock{}, &pageInput) So(err, ShouldBeNil) + So(len(imageSummaries.Results), ShouldEqual, 1) + }) - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "test", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: manifestDigest, MediaType: ispec.MediaTypeImageManifest}, - "1.0.2": {Digest: manifestDigest, MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, - }, - } + Convey("valid imageListForDigest, multiple matching tags", func() { + img1 := CreateRandomImage() - manifestsMetaData := map[string]mTypes.ManifestMetadata{ - manifestDigest: { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 0, - }, - } + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, + filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + imageDataList := []mTypes.ImageData{img1.AsImageData()} + repoMeta := getTestRepoMetaWithImages("repo", []Image{img1, img1}) + tags := []string{"tag1", "tag2"} - for i, repo := range repos { - matchedTags := repo.Tags + acceptedImages := []mTypes.ImageMeta{} - for tag, manifestDescriptor := range repo.Tags { - if !filterFunc(repo, manifestsMetaData[manifestDescriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMetaData, manifestDescriptor.Digest) + for i := range imageDataList { + if filterFunc(repoMeta, imageDataList[i]) { + acceptedImages = append(acceptedImages, + convert.GetImageMeta(tags[i], repoMeta, imageDataList[i])) - continue - } + continue } - - repos[i].Tags = matchedTags - - repos = append(repos, repo) } - return repos, manifestsMetaData, map[string]mTypes.IndexData{}, nil + return acceptedImages, nil }, } - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(1), + Offset: ref(0), + SortBy: ref(gql_generated.SortCriteriaAlphabeticAsc), } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) - imageSummaries, err := getImageListForDigest(responseContext, manifestDigest, - mockSearchDB, mocks.CveInfoMock{}, &pageInput) + imageSummaries, err := getImageListForDigest(responseContext, img1.DigestStr(), + mockMetaDB, mocks.CveInfoMock{}, &pageInput) So(err, ShouldBeNil) So(len(imageSummaries.Results), ShouldEqual, 1) }) }) } -func TestGetImageSummary(t *testing.T) { - Convey("GetImageSummary", t, func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - - Convey("Media Type: ImageManifest", func() { - Convey("metaDB.GetManifestMeta fails", func() { - var ( - metaDB = mocks.MetaDBMock{ - GetManifestDataFn: func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, ErrTestError - }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": {MediaType: ispec.MediaTypeImageManifest, Digest: "digest"}, - }, - }, nil - }, - } - - log = log.NewLogger("debug", "") - - skip = convert.SkipQGLField{ - Vulnerabilities: true, - } - ) - - _, err := getImageSummary(responseContext, "repo", "tag", nil, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldNotBeNil) - }) - - Convey("0 len return", func() { - var ( - metaDB = mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": {MediaType: ispec.MediaTypeImageManifest, Digest: "digest"}, - }, - }, nil - }, - } - - log = log.NewLogger("debug", "") - - skip = convert.SkipQGLField{ - Vulnerabilities: true, - } - ) - - _, err := getImageSummary(responseContext, "repo", "tag", nil, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldBeNil) - }) - - Convey("digest != nil && *digest != actual image digest", func() { - var ( - metaDB = mocks.MetaDBMock{ - GetManifestMetaFn: func(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - return mTypes.ManifestMetadata{}, ErrTestError - }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": {MediaType: ispec.MediaTypeImageManifest, Digest: "digest"}, - }, - }, nil - }, - } - - log = log.NewLogger("debug", "") - - digest = "wrongDigest" - - skip = convert.SkipQGLField{ - Vulnerabilities: true, - } - ) - - _, err := getImageSummary(responseContext, "repo", "tag", &digest, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldNotBeNil) - }) - }) - - Convey("Media Type: ImageIndex", func() { - Convey("metaDB.GetIndexData fails", func() { - var ( - metaDB = mocks.MetaDBMock{ - GetIndexDataFn: func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{}, ErrTestError - }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": {MediaType: ispec.MediaTypeImageIndex, Digest: "digest"}, - }, - }, nil - }, - } - - log = log.NewLogger("debug", "") - - skip = convert.SkipQGLField{ - Vulnerabilities: true, - } - ) - - _, err := getImageSummary(responseContext, "repo", "tag", nil, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldNotBeNil) - }) - - Convey("json.Unmarshal(indexData.IndexBlob, &indexContent) fails", func() { - var ( - metaDB = mocks.MetaDBMock{ - GetIndexDataFn: func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{ - IndexBlob: []byte("bad json"), - }, nil - }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": {MediaType: ispec.MediaTypeImageIndex, Digest: "digest"}, - }, - }, nil - }, - } - - log = log.NewLogger("debug", "") - - skip = convert.SkipQGLField{ - Vulnerabilities: true, - } - ) - - _, err := getImageSummary(responseContext, "repo", "tag", nil, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldNotBeNil) - }) - - Convey("digest != nil", func() { - index := ispec.Index{ - Manifests: []ispec.Descriptor{ - { - Digest: "digest", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - } - - indexBlob, err := json.Marshal(index) - So(err, ShouldBeNil) - - metaDB := mocks.MetaDBMock{ - GetIndexDataFn: func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{ - IndexBlob: indexBlob, - }, nil - }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": {MediaType: ispec.MediaTypeImageIndex, Digest: "digest"}, - }, - }, nil - }, - } - - log := log.NewLogger("debug", "") - - goodDigest := "goodDigest" - - Convey("digest not found", func() { - wrongDigest := "wrongDigest" - - skip := convert.SkipQGLField{ - Vulnerabilities: true, - } - - _, err = getImageSummary(responseContext, "repo", "tag", &wrongDigest, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldNotBeNil) - }) - - Convey("GetManifestData error", func() { - metaDB.GetManifestDataFn = func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, ErrTestError - } - - skip := convert.SkipQGLField{ - Vulnerabilities: true, - } - - _, err = getImageSummary(responseContext, "repo", "tag", &goodDigest, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldNotBeNil) - }) - }) - }) - - Convey("Media Type: not supported", func() { - var ( - metaDB = mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tag": {MediaType: "unknown", Digest: "digest"}, - }, - }, nil - }, - } - - log = log.NewLogger("debug", "") - - skip = convert.SkipQGLField{ - Vulnerabilities: true, - } - ) - - _, err := getImageSummary(responseContext, "repo", "tag", nil, skip, metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldBeNil) - }) - }) -} - -func TestFilterBaseImagesFn(t *testing.T) { - Convey("FilterBaseImages", t, func() { - filterFunc := filterBaseImages(&gql_generated.ImageSummary{}) - ok := filterFunc( - mTypes.RepoMetadata{}, - mTypes.ManifestMetadata{ - ManifestBlob: []byte("bad json"), - }, - ) - So(ok, ShouldBeFalse) - }) -} - -func TestImageList(t *testing.T) { +func TestImageListError(t *testing.T) { Convey("getImageList", t, func() { - testLogger := log.NewLogger("debug", "") + testLogger := log.NewLogger("debug", "/dev/null") Convey("no page requested, SearchRepoFn returns error", func() { - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{}, ErrTestError + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) - _, err := getImageList(responseContext, "test", mockSearchDB, mocks.CveInfoMock{}, nil, testLogger) + _, err := getImageList(responseContext, "test", mockMetaDB, mocks.CveInfoMock{}, nil, testLogger) So(err, ShouldNotBeNil) }) @@ -1430,81 +455,43 @@ func TestImageList(t *testing.T) { }) Convey("valid repoList returned", func() { - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "test", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": { - Digest: "digestTag1.0.1", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - Signatures: map[string]mTypes.ManifestSignatures{ - "digestTag1.0.1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "testSignature", LayersInfo: []mTypes.LayerInfo{}}, - }, - }, - }, - Stars: 100, - }, - } + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + repoName := "correct-repo" - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - manifestBlob, err := json.Marshal(ispec.Manifest{}) - So(err, ShouldBeNil) - - manifestsMetaData := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 0, - Signatures: mTypes.ManifestSignatures{ - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "digestSignature1"}, - }, - }, - }, + if !filterRepoTag(repoName, "tag") { + return []mTypes.ImageMeta{}, nil } - if !filterFunc(repos[0], manifestsMetaData["digestTag1.0.1"]) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{}, nil + image := CreateDefaultImage() + repoMeta := mTypes.RepoMeta{ + Name: "repo", + Tags: map[string]mTypes.Descriptor{image.DigestStr(): { + Digest: image.DigestStr(), + MediaType: ispec.MediaTypeImageManifest, + }}, } - return repos, manifestsMetaData, map[string]mTypes.IndexData{}, nil + return []mTypes.ImageMeta{convert.GetImageMeta("tag", repoMeta, image.AsImageData())}, nil }, } - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(1), + Offset: ref(0), + SortBy: ref(gql_generated.SortCriteriaAlphabeticAsc), } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) - imageSummaries, err := getImageList(responseContext, "test", mockSearchDB, + imageSummaries, err := getImageList(responseContext, "correct-repo", mockMetaDB, mocks.CveInfoMock{}, &pageInput, testLogger) So(err, ShouldBeNil) So(len(imageSummaries.Results), ShouldEqual, 1) - imageSummaries, err = getImageList(responseContext, "invalid", mockSearchDB, + imageSummaries, err = getImageList(responseContext, "invalid", mockMetaDB, mocks.CveInfoMock{}, &pageInput, testLogger) So(err, ShouldBeNil) So(len(imageSummaries.Results), ShouldEqual, 0) @@ -1580,35 +567,21 @@ func TestQueryResolverErrors(t *testing.T) { graphql.DefaultRecover) Convey("GlobalSearch error bad requested page", func() { - resolverConfig := NewResolver( - log, - storage.StoreController{}, - mocks.MetaDBMock{}, - mocks.CveInfoMock{}, - ) - - resolver := queryResolver{ - resolverConfig, - } - - limit := -1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc + resolverConfig := NewResolver(log, storage.StoreController{}, mocks.MetaDBMock{}, mocks.CveInfoMock{}) + resolver := queryResolver{resolverConfig} pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(-1), + Offset: ref(0), + SortBy: ref(gql_generated.SortCriteriaAlphabeticAsc), } _, err := resolver.GlobalSearch(ctx, "some_string", &gql_generated.Filter{}, &pageInput) So(err, ShouldNotBeNil) - limit = 0 - offset = -1 pageInput = gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(0), + Offset: ref(-1), + SortBy: ref(gql_generated.SortCriteriaAlphabeticAsc), } _, err = resolver.GlobalSearch(ctx, "some_string", &gql_generated.Filter{}, &pageInput) @@ -1622,17 +595,15 @@ func TestQueryResolverErrors(t *testing.T) { DefaultStore: mocks.MockedImageStore{}, }, mocks.MetaDBMock{ - GetMultipleRepoMetaFn: func(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata) bool, - ) ([]mTypes.RepoMetadata, error) { - return []mTypes.RepoMetadata{}, ErrTestError + GetMultipleRepoMetaFn: func(ctx context.Context, filter func(repoMeta mTypes.RepoMeta) bool, + ) ([]mTypes.RepoMeta, error) { + return []mTypes.RepoMeta{}, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.ImageListForCve(ctx, "cve1", &gql_generated.Filter{}, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) @@ -1645,21 +616,15 @@ func TestQueryResolverErrors(t *testing.T) { DefaultStore: mocks.MockedImageStore{}, }, mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - ErrTestError + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.ImageListForCve(ctx, "cve1", &gql_generated.Filter{}, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) @@ -1672,21 +637,15 @@ func TestQueryResolverErrors(t *testing.T) { DefaultStore: mocks.MockedImageStore{}, }, mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - ErrTestError + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.ImageListWithCVEFixed(ctx, "cve1", "image", &gql_generated.Filter{}, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) @@ -1700,18 +659,14 @@ func TestQueryResolverErrors(t *testing.T) { }, mocks.MetaDBMock{ SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return nil, nil, nil, ErrTestError + ) ([]mTypes.RepoMeta, error) { + return nil, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.RepoListWithNewestImage(ctx, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) @@ -1723,18 +678,14 @@ func TestQueryResolverErrors(t *testing.T) { storage.StoreController{}, mocks.MetaDBMock{ SearchReposFn: func(ctx context.Context, searchText string, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return nil, nil, nil, ErrTestError + ) ([]mTypes.RepoMeta, error) { + return nil, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.RepoListWithNewestImage(ctx, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) @@ -1745,21 +696,15 @@ func TestQueryResolverErrors(t *testing.T) { log, storage.StoreController{}, mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - ErrTestError + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.ImageList(ctx, "repo", &gql_generated.PageInput{}) So(err, ShouldNotBeNil) @@ -1779,16 +724,14 @@ func TestQueryResolverErrors(t *testing.T) { }, }, mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{}, ErrTestError + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{}, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.DerivedImageList(ctx, "repo:tag", nil, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) @@ -1808,71 +751,48 @@ func TestQueryResolverErrors(t *testing.T) { }, }, mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{}, ErrTestError + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{}, ErrTestError }, }, mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.BaseImageList(ctx, "repo:tag", nil, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) }) Convey("DerivedImageList and BaseImage List FilterTags() errors", func() { - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - manifest := ispec.Manifest{} - - manifestBlob, err := json.Marshal(manifest) - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob) + image := CreateDefaultImage() resolverConfig := NewResolver( log, storage.StoreController{}, mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - ErrTestError + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Name: "repo", Tags: map[string]mTypes.Descriptor{ - "tag": {Digest: manifestDigest.String(), MediaType: ispec.MediaTypeImageManifest}, + "tag": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, }, nil }, - GetManifestMetaFn: func(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - return mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - }, nil + GetImageDataFn: func(digest godigest.Digest) (mTypes.ImageData, error) { + return image.AsImageData(), nil }, }, mocks.CveInfoMock{}, ) - resolver := queryResolver{ - resolverConfig, - } + resolver := queryResolver{resolverConfig} - _, err = resolver.DerivedImageList(ctx, "repo:tag", nil, &gql_generated.PageInput{}) + _, err := resolver.DerivedImageList(ctx, "repo:tag", nil, &gql_generated.PageInput{}) So(err, ShouldNotBeNil) _, err = resolver.BaseImageList(ctx, "repo:tag", nil, &gql_generated.PageInput{}) @@ -1893,9 +813,7 @@ func TestQueryResolverErrors(t *testing.T) { mocks.CveInfoMock{}, ) - qr := queryResolver{ - resolverConfig, - } + qr := queryResolver{resolverConfig} _, err := qr.Referrers(ctx, "repo", "", nil) So(err, ShouldNotBeNil) @@ -1904,180 +822,78 @@ func TestQueryResolverErrors(t *testing.T) { } func TestCVEResolvers(t *testing.T) { //nolint:gocyclo - params := boltdb.DBParameters{ - RootDir: t.TempDir(), - } - + ctx := context.Background() + log := log.NewLogger("debug", "") LINUX := "linux" AMD := "amd" ARM := "arm64" - boltDriver, err := boltdb.GetBoltDriver(params) + boltDriver, err := boltdb.GetBoltDriver(boltdb.DBParameters{RootDir: t.TempDir()}) if err != nil { panic(err) } - log := log.NewLogger("debug", "") - metaDB, err := boltdb.New(boltDriver, log) if err != nil { panic(err) } - // Create metadb data for scannable image with vulnerabilities - // Create manifest metadata first - timeStamp1 := time.Date(2008, 1, 1, 12, 0, 0, 0, time.UTC) - - configBlob1, err := json.Marshal(ispec.Image{ - Created: &timeStamp1, - Platform: ispec.Platform{ - Architecture: AMD, - OS: LINUX, - }, - }) - if err != nil { - panic(err) - } - - manifestBlob1, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, - Size: 0, - Digest: godigest.FromBytes(configBlob1), - }, - Layers: []ispec.Descriptor{ - { - MediaType: ispec.MediaTypeImageLayerGzip, - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), - }, - }, - }) - if err != nil { - panic(err) - } - - repoMeta1 := mTypes.ManifestData{ - ManifestBlob: manifestBlob1, - ConfigBlob: configBlob1, - } - - digest1 := godigest.FromBytes(manifestBlob1) - - err = metaDB.SetManifestData(digest1, repoMeta1) - if err != nil { - panic(err) - } - - timeStamp2 := time.Date(2009, 1, 1, 12, 0, 0, 0, time.UTC) - - configBlob2, err := json.Marshal(ispec.Image{ - Created: &timeStamp2, + image1 := CreateImageWith().RandomLayers(5, 2).ImageConfig(ispec.Image{ + Created: DateRef(2008, 1, 1, 12, 0, 0, 0, time.UTC), Platform: ispec.Platform{ Architecture: AMD, OS: LINUX, - }, - }) - if err != nil { - panic(err) - } - - manifestBlob2, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, - Size: 0, - Digest: godigest.FromBytes(configBlob2), - }, - Layers: []ispec.Descriptor{ - { - MediaType: ispec.MediaTypeImageLayerGzip, - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), - }, - }, - }) - if err != nil { - panic(err) - } - - repoMeta2 := mTypes.ManifestData{ - ManifestBlob: manifestBlob2, - ConfigBlob: configBlob2, - } - - digest2 := godigest.FromBytes(manifestBlob2) - - err = metaDB.SetManifestData(digest2, repoMeta2) - if err != nil { - panic(err) - } + }, + }).Build() + digest1 := image1.Digest() - timeStamp3 := time.Date(2010, 1, 1, 12, 0, 0, 0, time.UTC) + image2 := CreateImageWith().RandomLayers(5, 2).ImageConfig(ispec.Image{ + Created: DateRef(2009, 1, 1, 12, 0, 0, 0, time.UTC), + Platform: ispec.Platform{ + Architecture: AMD, + OS: LINUX, + }, + }).Build() + digest2 := image2.Digest() - configBlob3, err := json.Marshal(ispec.Image{ - Created: &timeStamp3, + image3 := CreateImageWith().RandomLayers(5, 2).ImageConfig(ispec.Image{ + Created: DateRef(2010, 1, 1, 12, 0, 0, 0, time.UTC), Platform: ispec.Platform{ Architecture: ARM, OS: LINUX, }, - }) - if err != nil { - panic(err) - } + }).Build() + digest3 := image3.Digest() - manifestBlob3, err := json.Marshal(ispec.Manifest{ - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, - Size: 0, - Digest: godigest.FromBytes(configBlob3), + ctx, err = ociutils.InitializeTestMetaDB(ctx, metaDB, + ociutils.Repo{ + Name: "repo1", Images: []ociutils.RepoImage{ + {Image: image1, Reference: "1.0.0"}, + {Image: image2, Reference: "1.0.1"}, + {Image: image3, Reference: "1.1.0"}, + {Image: image3, Reference: "latest"}, + }, }, - Layers: []ispec.Descriptor{ - { - MediaType: ispec.MediaTypeImageLayerGzip, - Size: 0, - Digest: godigest.NewDigestFromEncoded(godigest.SHA256, "digest"), + ociutils.Repo{ + Name: "repo2", Images: []ociutils.RepoImage{ + {Image: image1, Reference: "2.0.0"}, + {Image: image2, Reference: "2.0.1"}, + {Image: image3, Reference: "2.1.0"}, + {Image: image3, Reference: "latest"}, }, }, - }) - if err != nil { - panic(err) - } - - repoMeta3 := mTypes.ManifestData{ - ManifestBlob: manifestBlob3, - ConfigBlob: configBlob3, - } - - digest3 := godigest.FromBytes(manifestBlob3) - - err = metaDB.SetManifestData(digest3, repoMeta3) + ociutils.Repo{ + Name: "repo3", Images: []ociutils.RepoImage{ + {Image: image2, Reference: "3.0.1"}, + {Image: image3, Reference: "3.1.0"}, + {Image: image3, Reference: "latest"}, + }, + }, + ) if err != nil { panic(err) } - // Create the repo metadata using previously defined manifests - tagsMap := map[string]godigest.Digest{} - tagsMap["repo1:1.0.0"] = digest1 - tagsMap["repo1:1.0.1"] = digest2 - tagsMap["repo1:1.1.0"] = digest3 - tagsMap["repo1:latest"] = digest3 - tagsMap["repo2:2.0.0"] = digest1 - tagsMap["repo2:2.0.1"] = digest2 - tagsMap["repo2:2.1.0"] = digest3 - tagsMap["repo2:latest"] = digest3 - tagsMap["repo3:3.0.1"] = digest2 - tagsMap["repo3:3.1.0"] = digest3 - tagsMap["repo3:latest"] = digest3 - - for image, digest := range tagsMap { - repo, tag := common.GetImageDirAndTag(image) - - err := metaDB.SetRepoReference(repo, tag, digest, ispec.MediaTypeImageManifest) - if err != nil { - panic(err) - } - } - getCveResults := func(digestStr string) map[string]cvemodel.CVE { if digestStr == digest1.String() { return map[string]cvemodel.CVE{ @@ -2144,17 +960,19 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo // Setup test CVE data in mock scanner scanner := mocks.CveScannerMock{ ScanImageFn: func(image string) (map[string]cvemodel.CVE, error) { - digest, ok := tagsMap[image] - if !ok { - if !strings.Contains(image, "@") { - return map[string]cvemodel.CVE{}, nil - } + repo, ref, _, _ := common.GetRepoReference(image) - _, digestStr := common.GetImageDirAndDigest(image) - digest = godigest.Digest(digestStr) + if common.IsDigest(ref) { + return getCveResults(ref), nil } - return getCveResults(digest.String()), nil + repoMeta, _ := metaDB.GetRepoMeta(context.Background(), repo) + + if _, ok := repoMeta.Tags[ref]; !ok { + panic("unexpected tag '" + ref + "', test might be wrong") + } + + return getCveResults(repoMeta.Tags[ref].Digest), nil }, GetCachedResultFn: func(digestStr string) map[string]cvemodel.CVE { return getCveResults(digestStr) @@ -2172,12 +990,11 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo Convey("Get CVE list for image ", t, func() { Convey("Unpaginated request to get all CVEs in an image", func() { - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc pageInput := &gql_generated.PageInput{ - SortBy: &sortCriteria, + SortBy: ref(gql_generated.SortCriteriaAlphabeticAsc), } - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover) dig := godigest.FromString("dig") @@ -2267,7 +1084,7 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo Limit: ref(-1), } - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover) _, err = getCVEListForImage(responseContext, "repo1:1.1.0", cveInfo, pageInput, "", log) @@ -2277,7 +1094,7 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo Convey("Get a list of images affected by a particular CVE ", t, func() { Convey("Unpaginated request", func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover) images, err := getImageListForCVE(responseContext, "CVE1", cveInfo, nil, nil, metaDB, log) @@ -2327,7 +1144,7 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo Limit: ref(-1), } - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover) _, err = getImageListForCVE(responseContext, "repo1:1.1.0", cveInfo, &gql_generated.Filter{}, @@ -2336,7 +1153,7 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo }) Convey("Paginated requests", func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover, ) @@ -2546,7 +1363,7 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo Convey("Get a list of images where a particular CVE is fixed", t, func() { Convey("Unpaginated request", func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover) images, err := getImageListWithCVEFixed(responseContext, "CVE1", "repo1", cveInfo, nil, nil, metaDB, log) @@ -2583,13 +1400,13 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo Limit: ref(-1), } - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover) _, err = getImageListWithCVEFixed(responseContext, "cve", "repo1:1.1.0", cveInfo, &gql_generated.Filter{}, pageInput, mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Tags: map[string]mTypes.Descriptor{ "1.1.0": { Digest: godigest.FromString("str").String(), @@ -2603,7 +1420,7 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo }) Convey("Paginated requests", func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, + responseContext := graphql.WithResponseContext(ctx, graphql.DefaultErrorPresenter, graphql.DefaultRecover, ) @@ -2778,7 +1595,7 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo Convey("Errors for cve resolvers", t, func() { _, err := getImageListForCVE( - context.Background(), + ctx, "id", mocks.CveInfoMock{ GetImageListForCVEFn: func(repo, cveID string) ([]cvemodel.TagInfo, error) { @@ -2788,9 +1605,9 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo nil, nil, mocks.MetaDBMock{ - GetMultipleRepoMetaFn: func(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata) bool, - ) ([]mTypes.RepoMetadata, error) { - return []mTypes.RepoMetadata{{}}, nil + GetMultipleRepoMetaFn: func(ctx context.Context, filter func(repoMeta mTypes.RepoMeta) bool, + ) ([]mTypes.RepoMeta, error) { + return []mTypes.RepoMeta{{}}, nil }, }, log, @@ -2799,45 +1616,36 @@ func TestCVEResolvers(t *testing.T) { //nolint:gocyclo }) } -func getPageInput(limit int, offset int) *gql_generated.PageInput { - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - - return &gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, - } -} - -func TestDerivedImageList(t *testing.T) { +func TestMockedDerivedImageList(t *testing.T) { Convey("MetaDB FilterTags error", t, func() { - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return make([]mTypes.RepoMetadata, 0), make(map[string]mTypes.ManifestMetadata), - make(map[string]mTypes.IndexData), ErrTestError + log := log.NewLogger("debug", "/dev/null") + + image := CreateRandomImage() + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{}, ErrTestError + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{}, ErrTestError }, - GetManifestMetaFn: func(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - return mTypes.ManifestMetadata{}, ErrTestError + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return map[string]mTypes.ImageData{image.DigestStr(): image.AsImageData()}, nil }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) mockCve := mocks.CveInfoMock{} - images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockSearchDB, &gql_generated.PageInput{}, - mockCve, log.NewLogger("debug", "")) + images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockMetaDB, &gql_generated.PageInput{}, + mockCve, log) So(err, ShouldNotBeNil) So(images.Results, ShouldBeEmpty) }) Convey("paginated fail", t, func() { + log := log.NewLogger("debug", "/dev/null") + image := CreateRandomImage() pageInput := &gql_generated.PageInput{ Limit: ref(-1), } @@ -2847,223 +1655,133 @@ func TestDerivedImageList(t *testing.T) { _, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Tags: map[string]mTypes.Descriptor{ "1.0.1": { - Digest: godigest.FromString("str").String(), + Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest, }, }, }, nil }, + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return map[string]mTypes.ImageData{image.DigestStr(): image.AsImageData()}, nil + }, }, pageInput, - mocks.CveInfoMock{}, log.NewLogger("debug", "")) + mocks.CveInfoMock{}, log) So(err, ShouldNotBeNil) }) //nolint: dupl Convey("MetaDB FilterTags no repo available", t, func() { - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - manifest := ispec.Manifest{} - - manifestBlob, err := json.Marshal(manifest) - So(err, ShouldBeNil) + log := log.NewLogger("debug", "/dev/null") + image := CreateDefaultImage() - manifestDigest := godigest.FromBytes(manifestBlob) - - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - nil + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, nil }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Name: "repo1", Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: manifestDigest.String(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.1": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, }, nil }, - GetManifestMetaFn: func(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - return mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return map[string]mTypes.ImageData{ + digests[0]: image.AsImageData(), }, nil }, + GetImageDataFn: func(digest godigest.Digest) (mTypes.ImageData, error) { + return image.AsImageData(), nil + }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) mockCve := mocks.CveInfoMock{} - images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockSearchDB, &gql_generated.PageInput{}, - mockCve, log.NewLogger("debug", "")) + images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockMetaDB, &gql_generated.PageInput{}, + mockCve, log) So(err, ShouldBeNil) So(images.Results, ShouldBeEmpty) }) //nolint: dupl Convey("derived image list working", t, func() { - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - configDigest := godigest.FromBytes(configBlob) - - layers := [][]byte{ - {10, 11, 10, 11}, - {11, 11, 11, 11}, - {10, 10, 10, 11}, - {13, 14, 15, 11}, + log := log.NewLogger("debug", "/dev/null") + layer1 := []byte{10, 11, 10, 11} + layer2 := []byte{11, 11, 11, 11} + layer3 := []byte{10, 10, 10, 11} + layer4 := []byte{13, 14, 15, 11} + + image := CreateImageWith(). + LayerBlobs([][]byte{ + layer1, + layer2, + layer3, + }).DefaultConfig().Build() + + derivedImage := CreateImageWith(). + LayerBlobs([][]byte{ + layer1, + layer2, + layer3, + layer4, + }).DefaultConfig().Build() + + imageDataMap := map[string]mTypes.ImageData{ + image.DigestStr(): image.AsImageData(), + derivedImage.DigestStr(): derivedImage.AsImageData(), } - manifestBlob, err := json.Marshal(ispec.Manifest{ - Versioned: specs.Versioned{ - SchemaVersion: 2, - }, - Config: ispec.Descriptor{ - MediaType: "application/vnd.oci.image.config.v1+json", - Digest: configDigest, - Size: int64(len(configBlob)), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[0]), - Size: int64(len(layers[0])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[1]), - Size: int64(len(layers[1])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[2]), - Size: int64(len(layers[2])), - }, - }, - }) - So(err, ShouldBeNil) - - derivedManifestBlob, err := json.Marshal(ispec.Manifest{ - Versioned: specs.Versioned{ - SchemaVersion: 2, - }, - Config: ispec.Descriptor{ - MediaType: "application/vnd.oci.image.config.v1+json", - Digest: configDigest, - Size: int64(len(configBlob)), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[0]), - Size: int64(len(layers[0])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[1]), - Size: int64(len(layers[1])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[2]), - Size: int64(len(layers[2])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[3]), - Size: int64(len(layers[3])), - }, - }, - }) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 100, - Signatures: make(mTypes.ManifestSignatures), - }, - "digestTag1.0.2": { - ManifestBlob: derivedManifestBlob, - ConfigBlob: configBlob, - DownloadCount: 100, - Signatures: make(mTypes.ManifestSignatures), - }, - "digestTag1.0.3": { - ManifestBlob: derivedManifestBlob, - ConfigBlob: configBlob, - DownloadCount: 100, - Signatures: make(mTypes.ManifestSignatures), - }, - } - manifestDigest := godigest.FromBytes(manifestBlob) - - mockSearchDB := mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + mockMetaDB := mocks.MetaDBMock{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Name: "repo1", Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: manifestDigest.String(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.1": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, }, nil }, - GetManifestDataFn: func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - }, nil + GetImageDataFn: func(digest godigest.Digest) (mTypes.ImageData, error) { + return imageDataMap[digest.String()], nil }, - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: "digestTag1.0.1", MediaType: ispec.MediaTypeImageManifest}, - "1.0.2": {Digest: "digestTag1.0.2", MediaType: ispec.MediaTypeImageManifest}, - "1.0.3": {Digest: "digestTag1.0.3", MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, - }, + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + result := map[string]mTypes.ImageData{} + + for _, digest := range digests { + result[digest] = imageDataMap[digest] } - for i, repo := range repos { - matchedTags := repo.Tags + return result, nil + }, + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + fullImageDataList := []mTypes.ImageMeta{} + repos := []mTypes.RepoMeta{{ + Name: "repo1", + Tags: map[string]mTypes.Descriptor{ + "1.0.1": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.2": {Digest: derivedImage.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.3": {Digest: derivedImage.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, + }, + }} + for _, repo := range repos { for tag, descriptor := range repo.Tags { - if !filterFunc(repo, manifestsMeta[descriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMeta, descriptor.Digest) - - continue + if filterFunc(repo, imageDataMap[descriptor.Digest]) { + fullImageDataList = append(fullImageDataList, + convert.GetImageMeta(tag, repo, imageDataMap[descriptor.Digest])) } } - - repos[i].Tags = matchedTags } - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil + return fullImageDataList, nil }, } @@ -3073,79 +1791,74 @@ func TestDerivedImageList(t *testing.T) { mockCve := mocks.CveInfoMock{} Convey("valid derivedImageList, results not affected by pageInput", func() { - images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockSearchDB, &gql_generated.PageInput{}, - mockCve, log.NewLogger("debug", "")) + images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockMetaDB, &gql_generated.PageInput{}, + mockCve, log) So(err, ShouldBeNil) So(images.Results, ShouldNotBeEmpty) So(len(images.Results), ShouldEqual, 2) }) Convey("valid derivedImageList, results affected by pageInput", func() { - limit := 1 - offset := 0 - sortCriteria := gql_generated.SortCriteriaAlphabeticAsc pageInput := gql_generated.PageInput{ - Limit: &limit, - Offset: &offset, - SortBy: &sortCriteria, + Limit: ref(1), + Offset: ref(0), + SortBy: ref(gql_generated.SortCriteriaAlphabeticAsc), } - images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockSearchDB, &pageInput, - mockCve, log.NewLogger("debug", "")) + images, err := derivedImageList(responseContext, "repo1:1.0.1", nil, mockMetaDB, &pageInput, + mockCve, log) So(err, ShouldBeNil) So(images.Results, ShouldNotBeEmpty) - So(len(images.Results), ShouldEqual, limit) + So(len(images.Results), ShouldEqual, 1) }) }) } -func TestBaseImageList(t *testing.T) { +func TestMockedBaseImageList(t *testing.T) { Convey("MetaDB FilterTags error", t, func() { - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - ErrTestError + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, ErrTestError }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{}, ErrTestError + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{}, ErrTestError }, - GetManifestDataFn: func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, ErrTestError + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return map[string]mTypes.ImageData{}, ErrTestError }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) mockCve := mocks.CveInfoMock{} - images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockSearchDB, &gql_generated.PageInput{}, + images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockMetaDB, &gql_generated.PageInput{}, mockCve, log.NewLogger("debug", "")) So(err, ShouldNotBeNil) So(images.Results, ShouldBeEmpty) }) Convey("paginated fail", t, func() { - pageInput := &gql_generated.PageInput{ - Limit: ref(-1), - } + image := CreateDefaultImage() + pageInput := &gql_generated.PageInput{Limit: ref(-1)} responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) _, err := baseImageList(responseContext, "repo1:1.0.2", nil, mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Tags: map[string]mTypes.Descriptor{ "1.0.2": { - Digest: godigest.FromString("str").String(), + Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest, }, }, }, nil }, + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return map[string]mTypes.ImageData{image.DigestStr(): image.AsImageData()}, nil + }, }, pageInput, mocks.CveInfoMock{}, log.NewLogger("debug", "")) So(err, ShouldNotBeNil) @@ -3153,49 +1866,30 @@ func TestBaseImageList(t *testing.T) { //nolint: dupl Convey("MetaDB FilterTags no repo available", t, func() { - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - manifest := ispec.Manifest{} - - manifestBlob, err := json.Marshal(manifest) - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob) + image := CreateDefaultImage() - mockSearchDB := mocks.MetaDBMock{ - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - nil + mockMetaDB := mocks.MetaDBMock{ + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + return []mTypes.ImageMeta{}, nil }, - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Name: "repo1", Tags: map[string]mTypes.Descriptor{ - "1.0.2": {Digest: manifestDigest.String(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.2": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, }, nil }, - GetManifestDataFn: func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - }, nil + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return map[string]mTypes.ImageData{image.DigestStr(): image.AsImageData()}, nil }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) mockCve := mocks.CveInfoMock{} - images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockSearchDB, &gql_generated.PageInput{}, + images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockMetaDB, &gql_generated.PageInput{}, mockCve, log.NewLogger("debug", "")) So(err, ShouldBeNil) So(images.Results, ShouldBeEmpty) @@ -3203,149 +1897,65 @@ func TestBaseImageList(t *testing.T) { //nolint: dupl Convey("base image list working", t, func() { - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - configDigest := godigest.FromBytes(configBlob) - - layers := [][]byte{ - {10, 11, 10, 11}, - {11, 11, 11, 11}, - {10, 10, 10, 11}, - {13, 14, 15, 11}, - } - - manifestBlob, err := json.Marshal(ispec.Manifest{ - Versioned: specs.Versioned{ - SchemaVersion: 2, - }, - Config: ispec.Descriptor{ - MediaType: "application/vnd.oci.image.config.v1+json", - Digest: configDigest, - Size: int64(len(configBlob)), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[0]), - Size: int64(len(layers[0])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[1]), - Size: int64(len(layers[1])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[2]), - Size: int64(len(layers[2])), - }, - }, - }) - So(err, ShouldBeNil) - - derivedManifestBlob, err := json.Marshal(ispec.Manifest{ - Versioned: specs.Versioned{ - SchemaVersion: 2, - }, - Config: ispec.Descriptor{ - MediaType: "application/vnd.oci.image.config.v1+json", - Digest: configDigest, - Size: int64(len(configBlob)), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[0]), - Size: int64(len(layers[0])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[1]), - Size: int64(len(layers[1])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[2]), - Size: int64(len(layers[2])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[3]), - Size: int64(len(layers[3])), - }, - }, - }) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 100, - Signatures: make(mTypes.ManifestSignatures), - }, - "digestTag1.0.2": { - ManifestBlob: derivedManifestBlob, - ConfigBlob: configBlob, - DownloadCount: 100, - Signatures: make(mTypes.ManifestSignatures), - }, + layer1 := []byte{10, 11, 10, 11} + layer2 := []byte{11, 11, 11, 11} + layer3 := []byte{10, 10, 10, 11} + layer4 := []byte{13, 14, 15, 11} + + image := CreateImageWith(). + LayerBlobs([][]byte{ + layer1, + layer2, + layer3, + }).DefaultConfig().Build() + + derivedImage := CreateImageWith(). + LayerBlobs([][]byte{ + layer1, + layer2, + layer3, + layer4, + }).DefaultConfig().Build() + + imageDataMap := map[string]mTypes.ImageData{ + image.DigestStr(): image.AsImageData(), + derivedImage.DigestStr(): derivedImage.AsImageData(), } - derivedManifestDigest := godigest.FromBytes(derivedManifestBlob) - mockSearchDB := mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + mockMetaDB := mocks.MetaDBMock{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Name: "repo1", Tags: map[string]mTypes.Descriptor{ - "1.0.2": {Digest: derivedManifestDigest.String(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.2": {Digest: derivedImage.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, }, nil }, - GetManifestDataFn: func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{ - ManifestBlob: derivedManifestBlob, - ConfigBlob: configBlob, - }, nil + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return imageDataMap, nil }, - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: "digestTag1.0.1", MediaType: ispec.MediaTypeImageManifest}, - "1.0.3": {Digest: "digestTag1.0.1", MediaType: ispec.MediaTypeImageManifest}, - "1.0.2": {Digest: "digestTag1.0.2", MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + imageDataList := []mTypes.ImageMeta{} + repos := []mTypes.RepoMeta{{ + Name: "repo1", + Tags: map[string]mTypes.Descriptor{ + "1.0.1": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.3": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.2": {Digest: derivedImage.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, - } - - for i, repo := range repos { - matchedTags := repo.Tags + }} + for _, repo := range repos { for tag, descriptor := range repo.Tags { - if !filterFunc(repo, manifestsMeta[descriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMeta, descriptor.Digest) - - continue + if filterFunc(repo, imageDataMap[descriptor.Digest]) { + imageDataList = append(imageDataList, + convert.GetImageMeta(tag, repo, imageDataMap[descriptor.Digest])) } } - - repos[i].Tags = matchedTags } - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil + return imageDataList, nil }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, @@ -3354,7 +1964,7 @@ func TestBaseImageList(t *testing.T) { mockCve := mocks.CveInfoMock{} Convey("valid baseImageList, results not affected by pageInput", func() { - images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockSearchDB, + images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockMetaDB, &gql_generated.PageInput{}, mockCve, log.NewLogger("debug", "")) So(err, ShouldBeNil) So(images.Results, ShouldNotBeEmpty) @@ -3374,7 +1984,7 @@ func TestBaseImageList(t *testing.T) { SortBy: &sortCriteria, } - images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockSearchDB, + images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockMetaDB, &pageInput, mockCve, log.NewLogger("debug", "")) So(err, ShouldBeNil) So(images.Results, ShouldNotBeEmpty) @@ -3385,237 +1995,77 @@ func TestBaseImageList(t *testing.T) { //nolint: dupl Convey("filterTags working, no base image list found", t, func() { - configBlob, err := json.Marshal(ispec.Image{ - Config: ispec.ImageConfig{ - Labels: map[string]string{}, - }, - }) - So(err, ShouldBeNil) - - configDigest := godigest.FromBytes(configBlob) - - layers := [][]byte{ - {10, 11, 10, 11}, - {11, 11, 11, 11}, - {10, 10, 10, 11}, - {13, 14, 15, 11}, - } - - manifestBlob, err := json.Marshal(ispec.Manifest{ - Versioned: specs.Versioned{ - SchemaVersion: 2, - }, - Config: ispec.Descriptor{ - MediaType: "application/vnd.oci.image.config.v1+json", - Digest: configDigest, - Size: int64(len(configBlob)), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[0]), - Size: int64(len(layers[0])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[1]), - Size: int64(len(layers[1])), - }, - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[2]), - Size: int64(len(layers[2])), - }, - }, - }) - So(err, ShouldBeNil) - - derivedManifestBlob, err := json.Marshal(ispec.Manifest{ - Versioned: specs.Versioned{ - SchemaVersion: 2, - }, - Config: ispec.Descriptor{ - MediaType: "application/vnd.oci.image.config.v1+json", - Digest: configDigest, - Size: int64(len(configBlob)), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[3]), - Size: int64(len(layers[3])), - }, - }, - }) - So(err, ShouldBeNil) - - manifestsMeta := map[string]mTypes.ManifestMetadata{ - "digestTag1.0.1": { - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 100, - Signatures: make(mTypes.ManifestSignatures), - }, - "digestTag1.0.2": { - ManifestBlob: derivedManifestBlob, - ConfigBlob: configBlob, - DownloadCount: 100, - Signatures: make(mTypes.ManifestSignatures), - }, + layer1 := []byte{10, 11, 10, 11} + layer2 := []byte{11, 11, 11, 11} + layer3 := []byte{10, 10, 10, 11} + layer4 := []byte{13, 14, 15, 11} + + image := CreateImageWith(). + LayerBlobs([][]byte{ + layer1, + layer2, + layer3, + }).DefaultConfig().Build() + + derivedImage := CreateImageWith(). + LayerBlobs([][]byte{ + layer4, + }).DefaultConfig().Build() + + imageDataMap := map[string]mTypes.ImageData{ + image.DigestStr(): image.AsImageData(), + derivedImage.DigestStr(): derivedImage.AsImageData(), } - derivedManifestDigest := godigest.FromBytes(derivedManifestBlob) - mockSearchDB := mocks.MetaDBMock{ - GetRepoMetaFn: func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ + mockMetaDB := mocks.MetaDBMock{ + GetRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + return mTypes.RepoMeta{ Name: "repo1", Tags: map[string]mTypes.Descriptor{ - "1.0.2": {Digest: derivedManifestDigest.String(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.2": {Digest: derivedImage.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, }, nil }, - GetManifestDataFn: func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{ - ManifestBlob: derivedManifestBlob, - ConfigBlob: configBlob, - }, nil + FilterImageDataFn: func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) { + return imageDataMap, nil }, - FilterTagsFn: func(ctx context.Context, - filterFunc mTypes.FilterFunc, - ) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, - ) { - repos := []mTypes.RepoMetadata{ - { - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "1.0.1": {Digest: "digestTag1.0.1", MediaType: ispec.MediaTypeImageManifest}, - "1.0.2": {Digest: "digestTag1.0.2", MediaType: ispec.MediaTypeImageManifest}, - }, - Stars: 100, + FilterTagsFn: func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterFunc, + ) ([]mTypes.ImageMeta, error) { + imageDataList := []mTypes.ImageMeta{} + repos := []mTypes.RepoMeta{{ + Name: "repo1", + Tags: map[string]mTypes.Descriptor{ + "1.0.1": {Digest: image.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, + "1.0.2": {Digest: derivedImage.DigestStr(), MediaType: ispec.MediaTypeImageManifest}, }, - } - - for i, repo := range repos { - matchedTags := repo.Tags + }} + for _, repo := range repos { for tag, descriptor := range repo.Tags { - if !filterFunc(repo, manifestsMeta[descriptor.Digest]) { - delete(matchedTags, tag) - delete(manifestsMeta, descriptor.Digest) - - continue + if filterFunc(repo, imageDataMap[descriptor.Digest]) { + imageDataList = append(imageDataList, + convert.GetImageMeta(tag, repo, imageDataMap[descriptor.Digest])) } } - - repos[i].Tags = matchedTags } - return repos, manifestsMeta, map[string]mTypes.IndexData{}, nil + return imageDataList, nil }, } responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, graphql.DefaultRecover) mockCve := mocks.CveInfoMock{} - images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockSearchDB, &gql_generated.PageInput{}, + images, err := baseImageList(responseContext, "repo1:1.0.2", nil, mockMetaDB, &gql_generated.PageInput{}, mockCve, log.NewLogger("debug", "")) So(err, ShouldBeNil) So(images.Results, ShouldBeEmpty) }) } -func TestExpandedRepoInfo(t *testing.T) { +func TestExpandedRepoInfoErrors(t *testing.T) { log := log.NewLogger("debug", "") - Convey("ExpandedRepoInfo Errors", t, func() { - responseContext := graphql.WithResponseContext(context.Background(), graphql.DefaultErrorPresenter, - graphql.DefaultRecover) - - metaDB := mocks.MetaDBMock{ - GetUserRepoMetaFn: func(ctx context.Context, repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{ - "tagManifest": { - Digest: "errorDigest", - MediaType: ispec.MediaTypeImageManifest, - }, - "tagIndex": { - Digest: "digestIndex", - MediaType: ispec.MediaTypeImageIndex, - }, - "tagGetIndexError": { - Digest: "errorIndexDigest", - MediaType: ispec.MediaTypeImageIndex, - }, - "tagGoodIndexBadManifests": { - Digest: "goodIndexBadManifests", - MediaType: ispec.MediaTypeImageIndex, - }, - "tagGoodIndex1GoodManifest": { - Digest: "goodIndexGoodManifest", - MediaType: ispec.MediaTypeImageIndex, - }, - "tagGoodIndex2GoodManifest": { - Digest: "goodIndexGoodManifest", - MediaType: ispec.MediaTypeImageIndex, - }, - }, - }, nil - }, - GetManifestDataFn: func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - switch manifestDigest { - case "errorDigest": - return mTypes.ManifestData{}, ErrTestError - default: - return mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }, nil - } - }, - GetIndexDataFn: func(indexDigest godigest.Digest) (mTypes.IndexData, error) { - goodIndexBadManifestsBlob, err := json.Marshal(ispec.Index{ - Manifests: []ispec.Descriptor{ - { - Digest: "errorDigest", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - }) - So(err, ShouldBeNil) - - goodIndexGoodManifestBlob, err := json.Marshal(ispec.Index{ - Manifests: []ispec.Descriptor{ - { - Digest: "goodManifest", - MediaType: ispec.MediaTypeImageManifest, - }, - }, - }) - So(err, ShouldBeNil) - - switch indexDigest { - case "errorIndexDigest": - return mTypes.IndexData{}, ErrTestError - case "goodIndexBadManifests": - return mTypes.IndexData{ - IndexBlob: goodIndexBadManifestsBlob, - }, nil - case "goodIndexGoodManifest": - return mTypes.IndexData{ - IndexBlob: goodIndexGoodManifestBlob, - }, nil - default: - return mTypes.IndexData{}, nil - } - }, - } - - _, err := expandedRepoInfo(responseContext, "repo", metaDB, mocks.CveInfoMock{}, log) - So(err, ShouldBeNil) - }) - Convey("Access error", t, func() { userAc := reqCtx.NewUserAccessControl() userAc.SetUsername("user") @@ -3633,54 +2083,18 @@ func TestExpandedRepoInfo(t *testing.T) { }) } -func TestFilterFunctions(t *testing.T) { - Convey("Filter Functions", t, func() { - Convey("FilterByDigest bad manifest blob", func() { - filterFunc := FilterByDigest("digest") - ok := filterFunc( - mTypes.RepoMetadata{}, - mTypes.ManifestMetadata{ - ManifestBlob: []byte("bad blob"), - }, - ) - So(ok, ShouldBeFalse) - }) - - Convey("filterDerivedImages bad manifest blob", func() { - filterFunc := filterDerivedImages(&gql_generated.ImageSummary{}) - ok := filterFunc( - mTypes.RepoMetadata{}, - mTypes.ManifestMetadata{ - ManifestBlob: []byte("bad blob"), - }, - ) - So(ok, ShouldBeFalse) - }) - - Convey("FilterByTagInfo", func() { - fFunc := FilterByTagInfo([]cvemodel.TagInfo{ - { - Descriptor: cvemodel.Descriptor{ - MediaType: ispec.MediaTypeImageIndex, - }, - Manifests: []cvemodel.DescriptorInfo{ - { - Descriptor: cvemodel.Descriptor{ - Digest: godigest.FromString("{}"), - }, - }, - }, - }, - }) +func ref[T any](input T) *T { + ref := input - ok := fFunc(mTypes.RepoMetadata{}, mTypes.ManifestMetadata{ManifestBlob: []byte("{}")}) - So(ok, ShouldBeTrue) - }) - }) + return &ref } -func ref[T any](val T) *T { - ref := val +func getPageInput(limit int, offset int) *gql_generated.PageInput { + sortCriteria := gql_generated.SortCriteriaAlphabeticAsc - return &ref + return &gql_generated.PageInput{ + Limit: &limit, + Offset: &offset, + SortBy: &sortCriteria, + } } diff --git a/pkg/extensions/search/schema.resolvers.go b/pkg/extensions/search/schema.resolvers.go index 4caa350f5d..6d614b6890 100644 --- a/pkg/extensions/search/schema.resolvers.go +++ b/pkg/extensions/search/schema.resolvers.go @@ -138,7 +138,7 @@ func (r *queryResolver) Image(ctx context.Context, image string) (*gql_generated Vulnerabilities: canSkipField(convert.GetPreloads(ctx), "Vulnerabilities"), } - return getProtoImageSummary(ctx, repo, tag, nil, skip, r.metaDB, r.cveInfo, r.log) + return getImageSummary(ctx, repo, tag, nil, skip, r.metaDB, r.cveInfo, r.log) } // Referrers is the resolver for the Referrers field. diff --git a/pkg/extensions/search/search_test.go b/pkg/extensions/search/search_test.go index 8d09b93b32..f6b3b71b14 100644 --- a/pkg/extensions/search/search_test.go +++ b/pkg/extensions/search/search_test.go @@ -13,7 +13,6 @@ import ( "net/url" "os" "path" - "regexp" "strconv" "strings" "testing" @@ -305,7 +304,7 @@ func getMockCveScanner(metaDB mTypes.MetaDB) cveinfo.Scanner { imageDir := repo inputTag := reference - repoMeta, err := metaDB.GetRepoMeta(imageDir) + repoMeta, err := metaDB.GetRepoMeta(context.Background(), imageDir) if err != nil { return false, err } @@ -328,19 +327,12 @@ func getMockCveScanner(metaDB mTypes.MetaDB) cveinfo.Scanner { return false, err } - manifestData, err := metaDB.GetManifestData(manifestDigest) + manifestData, err := metaDB.GetImageData(manifestDigest) if err != nil { return false, err } - var manifestContent ispec.Manifest - - err = json.Unmarshal(manifestData.ManifestBlob, &manifestContent) - if err != nil { - return false, zerr.ErrScanNotSupported - } - - for _, imageLayer := range manifestContent.Layers { + for _, imageLayer := range manifestData.Manifests[0].Manifest.Layers { switch imageLayer.MediaType { case ispec.MediaTypeImageLayerGzip, ispec.MediaTypeImageLayer, string(regTypes.DockerLayer): @@ -1070,24 +1062,17 @@ func TestGetReferrersGQL(t *testing.T) { defer ctlrManager.StopServer() // Upload the index referrer - - targetImg, err := deprecated.GetRandomImage() //nolint:staticcheck - So(err, ShouldBeNil) + targetImg := CreateRandomImage() targetDigest := targetImg.Digest() - err = UploadImage(targetImg, baseURL, "repo", targetDigest.String()) - So(err, ShouldBeNil) - - indexReferrer, err := deprecated.GetRandomMultiarchImage("ref") //nolint:staticcheck + err := UploadImage(targetImg, baseURL, "repo", targetImg.DigestStr()) So(err, ShouldBeNil) artifactType := "com.artifact.art/type" - indexReferrer.Index.ArtifactType = artifactType - indexReferrer.Index.Subject = &ispec.Descriptor{ - MediaType: ispec.MediaTypeImageManifest, - Digest: targetDigest, - } - + indexReferrer := CreateMultiarchWith().RandomImages(2). + ArtifactType(artifactType). + Subject(targetImg.DescriptorRef()). + Build() indexReferrerDigest := indexReferrer.Digest() err = UploadMultiarchImage(indexReferrer, baseURL, "repo", "ref") @@ -4497,67 +4482,24 @@ func TestMetaDBWhenPushingImages(t *testing.T) { ctlrManager.StartAndWait(port) defer ctlrManager.StopServer() - Convey("SetManifestMeta fails", func() { - ctlr.MetaDB = mocks.MetaDBMock{ - SetManifestDataFn: func(manifestDigest godigest.Digest, mm mTypes.ManifestData) error { - return ErrTestError - }, - } - config1, layers1, manifest1, err := deprecated.GetImageComponents(100) //nolint:staticcheck - So(err, ShouldBeNil) - - configBlob, err := json.Marshal(config1) - So(err, ShouldBeNil) - - ctlr.StoreController.DefaultStore = mocks.MockedImageStore{ - NewBlobUploadFn: ctlr.StoreController.DefaultStore.NewBlobUpload, - PutBlobChunkFn: ctlr.StoreController.DefaultStore.PutBlobChunk, - GetBlobContentFn: func(repo string, digest godigest.Digest) ([]byte, error) { - return configBlob, nil - }, - DeleteImageManifestFn: func(repo, reference string, dc bool) error { - return ErrTestError - }, - } - - err = UploadImage( - Image{ - Manifest: manifest1, - Config: config1, - Layers: layers1, - }, baseURL, "repo1", "1.0.1", - ) - So(err, ShouldNotBeNil) - }) - Convey("SetManifestMeta succeeds but SetRepoReference fails", func() { ctlr.MetaDB = mocks.MetaDBMock{ - SetRepoReferenceFn: func(repo, reference string, manifestDigest godigest.Digest, mediaType string) error { + SetRepoReferenceFn: func(repo, reference string, imageData mTypes.ImageData) error { return ErrTestError }, } - config1, layers1, manifest1, err := deprecated.GetImageComponents(100) //nolint:staticcheck - So(err, ShouldBeNil) - - configBlob, err := json.Marshal(config1) - So(err, ShouldBeNil) + image := CreateRandomImage() ctlr.StoreController.DefaultStore = mocks.MockedImageStore{ NewBlobUploadFn: ctlr.StoreController.DefaultStore.NewBlobUpload, PutBlobChunkFn: ctlr.StoreController.DefaultStore.PutBlobChunk, GetBlobContentFn: func(repo string, digest godigest.Digest) ([]byte, error) { - return configBlob, nil + return image.ConfigDescriptor.Data, nil }, } - err = UploadImage( - Image{ - Manifest: manifest1, - Config: config1, - Layers: layers1, - }, baseURL, "repo1", "1.0.1", - ) + err := UploadImage(image, baseURL, "repo1", "1.0.1") So(err, ShouldNotBeNil) }) }) @@ -4592,15 +4534,9 @@ func RunMetaDBIndexTests(baseURL, port string) { Convey("Push test index", func() { const repo = "repo" - multiarchImage, err := deprecated.GetRandomMultiarchImage("tag1") //nolint:staticcheck - So(err, ShouldBeNil) + multiarchImage := CreateRandomMultiarch() - indexBlob, err := json.Marshal(multiarchImage.Index) - So(err, ShouldBeNil) - - indexDigest := godigest.FromBytes(indexBlob) - - err = UploadMultiarchImage(multiarchImage, baseURL, repo, "tag1") + err := UploadMultiarchImage(multiarchImage, baseURL, repo, "tag1") So(err, ShouldBeNil) query := ` @@ -4636,7 +4572,7 @@ func RunMetaDBIndexTests(baseURL, port string) { responseImage := responseImages[0] So(len(responseImage.Manifests), ShouldEqual, 3) - err = signature.SignImageUsingCosign(fmt.Sprintf("repo@%s", indexDigest), port) + err = signature.SignImageUsingCosign(fmt.Sprintf("repo@%s", multiarchImage.DigestStr()), port) So(err, ShouldBeNil) resp, err = resty.R().Get(baseURL + graphqlQueryPrefix + "?query=" + url.QueryEscape(query)) @@ -4656,7 +4592,7 @@ func RunMetaDBIndexTests(baseURL, port string) { So(responseImage.IsSigned, ShouldBeTrue) // remove signature - cosignTag := "sha256-" + indexDigest.Encoded() + ".sig" + cosignTag := "sha256-" + multiarchImage.Digest().Encoded() + ".sig" _, err = resty.R().Delete(baseURL + "/v2/" + "repo" + "/manifests/" + cosignTag) So(err, ShouldBeNil) @@ -5411,9 +5347,7 @@ func TestMetaDBWhenDeletingImages(t *testing.T) { for _, manifest := range indexContent.Manifests { tag := manifest.Annotations[ispec.AnnotationRefName] - cosignTagRule := regexp.MustCompile(`sha256\-.+\.sig`) - - if cosignTagRule.MatchString(tag) { + if zcommon.IsCosignTag(tag) { signatureTag = tag } } @@ -5687,7 +5621,10 @@ func TestMetaDBWhenDeletingImages(t *testing.T) { } ctlr.MetaDB = mocks.MetaDBMock{ - RemoveRepoReferenceFn: func(repo, reference string, manifestDigest godigest.Digest) error { return ErrTestError }, + RemoveRepoReferenceFn: func(repo, reference string, manifestDigest godigest.Digest, + ) error { + return ErrTestError + }, } resp, err = resty.R().Delete(baseURL + "/v2/" + "repo1" + "/manifests/" + @@ -6440,16 +6377,6 @@ func TestUploadingArtifactsWithDifferentMediaType(t *testing.T) { err = UploadImage(defaultImage, baseURL, "repo", "default-image") So(err, ShouldBeNil) - manifestData, err := ctlr.MetaDB.GetManifestData(imageWithIncompatibleConfig.ManifestDescriptor.Digest) - So(err, ShouldBeNil) - So(manifestData.ConfigBlob, ShouldEqual, imageWithIncompatibleConfig.ConfigDescriptor.Data) - So(manifestData.ManifestBlob, ShouldEqual, imageWithIncompatibleConfig.ManifestDescriptor.Data) - - manifestData, err = ctlr.MetaDB.GetManifestData(defaultImage.ManifestDescriptor.Digest) - So(err, ShouldBeNil) - So(manifestData.ConfigBlob, ShouldEqual, defaultImage.ConfigDescriptor.Data) - So(manifestData.ManifestBlob, ShouldEqual, defaultImage.ManifestDescriptor.Data) - query := ` { GlobalSearch(query:"repo:incompatible-image"){ diff --git a/pkg/extensions/search/userprefs_test.go b/pkg/extensions/search/userprefs_test.go index 7d0fda85c5..26e6dbb930 100644 --- a/pkg/extensions/search/userprefs_test.go +++ b/pkg/extensions/search/userprefs_test.go @@ -19,13 +19,11 @@ import ( "zotregistry.io/zot/pkg/api/constants" "zotregistry.io/zot/pkg/common" extconf "zotregistry.io/zot/pkg/extensions/config" - "zotregistry.io/zot/pkg/extensions/monitoring" "zotregistry.io/zot/pkg/log" - "zotregistry.io/zot/pkg/storage" - "zotregistry.io/zot/pkg/storage/local" . "zotregistry.io/zot/pkg/test/common" "zotregistry.io/zot/pkg/test/deprecated" . "zotregistry.io/zot/pkg/test/image-utils" + . "zotregistry.io/zot/pkg/test/oci-utils" ) //nolint:dupl @@ -539,25 +537,15 @@ func TestChangingRepoState(t *testing.T) { ctlr := api.NewController(conf) - img, err := deprecated.GetRandomImage() //nolint:staticcheck - if err != nil { - t.FailNow() - } + img := CreateRandomImage() + storeCtlr := GetDefaultStoreController(conf.Storage.RootDirectory, log.NewLogger("debug", "")) - // ------ Create the test repos - defaultStore := local.NewImageStore(conf.Storage.RootDirectory, false, false, - log.NewLogger("debug", ""), monitoring.NewMetricsServer(false, log.NewLogger("debug", "")), nil, nil) - - err = WriteImageToFileSystem(img, accesibleRepo, "tag", storage.StoreController{ - DefaultStore: defaultStore, - }) + err := WriteImageToFileSystem(img, accesibleRepo, "tag", storeCtlr) if err != nil { t.FailNow() } - err = WriteImageToFileSystem(img, forbiddenRepo, "tag", storage.StoreController{ - DefaultStore: defaultStore, - }) + err = WriteImageToFileSystem(img, forbiddenRepo, "tag", storeCtlr) if err != nil { t.FailNow() } diff --git a/pkg/extensions/sync/references/cosign.go b/pkg/extensions/sync/references/cosign.go index 1b842b6e69..1903e85ac1 100644 --- a/pkg/extensions/sync/references/cosign.go +++ b/pkg/extensions/sync/references/cosign.go @@ -153,21 +153,9 @@ func (ref CosignReference) SyncReferences(ctx context.Context, localRepo, remote ref.log.Debug().Str("repository", localRepo).Str("subject", subjectDigestStr). Msg("metaDB: trying to sync cosign reference for image") - isSig, sigType, signedManifestDig, err := storage.CheckIsImageSignature(localRepo, manifestBuf, - cosignTag) - if err != nil { - return refsDigests, fmt.Errorf("failed to check if cosign reference '%s@%s' is a signature: %w", localRepo, - cosignTag, err) - } - - if isSig { - err = addSigToMeta(ref.metaDB, localRepo, sigType, cosignTag, signedManifestDig, referenceDigest, - manifestBuf, imageStore, ref.log) - } else { - err = meta.SetImageMetaFromInput(localRepo, cosignTag, ispec.MediaTypeImageManifest, - referenceDigest, manifestBuf, ref.storeController.GetImageStore(localRepo), - ref.metaDB, ref.log) - } + err = meta.SetImageMetaFromInput(localRepo, cosignTag, ispec.MediaTypeImageManifest, + referenceDigest, manifestBuf, ref.storeController.GetImageStore(localRepo), + ref.metaDB, ref.log) if err != nil { return refsDigests, fmt.Errorf("failed to set metadata for cosign reference in '%s@%s': %w", diff --git a/pkg/extensions/sync/references/oci.go b/pkg/extensions/sync/references/oci.go index beac30311e..6519b05eb0 100644 --- a/pkg/extensions/sync/references/oci.go +++ b/pkg/extensions/sync/references/oci.go @@ -137,22 +137,9 @@ func (ref OciReferences) SyncReferences(ctx context.Context, localRepo, remoteRe ref.log.Debug().Str("repository", localRepo).Str("subject", subjectDigestStr). Msg("metaDB: trying to add oci references for image") - isSig, sigType, signedManifestDig, err := storage.CheckIsImageSignature(localRepo, referenceBuf, - referrer.Digest.String()) - if err != nil { - return refsDigests, fmt.Errorf("failed to check if oci reference '%s@%s' is a signature: %w", localRepo, - referrer.Digest.String(), err) - } - - if isSig { - err = addSigToMeta(ref.metaDB, localRepo, sigType, referrer.Digest.String(), signedManifestDig, referenceDigest, - referenceBuf, imageStore, ref.log) - } else { - err = meta.SetImageMetaFromInput(localRepo, referenceDigest.String(), referrer.MediaType, - referenceDigest, referenceBuf, ref.storeController.GetImageStore(localRepo), - ref.metaDB, ref.log) - } - + err = meta.SetImageMetaFromInput(localRepo, referenceDigest.String(), referrer.MediaType, + referenceDigest, referenceBuf, ref.storeController.GetImageStore(localRepo), + ref.metaDB, ref.log) if err != nil { return refsDigests, fmt.Errorf("failed to set metadata for oci reference in '%s@%s': %w", localRepo, subjectDigestStr, err) diff --git a/pkg/extensions/sync/references/referrers_tag.go b/pkg/extensions/sync/references/referrers_tag.go index 4628add388..2633e15b36 100644 --- a/pkg/extensions/sync/references/referrers_tag.go +++ b/pkg/extensions/sync/references/referrers_tag.go @@ -113,22 +113,9 @@ func (ref TagReferences) SyncReferences(ctx context.Context, localRepo, remoteRe ref.log.Debug().Str("repository", localRepo).Str("subject", subjectDigestStr). Msg("metaDB: trying to add oci references for image") - isSig, sigType, signedManifestDig, err := storage.CheckIsImageSignature(localRepo, referenceBuf, - referrer.Digest.String()) - if err != nil { - return refsDigests, fmt.Errorf("failed to check if oci reference '%s@%s' is a signature: %w", localRepo, - referrer.Digest.String(), err) - } - - if isSig { - err = addSigToMeta(ref.metaDB, localRepo, sigType, referrer.Digest.String(), signedManifestDig, referenceDigest, - referenceBuf, imageStore, ref.log) - } else { - err = meta.SetImageMetaFromInput(localRepo, referenceDigest.String(), referrer.MediaType, - referenceDigest, referenceBuf, ref.storeController.GetImageStore(localRepo), - ref.metaDB, ref.log) - } - + err = meta.SetImageMetaFromInput(localRepo, referenceDigest.String(), referrer.MediaType, + referenceDigest, referenceBuf, ref.storeController.GetImageStore(localRepo), + ref.metaDB, ref.log) if err != nil { return refsDigests, fmt.Errorf("failed to set metadata for oci reference in '%s@%s': %w", localRepo, subjectDigestStr, err) diff --git a/pkg/extensions/sync/sync_internal_test.go b/pkg/extensions/sync/sync_internal_test.go index 9b6c2968a8..67d54ecc69 100644 --- a/pkg/extensions/sync/sync_internal_test.go +++ b/pkg/extensions/sync/sync_internal_test.go @@ -27,6 +27,7 @@ import ( "zotregistry.io/zot/pkg/extensions/monitoring" client "zotregistry.io/zot/pkg/extensions/sync/httpclient" "zotregistry.io/zot/pkg/log" + mTypes "zotregistry.io/zot/pkg/meta/types" "zotregistry.io/zot/pkg/storage" "zotregistry.io/zot/pkg/storage/cache" storageConstants "zotregistry.io/zot/pkg/storage/constants" @@ -336,8 +337,8 @@ func TestLocalRegistry(t *testing.T) { Convey("trigger metaDB error on index manifest in CommitImage()", func() { registry := NewLocalRegistry(storage.StoreController{DefaultStore: syncImgStore}, mocks.MetaDBMock{ - SetRepoReferenceFn: func(repo, Reference string, manifestDigest godigest.Digest, mediaType string) error { - if Reference == "1.0" { + SetRepoReferenceFn: func(repo string, reference string, imageData mTypes.ImageData) error { + if reference == "1.0" { return zerr.ErrRepoMetaNotFound } @@ -351,7 +352,7 @@ func TestLocalRegistry(t *testing.T) { Convey("trigger metaDB error on image manifest in CommitImage()", func() { registry := NewLocalRegistry(storage.StoreController{DefaultStore: syncImgStore}, mocks.MetaDBMock{ - SetRepoReferenceFn: func(repo, Reference string, manifestDigest godigest.Digest, mediaType string) error { + SetRepoReferenceFn: func(repo, reference string, imageData mTypes.ImageData) error { return zerr.ErrRepoMetaNotFound }, }, log) diff --git a/pkg/extensions/sync/sync_test.go b/pkg/extensions/sync/sync_test.go index b0519c5560..bdfef94699 100644 --- a/pkg/extensions/sync/sync_test.go +++ b/pkg/extensions/sync/sync_test.go @@ -875,9 +875,7 @@ func TestOnDemand(t *testing.T) { return nil }, - SetRepoReferenceFn: func(repo, reference string, manifestDigest godigest.Digest, - mediaType string, - ) error { + SetRepoReferenceFn: func(repo, reference string, imageData mTypes.ImageData) error { if strings.HasPrefix(reference, "sha256-") && (strings.HasSuffix(reference, remote.SignatureTagSuffix) || strings.HasSuffix(reference, remote.SBOMTagSuffix)) || @@ -1017,8 +1015,8 @@ func TestOnDemand(t *testing.T) { // metadb fails for syncReferrersTag" dctlr.MetaDB = mocks.MetaDBMock{ - SetManifestDataFn: func(manifestDigest godigest.Digest, mm mTypes.ManifestData) error { - if manifestDigest.String() == ociRefImage.ManifestDescriptor.Digest.String() { + SetRepoReferenceFn: func(repo, reference string, imageData mTypes.ImageData) error { + if imageData.Digest.String() == ociRefImage.ManifestDescriptor.Digest.String() { return sync.ErrTestError } @@ -4666,7 +4664,7 @@ func TestSyncedSignaturesMetaDB(t *testing.T) { So(err, ShouldBeNil) So(resp.StatusCode(), ShouldEqual, http.StatusOK) - repoMeta, err := dctlr.MetaDB.GetRepoMeta(repoName) + repoMeta, err := dctlr.MetaDB.GetRepoMeta(context.Background(), repoName) So(err, ShouldBeNil) So(repoMeta.Tags, ShouldContainKey, tag) So(len(repoMeta.Tags), ShouldEqual, 1) diff --git a/pkg/meta/boltdb/boltdb.go b/pkg/meta/boltdb/boltdb.go index c633e05990..989d46b97a 100644 --- a/pkg/meta/boltdb/boltdb.go +++ b/pkg/meta/boltdb/boltdb.go @@ -11,12 +11,16 @@ import ( godigest "github.com/opencontainers/go-digest" ispec "github.com/opencontainers/image-spec/specs-go/v1" "go.etcd.io/bbolt" + "google.golang.org/protobuf/proto" + "google.golang.org/protobuf/types/known/timestamppb" zerr "zotregistry.io/zot/errors" "zotregistry.io/zot/pkg/api/constants" zcommon "zotregistry.io/zot/pkg/common" "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta/common" + mConvert "zotregistry.io/zot/pkg/meta/convert" + "zotregistry.io/zot/pkg/meta/proto_go" mTypes "zotregistry.io/zot/pkg/meta/types" "zotregistry.io/zot/pkg/meta/version" reqCtx "zotregistry.io/zot/pkg/requestcontext" @@ -41,21 +45,6 @@ func New(boltDB *bbolt.DB, log log.Logger) (*BoltDB, error) { return err } - _, err = transaction.CreateBucketIfNotExists([]byte(ManifestDataBucket)) - if err != nil { - return err - } - - _, err = transaction.CreateBucketIfNotExists([]byte(IndexDataBucket)) - if err != nil { - return err - } - - _, err = transaction.CreateBucketIfNotExists([]byte(RepoMetadataBucket)) - if err != nil { - return err - } - _, err = transaction.CreateBucketIfNotExists([]byte(UserDataBucket)) if err != nil { return err @@ -66,18 +55,17 @@ func New(boltDB *bbolt.DB, log log.Logger) (*BoltDB, error) { return err } - // TODO: PROTO STUFF - _, err = transaction.CreateBucketIfNotExists([]byte(ProtoImageDataBuck)) + _, err = transaction.CreateBucketIfNotExists([]byte(ImageDataBuck)) if err != nil { return err } - _, err = transaction.CreateBucketIfNotExists([]byte(ProtoRepoMetaBuck)) + _, err = transaction.CreateBucketIfNotExists([]byte(RepoMetaBuck)) if err != nil { return err } - _, err = transaction.CreateBucketIfNotExists([]byte(ProtoRepoBlobsBuck)) + _, err = transaction.CreateBucketIfNotExists([]byte(RepoBlobsBuck)) if err != nil { return err } @@ -96,855 +84,716 @@ func New(boltDB *bbolt.DB, log log.Logger) (*BoltDB, error) { }, nil } -func (bdw *BoltDB) ImageTrustStore() mTypes.ImageTrustStore { - return bdw.imgTrustStore -} +func (bdw *BoltDB) SetImageData(digest godigest.Digest, imageData mTypes.ImageData) error { + err := bdw.DB.Update(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(ImageDataBuck)) -func (bdw *BoltDB) SetImageTrustStore(imgTrustStore mTypes.ImageTrustStore) { - bdw.imgTrustStore = imgTrustStore -} + protoImageData := &proto_go.ImageData{} -func (bdw *BoltDB) SetManifestData(manifestDigest godigest.Digest, manifestData mTypes.ManifestData) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ManifestDataBucket)) + switch imageData.MediaType { + case ispec.MediaTypeImageManifest: + manifest := imageData.Manifests[0] - mdBlob, err := json.Marshal(manifestData) - if err != nil { - return fmt.Errorf("metadb: error while calculating blob for manifest with digest %s %w", manifestDigest, err) + protoImageData = mConvert.GetProtoImageManifestData(manifest.Manifest, manifest.Config, + manifest.Size, manifest.Digest.String()) + case ispec.MediaTypeImageIndex: + protoImageData = mConvert.GetProtoImageIndexData(*imageData.Index, imageData.Size, imageData.Digest.String()) } - err = buck.Put([]byte(manifestDigest), mdBlob) + pImageDataBlob, err := proto.Marshal(protoImageData) if err != nil { - return fmt.Errorf("metadb: error while setting manifest data with for digest %s %w", manifestDigest, err) - } - - return nil - }) - - return err -} - -func (bdw *BoltDB) GetManifestData(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - var manifestData mTypes.ManifestData - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ManifestDataBucket)) - - mdBlob := buck.Get([]byte(manifestDigest)) - - if len(mdBlob) == 0 { - return zerr.ErrManifestDataNotFound + return fmt.Errorf("metadb: error while calculating blob for manifest with digest %s %w", digest, err) } - err := json.Unmarshal(mdBlob, &manifestData) + err = buck.Put([]byte(digest), pImageDataBlob) if err != nil { - return fmt.Errorf("metadb: error while unmashaling manifest meta for digest %s %w", manifestDigest, err) + return fmt.Errorf("metadb: error while setting manifest data with for digest %s %w", digest, err) } return nil }) - return manifestData, err + return err } -func (bdw *BoltDB) SetManifestMeta(repo string, manifestDigest godigest.Digest, manifestMeta mTypes.ManifestMetadata, +func (bdw *BoltDB) SetRepoReference(repo string, reference string, imageData mTypes.ImageData, ) error { + if err := common.ValidateRepoReferenceInput(repo, reference, imageData.Digest); err != nil { + return err + } + err := bdw.DB.Update(func(tx *bbolt.Tx) error { - dataBuck := tx.Bucket([]byte(ManifestDataBucket)) - repoBuck := tx.Bucket([]byte(RepoMetadataBucket)) + repoBuck := tx.Bucket([]byte(RepoMetaBuck)) + repoBlobsBuck := tx.Bucket([]byte(RepoBlobsBuck)) + imageBuck := tx.Bucket([]byte(ImageDataBuck)) - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - } + // 1. Add image data to db if needed - repoMetaBlob := repoBuck.Get([]byte(repo)) - if len(repoMetaBlob) > 0 { - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - } + protoImageData := mConvert.GetProtoImageData(imageData) - mdBlob, err := json.Marshal(mTypes.ManifestData{ - ManifestBlob: manifestMeta.ManifestBlob, - ConfigBlob: manifestMeta.ConfigBlob, - }) + imageDataBlob, err := proto.Marshal(protoImageData) if err != nil { - return fmt.Errorf("metadb: error while calculating blob for manifest with digest %s %w", manifestDigest, err) - } - - err = dataBuck.Put([]byte(manifestDigest), mdBlob) - if err != nil { - return fmt.Errorf("metadb: error while setting manifest meta with for digest %s %w", manifestDigest, err) + return err } - updatedRepoMeta := common.UpdateManifestMeta(repoMeta, manifestDigest, manifestMeta) - - updatedRepoMetaBlob, err := json.Marshal(updatedRepoMeta) + err = imageBuck.Put([]byte(imageData.Digest), imageDataBlob) if err != nil { - return fmt.Errorf("metadb: error while calculating blob for updated repo meta '%s' %w", repo, err) - } - - return repoBuck.Put([]byte(repo), updatedRepoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) GetManifestMeta(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - var manifestMetadata mTypes.ManifestMetadata - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - dataBuck := tx.Bucket([]byte(ManifestDataBucket)) - repoBuck := tx.Bucket([]byte(RepoMetadataBucket)) - - mdBlob := dataBuck.Get([]byte(manifestDigest)) - - if len(mdBlob) == 0 { - return zerr.ErrManifestMetaNotFound + return err } - var manifestData mTypes.ManifestData + repoMetaBlob := repoBuck.Get([]byte(repo)) - err := json.Unmarshal(mdBlob, &manifestData) - if err != nil { - return fmt.Errorf("metadb: error while unmashaling manifest meta for digest %s %w", manifestDigest, err) + protoRepoMeta := &proto_go.RepoMeta{ + Name: repo, + Tags: map[string]*proto_go.TagDescriptor{"": {}}, // This is done so Protobuf can initialize a non-nil map + Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, + Signatures: map[string]*proto_go.ManifestSignatures{"": {Map: map[string]*proto_go.SignaturesInfo{"": {}}}}, + Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, } - var repoMeta mTypes.RepoMetadata - - repoMetaBlob := repoBuck.Get([]byte(repo)) if len(repoMetaBlob) > 0 { - err = json.Unmarshal(repoMetaBlob, &repoMeta) + err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) if err != nil { - return fmt.Errorf("metadb: error while unmashaling manifest meta for digest %s %w", manifestDigest, err) + return err } } - manifestMetadata.ManifestBlob = manifestData.ManifestBlob - manifestMetadata.ConfigBlob = manifestData.ConfigBlob - manifestMetadata.DownloadCount = repoMeta.Statistics[manifestDigest.String()].DownloadCount + // 2. Referrers + if subject := mConvert.GetImageSubject(protoImageData); subject != nil { + refInfo := &proto_go.ReferrersInfo{} + if protoRepoMeta.Referrers[subject.Digest.String()] != nil { + refInfo = protoRepoMeta.Referrers[subject.Digest.String()] + } - manifestMetadata.Signatures = mTypes.ManifestSignatures{} - if repoMeta.Signatures[manifestDigest.String()] != nil { - manifestMetadata.Signatures = repoMeta.Signatures[manifestDigest.String()] - } + foundReferrer := false - return nil - }) + for i := range refInfo.List { + if refInfo.List[i].Digest == mConvert.GetImageDigestStr(protoImageData) { + foundReferrer = true + refInfo.List[i].Count += 1 - return manifestMetadata, err -} + break + } + } -func (bdw *BoltDB) SetIndexData(indexDigest godigest.Digest, indexMetadata mTypes.IndexData) error { - // we make the assumption that the oci layout is consistent and all manifests refferenced inside the - // index are present - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(IndexDataBucket)) + if !foundReferrer { + refInfo.List = append(refInfo.List, &proto_go.ReferrerInfo{ + Count: 1, + MediaType: protoImageData.MediaType, + Digest: mConvert.GetImageDigestStr(protoImageData), + ArtifactType: mConvert.GetImageArtifactType(protoImageData), + Size: mConvert.GetImageManifestSize(protoImageData), + Annotations: mConvert.GetImageAnnotations(protoImageData), + }) + } - imBlob, err := json.Marshal(indexMetadata) - if err != nil { - return fmt.Errorf("metadb: error while calculating blob for manifest with digest %s %w", indexDigest, err) + protoRepoMeta.Referrers[subject.Digest.String()] = refInfo } - err = buck.Put([]byte(indexDigest), imBlob) - if err != nil { - return fmt.Errorf("metadb: error while setting manifest meta with for digest %s %w", indexDigest, err) + // 3. Update tag + if !common.ReferenceIsDigest(reference) { + protoRepoMeta.Tags[reference] = &proto_go.TagDescriptor{ + Digest: imageData.Digest.String(), + MediaType: imageData.MediaType, + } } - return nil - }) - - return err -} - -func (bdw *BoltDB) GetIndexData(indexDigest godigest.Digest) (mTypes.IndexData, error) { - var indexMetadata mTypes.IndexData - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(IndexDataBucket)) - - mmBlob := buck.Get([]byte(indexDigest)) - - if len(mmBlob) == 0 { - return zerr.ErrManifestMetaNotFound + if _, ok := protoRepoMeta.Statistics[imageData.Digest.String()]; !ok { + protoRepoMeta.Statistics[imageData.Digest.String()] = &proto_go.DescriptorStatistics{DownloadCount: 0} } - err := json.Unmarshal(mmBlob, &indexMetadata) - if err != nil { - return fmt.Errorf("metadb: error while unmashaling manifest meta for digest %s %w", indexDigest, err) + if _, ok := protoRepoMeta.Signatures[imageData.Digest.String()]; !ok { + protoRepoMeta.Signatures[imageData.Digest.String()] = &proto_go.ManifestSignatures{ + Map: map[string]*proto_go.SignaturesInfo{"": {}}, + } } - return nil - }) - - return indexMetadata, err -} - -func (bdw BoltDB) SetReferrer(repo string, referredDigest godigest.Digest, referrer mTypes.ReferrerInfo) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + if _, ok := protoRepoMeta.Referrers[imageData.Digest.String()]; !ok { + protoRepoMeta.Referrers[imageData.Digest.String()] = &proto_go.ReferrersInfo{ + List: []*proto_go.ReferrerInfo{}, + } + } - repoMetaBlob := buck.Get([]byte(repo)) + // 4. Blobs + repoBlobsBytes := repoBlobsBuck.Get([]byte(protoRepoMeta.Name)) - // object not found - if len(repoMetaBlob) == 0 { - var err error + repoBlobs := &proto_go.RepoBlobs{} - // create a new object - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{ - referredDigest.String(): { - referrer, - }, - }, - } - - repoMetaBlob, err = json.Marshal(repoMeta) + if repoBlobsBytes == nil { + repoBlobs.Blobs = map[string]*proto_go.BlobInfo{} + } else { + err := proto.Unmarshal(repoBlobsBytes, repoBlobs) if err != nil { return err } - - return buck.Put([]byte(repo), repoMetaBlob) } - var repoMeta mTypes.RepoMetadata - err := json.Unmarshal(repoMetaBlob, &repoMeta) + protoRepoMeta, repoBlobs, err = common.AddImageDataToRepoMeta(protoRepoMeta, repoBlobs, reference, imageData) if err != nil { return err } - referrers := repoMeta.Referrers[referredDigest.String()] - - for i := range referrers { - if referrers[i].Digest == referrer.Digest { - return nil - } - } - - referrers = append(referrers, referrer) - - repoMeta.Referrers[referredDigest.String()] = referrers - - repoMetaBlob, err = json.Marshal(repoMeta) + repoBlobsBytes, err = proto.Marshal(repoBlobs) if err != nil { return err } - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw BoltDB) DeleteReferrer(repo string, referredDigest godigest.Digest, - referrerDigest godigest.Digest, -) error { - return bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) - - repoMetaBlob := buck.Get([]byte(repo)) - - if len(repoMetaBlob) == 0 { - return zerr.ErrRepoMetaNotFound - } - - var repoMeta mTypes.RepoMetadata - - err := json.Unmarshal(repoMetaBlob, &repoMeta) + err = repoBlobsBuck.Put([]byte(protoRepoMeta.Name), repoBlobsBytes) if err != nil { return err } - referrers := repoMeta.Referrers[referredDigest.String()] - - for i := range referrers { - if referrers[i].Digest == referrerDigest.String() { - referrers = append(referrers[:i], referrers[i+1:]...) - - break - } - } - - repoMeta.Referrers[referredDigest.String()] = referrers - - repoMetaBlob, err = json.Marshal(repoMeta) + repoMetaBlob, err = proto.Marshal(protoRepoMeta) if err != nil { return err } - return buck.Put([]byte(repo), repoMetaBlob) + return repoBuck.Put([]byte(repo), repoMetaBlob) }) + + return err } -func (bdw BoltDB) GetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string, -) ([]mTypes.ReferrerInfo, error) { - referrersInfoResult := []mTypes.ReferrerInfo{} +func (bdw *BoltDB) FilterImageData(ctx context.Context, digests []string, +) (map[string]mTypes.ImageData, error) { + imageDataMap := map[string]mTypes.ImageData{} - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + err := bdw.DB.View(func(transaction *bbolt.Tx) error { + imageBuck := transaction.Bucket([]byte(ImageDataBuck)) - repoMetaBlob := buck.Get([]byte(repo)) - if len(repoMetaBlob) == 0 { - return zerr.ErrRepoMetaNotFound - } + for _, digest := range digests { + protoImageData, err := fetchProtoImageData(imageBuck, digest) + if err != nil { + return err + } - var repoMeta mTypes.RepoMetadata + if protoImageData.MediaType == ispec.MediaTypeImageIndex { + manifestDataList := make([]*proto_go.ManifestData, 0, len(protoImageData.Index.Index.Manifests)) - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } + for _, manifest := range protoImageData.Index.Index.Manifests { + imageManifestData, err := fetchProtoImageData(imageBuck, manifest.Digest) + if err != nil { + return err + } - referrersInfo := repoMeta.Referrers[referredDigest.String()] + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) + } - for i := range referrersInfo { - if !common.MatchesArtifactTypes(referrersInfo[i].ArtifactType, artifactTypes) { - continue + protoImageData.Manifests = manifestDataList } - referrersInfoResult = append(referrersInfoResult, referrersInfo[i]) + imageDataMap[digest] = mConvert.GetImageData(protoImageData) } return nil }) - return referrersInfoResult, err + return imageDataMap, err } -/* - RemoveRepoReference removes the tag from RepoMetadata if the reference is a tag, - -it also removes its corresponding digest from Statistics, Signatures and Referrers if there are no tags -pointing to it. -If the reference is a digest then it will remove the digest from Statistics, Signatures and Referrers only -if there are no tags pointing to the digest, otherwise it's noop. -*/ -func (bdw *BoltDB) RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) +func (bdw *BoltDB) SearchRepos(ctx context.Context, searchText string, +) ([]mTypes.RepoMeta, error) { + repos := []mTypes.RepoMeta{} - repoMetaBlob := buck.Get([]byte(repo)) + err := bdw.DB.View(func(transaction *bbolt.Tx) error { + var ( + repoBuck = transaction.Bucket([]byte(RepoMetaBuck)) + userBookmarks = getUserBookmarks(ctx, transaction) + userStars = getUserStars(ctx, transaction) + ) - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - } + cursor := repoBuck.Cursor() - // object not found - if len(repoMetaBlob) > 0 { - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err + for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { + if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { + continue } - } - if !common.ReferenceIsDigest(reference) { - delete(repoMeta.Tags, reference) - } else { - // remove all tags pointing to this digest - for tag, desc := range repoMeta.Tags { - if desc.Digest == reference { - delete(repoMeta.Tags, tag) - } + rank := common.RankRepoName(searchText, string(repoName)) + if rank == -1 { + continue } - } - /* try to find at least one tag pointing to manifestDigest - if not found then we can also remove everything related to this digest */ - var foundTag bool - for _, desc := range repoMeta.Tags { - if desc.Digest == manifestDigest.String() { - foundTag = true + var protoRepoMeta proto_go.RepoMeta + + err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) + if err != nil { + return err } - } - if !foundTag { - delete(repoMeta.Statistics, manifestDigest.String()) - delete(repoMeta.Signatures, manifestDigest.String()) - delete(repoMeta.Referrers, manifestDigest.String()) - } + delete(protoRepoMeta.Tags, "") - repoMetaBlob, err := json.Marshal(repoMeta) - if err != nil { - return err + protoRepoMeta.Rank = int32(rank) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) + + repos = append(repos, mConvert.GetRepoMeta(&protoRepoMeta)) } - return buck.Put([]byte(repo), repoMetaBlob) + return nil }) - return err + return repos, err } -func (bdw *BoltDB) SetRepoReference(repo string, reference string, manifestDigest godigest.Digest, - mediaType string, -) error { - if err := common.ValidateRepoReferenceInput(repo, reference, manifestDigest); err != nil { - return err +func fetchProtoImageData(imageBuck *bbolt.Bucket, digest string) (*proto_go.ImageData, error) { + imageDataBlob := imageBuck.Get([]byte(digest)) + + if len(imageDataBlob) == 0 { + return nil, zerr.ErrImageDataNotFound } - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + imageData := proto_go.ImageData{} - repoMetaBlob := buck.Get([]byte(repo)) + err := proto.Unmarshal(imageDataBlob, &imageData) + if err != nil { + return nil, err + } - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - } + return &imageData, nil +} - // object not found - if len(repoMetaBlob) > 0 { - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - } +func (bdw *BoltDB) SearchTags(ctx context.Context, searchText string, +) ([]mTypes.ImageMeta, error) { + images := []mTypes.ImageMeta{} - if !common.ReferenceIsDigest(reference) { - repoMeta.Tags[reference] = mTypes.Descriptor{ - Digest: manifestDigest.String(), - MediaType: mediaType, - } - } + searchedRepo, searchedTag, err := common.GetRepoTag(searchText) + if err != nil { + return []mTypes.ImageMeta{}, + fmt.Errorf("metadb: error while parsing search text, invalid format %w", err) + } - if _, ok := repoMeta.Statistics[manifestDigest.String()]; !ok { - repoMeta.Statistics[manifestDigest.String()] = mTypes.DescriptorStatistics{DownloadCount: 0} - } + err = bdw.DB.View(func(transaction *bbolt.Tx) error { + var ( + repoBuck = transaction.Bucket([]byte(RepoMetaBuck)) + imageBuck = transaction.Bucket([]byte(ImageDataBuck)) + userBookmarks = getUserBookmarks(ctx, transaction) + userStars = getUserStars(ctx, transaction) + ) - if _, ok := repoMeta.Signatures[manifestDigest.String()]; !ok { - repoMeta.Signatures[manifestDigest.String()] = mTypes.ManifestSignatures{} - } + repoName, repoMetaBlob := repoBuck.Cursor().Seek([]byte(searchedRepo)) - if _, ok := repoMeta.Referrers[manifestDigest.String()]; !ok { - repoMeta.Referrers[manifestDigest.String()] = []mTypes.ReferrerInfo{} + if string(repoName) != searchedRepo { + return nil } - repoMetaBlob, err := json.Marshal(repoMeta) - if err != nil { + if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { return err } - return buck.Put([]byte(repo), repoMetaBlob) - }) + protoRepoMeta := &proto_go.RepoMeta{} - return err -} - -func (bdw *BoltDB) GetRepoMeta(repo string) (mTypes.RepoMetadata, error) { - var repoMeta mTypes.RepoMetadata - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) - - repoMetaBlob := buck.Get([]byte(repo)) - - // object not found - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - // object found - err := json.Unmarshal(repoMetaBlob, &repoMeta) + err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) if err != nil { return err } - return nil - }) + delete(protoRepoMeta.Tags, "") - return repoMeta, err -} + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) -func (bdw *BoltDB) GetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMetadata, error) { - var repoMeta mTypes.RepoMetadata + for tag, descriptor := range protoRepoMeta.Tags { + if !strings.HasPrefix(tag, searchedTag) || tag == "" { + continue + } - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) - userBookmarks := getUserBookmarks(ctx, tx) - userStars := getUserStars(ctx, tx) + var protoImageData *proto_go.ImageData - repoMetaBlob := buck.Get([]byte(repo)) + switch descriptor.MediaType { + case ispec.MediaTypeImageManifest: + manifestDigest := descriptor.Digest - // object not found - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } + imageManifestData, err := fetchProtoImageData(imageBuck, manifestDigest) + if err != nil { + return fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", + manifestDigest, err) + } - // object found - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } + protoImageData = imageManifestData + case ispec.MediaTypeImageIndex: + indexDigest := descriptor.Digest - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repo) - repoMeta.IsStarred = zcommon.Contains(userStars, repo) + imageIndexData, err := fetchProtoImageData(imageBuck, indexDigest) + if err != nil { + return fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", + indexDigest, err) + } - return nil - }) + manifestDataList := make([]*proto_go.ManifestData, 0, len(imageIndexData.Index.Index.Manifests)) - return repoMeta, err -} + for _, manifest := range imageIndexData.Index.Index.Manifests { + imageManifestData, err := fetchProtoImageData(imageBuck, manifest.Digest) + if err != nil { + return err + } -func (bdw *BoltDB) SetRepoMeta(repo string, repoMeta mTypes.RepoMetadata) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) + } - repoMeta.Name = repo + imageIndexData.Manifests = manifestDataList - repoMetaBlob, err := json.Marshal(repoMeta) - if err != nil { - return err + protoImageData = imageIndexData + default: + bdw.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + + continue + } + + images = append(images, mConvert.GetImageMetaFromProto(tag, protoRepoMeta, protoImageData)) } - return buck.Put([]byte(repo), repoMetaBlob) + return nil }) - return err + return images, err } -func (bdw *BoltDB) DeleteRepoTag(repo string, tag string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) - - repoMetaBlob := buck.Get([]byte(repo)) - - // object not found - if repoMetaBlob == nil { - return nil - } - - // object found - var repoMeta mTypes.RepoMetadata - - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } +func (bdw *BoltDB) FilterTags(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, + filterFunc mTypes.FilterFunc, +) ([]mTypes.ImageMeta, error) { + images := []mTypes.ImageMeta{} - delete(repoMeta.Tags, tag) + err := bdw.DB.View(func(transaction *bbolt.Tx) error { + var ( + repoBuck = transaction.Bucket([]byte(RepoMetaBuck)) + imageDataBuck = transaction.Bucket([]byte(ImageDataBuck)) + userBookmarks = getUserBookmarks(ctx, transaction) + userStars = getUserStars(ctx, transaction) + viewError error + ) - repoMetaBlob, err = json.Marshal(repoMeta) - if err != nil { - return err - } + cursor := repoBuck.Cursor() + repoName, repoMetaBlob := cursor.First() - return buck.Put([]byte(repo), repoMetaBlob) - }) + for ; repoName != nil; repoName, repoMetaBlob = cursor.Next() { + if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { + continue + } - return err -} + protoRepoMeta := &proto_go.RepoMeta{} -func (bdw *BoltDB) IncrementRepoStars(repo string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) + if err != nil { + viewError = errors.Join(viewError, err) - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } + continue + } - var repoMeta mTypes.RepoMetadata + delete(protoRepoMeta.Tags, "") + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) + repoMeta := mConvert.GetRepoMeta(protoRepoMeta) - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - repoMeta.Stars++ + for tag, descriptor := range protoRepoMeta.Tags { + if !filterRepoTag(string(repoName), tag) { + continue + } - repoMetaBlob, err = json.Marshal(repoMeta) - if err != nil { - return err - } + switch descriptor.MediaType { + case ispec.MediaTypeImageManifest: + manifestDigest := descriptor.Digest - return buck.Put([]byte(repo), repoMetaBlob) - }) + imageManifestData, err := fetchProtoImageData(imageDataBuck, manifestDigest) + if err != nil { + viewError = errors.Join(viewError, err) - return err -} + continue + } -func (bdw *BoltDB) DecrementRepoStars(repo string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + imageData := mConvert.GetImageData(imageManifestData) - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } + if filterFunc(repoMeta, imageData) { + images = append(images, mConvert.GetImageMetaFromProto(tag, protoRepoMeta, imageManifestData)) + } + case ispec.MediaTypeImageIndex: + indexDigest := descriptor.Digest - var repoMeta mTypes.RepoMetadata + imageIndexData, err := fetchProtoImageData(imageDataBuck, indexDigest) + if err != nil { + viewError = errors.Join(viewError, err) - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } + continue + } - if repoMeta.Stars > 0 { - repoMeta.Stars-- - } + matchedManifests := []*proto_go.ManifestData{} - repoMetaBlob, err = json.Marshal(repoMeta) - if err != nil { - return err - } + for _, manifest := range imageIndexData.Index.Index.Manifests { + manifestDigest := manifest.Digest - return buck.Put([]byte(repo), repoMetaBlob) - }) + imageManifestData, err := fetchProtoImageData(imageDataBuck, manifestDigest) + if err != nil { + viewError = errors.Join(viewError, err) - return err -} + continue + } -func (bdw *BoltDB) GetRepoStars(repo string) (int, error) { - stars := 0 + imageData := mConvert.GetImageData(imageManifestData) - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + if filterFunc(repoMeta, imageData) { + matchedManifests = append(matchedManifests, imageManifestData.Manifests[0]) + } + } - buck.Get([]byte(repo)) - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } + if len(matchedManifests) > 0 { + imageIndexData.Manifests = matchedManifests - var repoMeta mTypes.RepoMetadata + images = append(images, mConvert.GetImageMetaFromProto(tag, protoRepoMeta, imageIndexData)) + } + default: + bdw.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err + continue + } + } } - stars = repoMeta.Stars - - return nil + return viewError }) - return stars, err + return images, err } -func (bdw *BoltDB) GetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata) bool, -) ([]mTypes.RepoMetadata, error) { - foundRepos := []mTypes.RepoMetadata{} +func (bdw *BoltDB) FilterRepos(ctx context.Context, acceptName mTypes.FilterRepoNameFunc, + filter mTypes.FilterFullRepoFunc, +) ([]mTypes.RepoMeta, error) { + repos := []mTypes.RepoMeta{} err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) - - cursor := buck.Cursor() + var ( + buck = tx.Bucket([]byte(RepoMetaBuck)) + cursor = buck.Cursor() + userBookmarks = getUserBookmarks(ctx, tx) + userStars = getUserStars(ctx, tx) + ) for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { continue } - repoMeta := mTypes.RepoMetadata{} + if !acceptName(string(repoName)) { + continue + } + + repoMeta := proto_go.RepoMeta{} - err := json.Unmarshal(repoMetaBlob, &repoMeta) + err := proto.Unmarshal(repoMetaBlob, &repoMeta) if err != nil { return err } - if filter(repoMeta) { - foundRepos = append(foundRepos, repoMeta) + repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) + repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) + + fullRepoMeta := mConvert.GetRepoMeta(&repoMeta) + + if filter(fullRepoMeta) { + repos = append(repos, fullRepoMeta) } } return nil }) + if err != nil { + return []mTypes.RepoMeta{}, err + } - return foundRepos, err + return repos, err } -func (bdw *BoltDB) IncrementImageDownloads(repo string, reference string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) +func (bdw *BoltDB) GetRepoMeta(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + var protoRepoMeta proto_go.RepoMeta + + err := bdw.DB.View(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) + userBookmarks := getUserBookmarks(ctx, tx) + userStars := getUserStars(ctx, tx) repoMetaBlob := buck.Get([]byte(repo)) + + // object not found if repoMetaBlob == nil { - return zerr.ErrManifestMetaNotFound + return zerr.ErrRepoMetaNotFound } - var repoMeta mTypes.RepoMetadata - - err := json.Unmarshal(repoMetaBlob, &repoMeta) + // object found + err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) if err != nil { return err } - manifestDigest := reference + delete(protoRepoMeta.Tags, "") + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repo) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, repo) - if !common.ReferenceIsDigest(reference) { - // search digest for tag - descriptor, found := repoMeta.Tags[reference] + return nil + }) - if !found { - return zerr.ErrManifestMetaNotFound - } + return mConvert.GetRepoMeta(&protoRepoMeta), err +} - manifestDigest = descriptor.Digest +func (bdw *BoltDB) GetImageMeta(ctx context.Context, repo string, tag string) (mTypes.ImageMeta, error) { + protoRepoMeta := &proto_go.RepoMeta{} + protoImageData := &proto_go.ImageData{} + + err := bdw.DB.View(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) + imageBuck := tx.Bucket([]byte(ImageDataBuck)) + userBookmarks := getUserBookmarks(ctx, tx) + userStars := getUserStars(ctx, tx) + + repoMetaBlob := buck.Get([]byte(repo)) + + // object not found + if repoMetaBlob == nil { + return zerr.ErrRepoMetaNotFound + } + + // object found + err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) + if err != nil { + return err } - manifestStatistics := repoMeta.Statistics[manifestDigest] - manifestStatistics.DownloadCount++ - repoMeta.Statistics[manifestDigest] = manifestStatistics + delete(protoRepoMeta.Tags, "") + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repo) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, repo) - repoMetaBlob, err = json.Marshal(repoMeta) + descriptor, ok := protoRepoMeta.Tags[tag] + if !ok { + return zerr.ErrImageDataNotFound + } + + protoImageData, err = fetchProtoImageData(imageBuck, descriptor.Digest) if err != nil { return err } - return buck.Put([]byte(repo), repoMetaBlob) - }) + if protoImageData.MediaType == ispec.MediaTypeImageIndex { + manifestDataList := make([]*proto_go.ManifestData, 0, len(protoImageData.Index.Index.Manifests)) - return err -} + for _, manifest := range protoImageData.Index.Index.Manifests { + imageManifestData, err := fetchProtoImageData(imageBuck, manifest.Digest) + if err != nil { + return err + } -func (bdw *BoltDB) UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { - err := bdw.DB.Update(func(transaction *bbolt.Tx) error { - // TODO: Update RepoMetaProto as well - imgTrustStore := bdw.ImageTrustStore() + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) + } - if imgTrustStore == nil { - return nil + protoImageData.Manifests = manifestDataList } - // get ManifestData of signed manifest - manifestBuck := transaction.Bucket([]byte(ManifestDataBucket)) - mdBlob := manifestBuck.Get([]byte(manifestDigest)) + return nil + }) - var blob []byte + return mConvert.GetImageMetaFromProto(tag, protoRepoMeta, protoImageData), err +} - if len(mdBlob) != 0 { - var manifestData mTypes.ManifestData +func (bdw *BoltDB) GetImageData(digest godigest.Digest) (mTypes.ImageData, error) { + imageData := mTypes.ImageData{} - err := json.Unmarshal(mdBlob, &manifestData) - if err != nil { - return fmt.Errorf("metadb: %w error while unmashaling manifest meta for digest %s", err, manifestDigest) - } + err := bdw.DB.View(func(tx *bbolt.Tx) error { + imageBuck := tx.Bucket([]byte(ImageDataBuck)) - blob = manifestData.ManifestBlob - } else { - var indexData mTypes.IndexData + protoImageData, err := fetchProtoImageData(imageBuck, digest.String()) + if err != nil { + return err + } - indexBuck := transaction.Bucket([]byte(IndexDataBucket)) - idBlob := indexBuck.Get([]byte(manifestDigest)) + if protoImageData.MediaType == ispec.MediaTypeImageIndex { + manifestDataList := make([]*proto_go.ManifestData, 0, len(protoImageData.Index.Index.Manifests)) - if len(idBlob) == 0 { - // manifest meta not found, updating signatures with details about validity and author will not be performed - return nil - } + for _, manifest := range protoImageData.Index.Index.Manifests { + imageManifestData, err := fetchProtoImageData(imageBuck, manifest.Digest) + if err != nil { + return err + } - err := json.Unmarshal(idBlob, &indexData) - if err != nil { - return fmt.Errorf("metadb: %w error while unmashaling index meta for digest %s", err, manifestDigest) + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) } - blob = indexData.IndexBlob - } - - // update signatures with details about validity and author - repoBuck := transaction.Bucket([]byte(RepoMetadataBucket)) - - repoMetaBlob := repoBuck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound + protoImageData.Manifests = manifestDataList } - var repoMeta mTypes.RepoMetadata + imageData = mConvert.GetImageData(protoImageData) - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } + return nil + }) - manifestSignatures := mTypes.ManifestSignatures{} - for sigType, sigs := range repoMeta.Signatures[manifestDigest.String()] { - signaturesInfo := []mTypes.SignatureInfo{} + return imageData, err +} - for _, sigInfo := range sigs { - layersInfo := []mTypes.LayerInfo{} +func (bdw *BoltDB) GetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMeta) bool, +) ([]mTypes.RepoMeta, error) { + foundRepos := []mTypes.RepoMeta{} - for _, layerInfo := range sigInfo.LayersInfo { - author, date, isTrusted, _ := imgTrustStore.VerifySignature(sigType, layerInfo.LayerContent, - layerInfo.SignatureKey, manifestDigest, blob, repo) + err := bdw.DB.View(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) - if isTrusted { - layerInfo.Signer = author - } + cursor := buck.Cursor() - if !date.IsZero() { - layerInfo.Signer = author - layerInfo.Date = date - } + for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { + if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { + continue + } - layersInfo = append(layersInfo, layerInfo) - } + protoRepoMeta := proto_go.RepoMeta{} - signaturesInfo = append(signaturesInfo, mTypes.SignatureInfo{ - SignatureManifestDigest: sigInfo.SignatureManifestDigest, - LayersInfo: layersInfo, - }) + err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) + if err != nil { + return err } - manifestSignatures[sigType] = signaturesInfo - } + delete(protoRepoMeta.Tags, "") - repoMeta.Signatures[manifestDigest.String()] = manifestSignatures + repoMeta := mConvert.GetRepoMeta(&protoRepoMeta) - repoMetaBlob, err = json.Marshal(repoMeta) - if err != nil { - return err + if filter(repoMeta) { + foundRepos = append(foundRepos, repoMeta) + } } - return repoBuck.Put([]byte(repo), repoMetaBlob) + return nil }) - return err + return foundRepos, err } func (bdw *BoltDB) AddManifestSignature(repo string, signedManifestDigest godigest.Digest, sygMeta mTypes.SignatureMetadata, ) error { - // TODO: add support for RepoMetaProto err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + buck := tx.Bucket([]byte(RepoMetaBuck)) repoMetaBlob := buck.Get([]byte(repo)) if len(repoMetaBlob) == 0 { var err error // create a new object - repoMeta := mTypes.RepoMetadata{ + repoMeta := proto_go.RepoMeta{ Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Signatures: map[string]mTypes.ManifestSignatures{ + Tags: map[string]*proto_go.TagDescriptor{"": {}}, + Signatures: map[string]*proto_go.ManifestSignatures{ signedManifestDigest.String(): { - sygMeta.SignatureType: []mTypes.SignatureInfo{ - { - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: sygMeta.LayersInfo, + Map: map[string]*proto_go.SignaturesInfo{ + sygMeta.SignatureType: { + List: []*proto_go.SignatureInfo{ + { + SignatureManifestDigest: sygMeta.SignatureDigest, + LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), + }, + }, }, }, }, }, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, + Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, + Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, } - repoMetaBlob, err = json.Marshal(repoMeta) + repoMetaBlob, err = proto.Marshal(&repoMeta) if err != nil { return err } @@ -952,42 +801,46 @@ func (bdw *BoltDB) AddManifestSignature(repo string, signedManifestDigest godige return buck.Put([]byte(repo), repoMetaBlob) } - var repoMeta mTypes.RepoMetadata + protoRepoMeta := &proto_go.RepoMeta{} - err := json.Unmarshal(repoMetaBlob, &repoMeta) + err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) if err != nil { return err } var ( - manifestSignatures mTypes.ManifestSignatures + manifestSignatures *proto_go.ManifestSignatures found bool ) - if manifestSignatures, found = repoMeta.Signatures[signedManifestDigest.String()]; !found { - manifestSignatures = mTypes.ManifestSignatures{} + if manifestSignatures, found = protoRepoMeta.Signatures[signedManifestDigest.String()]; !found { + manifestSignatures = &proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} + } + + signatureSlice := &proto_go.SignaturesInfo{List: []*proto_go.SignatureInfo{}} + if sigSlice, found := manifestSignatures.Map[sygMeta.SignatureType]; found { + signatureSlice = sigSlice } - signatureSlice := manifestSignatures[sygMeta.SignatureType] - if !common.SignatureAlreadyExists(signatureSlice, sygMeta) { - if sygMeta.SignatureType == zcommon.NotationSignature { - signatureSlice = append(signatureSlice, mTypes.SignatureInfo{ + if !common.ProtoSignatureAlreadyExists(signatureSlice.List, sygMeta) { + switch sygMeta.SignatureType { + case zcommon.NotationSignature: + signatureSlice.List = append(signatureSlice.List, &proto_go.SignatureInfo{ SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: sygMeta.LayersInfo, + LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), }) - } else if sygMeta.SignatureType == zcommon.CosignSignature { - signatureSlice = []mTypes.SignatureInfo{{ + case zcommon.CosignSignature: + signatureSlice.List = []*proto_go.SignatureInfo{{ SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: sygMeta.LayersInfo, + LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), }} } } - manifestSignatures[sygMeta.SignatureType] = signatureSlice + manifestSignatures.Map[sygMeta.SignatureType] = signatureSlice + protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures - repoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures - - repoMetaBlob, err = json.Marshal(repoMeta) + repoMetaBlob, err = proto.Marshal(protoRepoMeta) if err != nil { return err } @@ -1002,46 +855,40 @@ func (bdw *BoltDB) DeleteSignature(repo string, signedManifestDigest godigest.Di sigMeta mTypes.SignatureMetadata, ) error { err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(RepoMetadataBucket)) + buck := tx.Bucket([]byte(RepoMetaBuck)) repoMetaBlob := buck.Get([]byte(repo)) if repoMetaBlob == nil { return zerr.ErrManifestMetaNotFound } - var repoMeta mTypes.RepoMetadata + protoRepoMeta := proto_go.RepoMeta{} - err := json.Unmarshal(repoMetaBlob, &repoMeta) + err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) if err != nil { return err } - sigType := sigMeta.SignatureType - - var ( - manifestSignatures mTypes.ManifestSignatures - found bool - ) - - if manifestSignatures, found = repoMeta.Signatures[signedManifestDigest.String()]; !found { + manifestSignatures, found := protoRepoMeta.Signatures[signedManifestDigest.String()] + if !found { return zerr.ErrManifestMetaNotFound } - signatureSlice := manifestSignatures[sigType] + signatureSlice := manifestSignatures.Map[sigMeta.SignatureType] - newSignatureSlice := make([]mTypes.SignatureInfo, 0, len(signatureSlice)-1) + newSignatureSlice := make([]*proto_go.SignatureInfo, 0, len(signatureSlice.List)) - for _, sigDigest := range signatureSlice { - if sigDigest.SignatureManifestDigest != sigMeta.SignatureDigest { - newSignatureSlice = append(newSignatureSlice, sigDigest) + for _, sigInfo := range signatureSlice.List { + if sigInfo.SignatureManifestDigest != sigMeta.SignatureDigest { + newSignatureSlice = append(newSignatureSlice, sigInfo) } } - manifestSignatures[sigType] = newSignatureSlice + manifestSignatures.Map[sigMeta.SignatureType] = &proto_go.SignaturesInfo{List: newSignatureSlice} - repoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures + protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures - repoMetaBlob, err = json.Marshal(repoMeta) + repoMetaBlob, err = proto.Marshal(&protoRepoMeta) if err != nil { return err } @@ -1052,464 +899,449 @@ func (bdw *BoltDB) DeleteSignature(repo string, signedManifestDigest godigest.Di return err } -func (bdw *BoltDB) SearchRepos(ctx context.Context, searchText string, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - var ( - foundRepos = make([]mTypes.RepoMetadata, 0) - manifestMetadataMap = make(map[string]mTypes.ManifestMetadata) - indexDataMap = make(map[string]mTypes.IndexData) - ) +func (bdw *BoltDB) IncrementRepoStars(repo string) error { + err := bdw.DB.Update(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) - err := bdw.DB.View(func(transaction *bbolt.Tx) error { - var ( - repoBuck = transaction.Bucket([]byte(RepoMetadataBucket)) - indexBuck = transaction.Bucket([]byte(IndexDataBucket)) - manifestBuck = transaction.Bucket([]byte(ManifestDataBucket)) - userBookmarks = getUserBookmarks(ctx, transaction) - userStars = getUserStars(ctx, transaction) - ) + repoMetaBlob := buck.Get([]byte(repo)) + if repoMetaBlob == nil { + return zerr.ErrRepoMetaNotFound + } - cursor := repoBuck.Cursor() + var repoMeta proto_go.RepoMeta - for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - continue - } + err := proto.Unmarshal(repoMetaBlob, &repoMeta) + if err != nil { + return err + } - var repoMeta mTypes.RepoMetadata + repoMeta.Stars++ - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } + repoMetaBlob, err = proto.Marshal(&repoMeta) + if err != nil { + return err + } - rank := common.RankRepoName(searchText, repoMeta.Name) - if rank == -1 { - continue - } + return buck.Put([]byte(repo), repoMetaBlob) + }) - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) - repoMeta.Rank = rank + return err +} - for tag, descriptor := range repoMeta.Tags { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest +func (bdw *BoltDB) DecrementRepoStars(repo string) error { + err := bdw.DB.Update(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) - manifestMeta, err := fetchManifestMetaWithCheck(repoMeta, manifestDigest, - manifestMetadataMap, manifestBuck) - if err != nil { - return fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", - manifestDigest, err) - } + repoMetaBlob := buck.Get([]byte(repo)) + if repoMetaBlob == nil { + return zerr.ErrRepoMetaNotFound + } - manifestMetadataMap[descriptor.Digest] = manifestMeta - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest + var repoMeta proto_go.RepoMeta - indexData, err := fetchIndexDataWithCheck(indexDigest, indexDataMap, indexBuck) - if err != nil { - return fmt.Errorf("metadb: error fetching index data for index with digest %s %w", - indexDigest, err) - } + err := proto.Unmarshal(repoMetaBlob, &repoMeta) + if err != nil { + return err + } - var indexContent ispec.Index + if repoMeta.Stars == 0 { + return nil + } - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return fmt.Errorf("metadb: error while unmashaling index content for %s:%s %w", - repoName, tag, err) - } + repoMeta.Stars-- - for _, manifest := range indexContent.Manifests { - manifestDigest := manifest.Digest + repoMetaBlob, err = proto.Marshal(&repoMeta) + if err != nil { + return err + } - manifestMeta, err := fetchManifestMetaWithCheck(repoMeta, manifestDigest.String(), - manifestMetadataMap, manifestBuck) - if err != nil { - return err - } + return buck.Put([]byte(repo), repoMetaBlob) + }) - manifestMetadataMap[manifest.Digest.String()] = manifestMeta - } + return err +} - indexDataMap[indexDigest] = indexData - default: - bdw.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") +func (bdw *BoltDB) GetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + var repoMeta proto_go.RepoMeta - continue - } - } + err := bdw.DB.View(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) + userBookmarks := getUserBookmarks(ctx, tx) + userStars := getUserStars(ctx, tx) + + repoMetaBlob := buck.Get([]byte(repo)) + + // object not found + if repoMetaBlob == nil { + return zerr.ErrRepoMetaNotFound + } - foundRepos = append(foundRepos, repoMeta) + // object found + err := proto.Unmarshal(repoMetaBlob, &repoMeta) + if err != nil { + return err } + repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repo) + repoMeta.IsStarred = zcommon.Contains(userStars, repo) + return nil }) - return foundRepos, manifestMetadataMap, indexDataMap, err + return mConvert.GetRepoMeta(&repoMeta), err } -func fetchManifestMetaWithCheck(repoMeta mTypes.RepoMetadata, manifestDigest string, - manifestMetadataMap map[string]mTypes.ManifestMetadata, manifestBuck *bbolt.Bucket, -) (mTypes.ManifestMetadata, error) { - manifestMeta, manifestDownloaded := manifestMetadataMap[manifestDigest] - - if !manifestDownloaded { - var manifestData mTypes.ManifestData +func (bdw *BoltDB) SetRepoMeta(repo string, repoMeta mTypes.RepoMeta) error { + err := bdw.DB.Update(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) - manifestDataBlob := manifestBuck.Get([]byte(manifestDigest)) - if manifestDataBlob == nil { - return mTypes.ManifestMetadata{}, zerr.ErrManifestMetaNotFound - } + repoMeta.Name = repo - err := json.Unmarshal(manifestDataBlob, &manifestData) + repoMetaBlob, err := proto.Marshal(mConvert.GetProtoRepoMeta(repoMeta)) if err != nil { - return mTypes.ManifestMetadata{}, fmt.Errorf("metadb: error while unmarshaling manifest metadata for digest %s %w", - manifestDigest, err) + return err } - manifestMeta = NewManifestMetadata(manifestDigest, repoMeta, manifestData) - } + return buck.Put([]byte(repo), repoMetaBlob) + }) - return manifestMeta, nil + return err } -func fetchIndexDataWithCheck(indexDigest string, indexDataMap map[string]mTypes.IndexData, - indexBuck *bbolt.Bucket, -) (mTypes.IndexData, error) { - var ( - indexData mTypes.IndexData - err error - ) +func (bdw *BoltDB) ResetRepoReferences(repo string) error { + err := bdw.DB.Update(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) - indexData, indexExists := indexDataMap[indexDigest] + repoMetaBlob := buck.Get([]byte(repo)) - if !indexExists { - indexDataBlob := indexBuck.Get([]byte(indexDigest)) - if indexDataBlob == nil { - return mTypes.IndexData{}, zerr.ErrIndexDataNotFount + if repoMetaBlob == nil { + return nil } - err := json.Unmarshal(indexDataBlob, &indexData) + protoRepoMeta := &proto_go.RepoMeta{} + + err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) if err != nil { - return mTypes.IndexData{}, - fmt.Errorf("metadb: error while unmashaling index data for digest %s %w", indexDigest, err) + return err } - } - - return indexData, err -} - -func NewManifestMetadata(manifestDigest string, repoMeta mTypes.RepoMetadata, - manifestData mTypes.ManifestData, -) mTypes.ManifestMetadata { - manifestMeta := mTypes.ManifestMetadata{ - ManifestBlob: manifestData.ManifestBlob, - ConfigBlob: manifestData.ConfigBlob, - } - manifestMeta.DownloadCount = repoMeta.Statistics[manifestDigest].DownloadCount + repoMetaBlob, err = proto.Marshal(&proto_go.RepoMeta{ + Name: repo, + Statistics: protoRepoMeta.Statistics, + Stars: protoRepoMeta.Stars, + Tags: map[string]*proto_go.TagDescriptor{"": {}}, + Signatures: map[string]*proto_go.ManifestSignatures{"": {Map: map[string]*proto_go.SignaturesInfo{"": {}}}}, + Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, + }) + if err != nil { + return err + } - manifestMeta.Signatures = mTypes.ManifestSignatures{} - if repoMeta.Signatures[manifestDigest] != nil { - manifestMeta.Signatures = repoMeta.Signatures[manifestDigest] - } + return buck.Put([]byte(repo), repoMetaBlob) + }) - return manifestMeta + return err } -func (bdw *BoltDB) FilterTags(ctx context.Context, filterFunc mTypes.FilterFunc, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error, -) { - var ( - foundRepos = make([]mTypes.RepoMetadata, 0) - manifestMetadataMap = make(map[string]mTypes.ManifestMetadata) - indexDataMap = make(map[string]mTypes.IndexData) - ) +func (bdw *BoltDB) GetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string, +) ([]mTypes.ReferrerInfo, error) { + referrersInfoResult := []mTypes.ReferrerInfo{} - err := bdw.DB.View(func(transaction *bbolt.Tx) error { - var ( - repoBuck = transaction.Bucket([]byte(RepoMetadataBucket)) - indexBuck = transaction.Bucket([]byte(IndexDataBucket)) - manifestBuck = transaction.Bucket([]byte(ManifestDataBucket)) - cursor = repoBuck.Cursor() - userBookmarks = getUserBookmarks(ctx, transaction) - userStars = getUserStars(ctx, transaction) - viewError error - ) + err := bdw.DB.View(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) - repoName, repoMetaBlob := cursor.First() + repoMetaBlob := buck.Get([]byte(repo)) + if len(repoMetaBlob) == 0 { + return zerr.ErrRepoMetaNotFound + } - for ; repoName != nil; repoName, repoMetaBlob = cursor.Next() { - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - continue - } + var repoMeta proto_go.RepoMeta - repoMeta := mTypes.RepoMetadata{} + err := proto.Unmarshal(repoMetaBlob, &repoMeta) + if err != nil { + return err + } - if err := json.Unmarshal(repoMetaBlob, &repoMeta); err != nil { - viewError = errors.Join(viewError, err) + referrersInfo := repoMeta.Referrers[referredDigest.String()].List + for i := range referrersInfo { + if !common.MatchesArtifactTypes(referrersInfo[i].ArtifactType, artifactTypes) { continue } - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) + referrersInfoResult = append(referrersInfoResult, mTypes.ReferrerInfo{ + Digest: referrersInfo[i].Digest, + MediaType: referrersInfo[i].MediaType, + ArtifactType: referrersInfo[i].ArtifactType, + Size: int(referrersInfo[i].Size), + Annotations: referrersInfo[i].Annotations, + }) + } - matchedTags := make(map[string]mTypes.Descriptor) - // take all manifestsMeta - for tag, descriptor := range repoMeta.Tags { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest + return nil + }) - manifestMeta, err := fetchManifestMetaWithCheck(repoMeta, manifestDigest, manifestMetadataMap, manifestBuck) - if err != nil { - err = fmt.Errorf("metadb: error while unmashaling manifest metadata for digest %s %w", manifestDigest, err) - viewError = errors.Join(viewError, err) + return referrersInfoResult, err +} - continue - } +func (bdw *BoltDB) IncrementImageDownloads(repo string, reference string) error { + err := bdw.DB.Update(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) + + repoMetaBlob := buck.Get([]byte(repo)) + if repoMetaBlob == nil { + return zerr.ErrRepoMetaNotFound + } - if filterFunc(repoMeta, manifestMeta) { - matchedTags[tag] = descriptor - manifestMetadataMap[manifestDigest] = manifestMeta - } - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest + var repoMeta proto_go.RepoMeta - indexData, err := fetchIndexDataWithCheck(indexDigest, indexDataMap, indexBuck) - if err != nil { - err = fmt.Errorf("metadb: error while getting index data for digest %s %w", indexDigest, err) - viewError = errors.Join(viewError, err) + err := proto.Unmarshal(repoMetaBlob, &repoMeta) + if err != nil { + return err + } - continue - } + manifestDigest := reference - var indexContent ispec.Index + if !common.ReferenceIsDigest(reference) { + // search digest for tag + descriptor, found := repoMeta.Tags[reference] - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - err = fmt.Errorf("metadb: error while unmashaling index content for digest %s %w", indexDigest, err) - viewError = errors.Join(viewError, err) + if !found { + return zerr.ErrManifestMetaNotFound + } - continue - } + manifestDigest = descriptor.Digest + } - matchedManifests := []ispec.Descriptor{} + manifestStatistics, ok := repoMeta.Statistics[manifestDigest] + if !ok { + return zerr.ErrManifestMetaNotFound + } - for _, manifest := range indexContent.Manifests { - manifestDigest := manifest.Digest.String() + manifestStatistics.DownloadCount++ + repoMeta.Statistics[manifestDigest] = manifestStatistics - manifestMeta, err := fetchManifestMetaWithCheck(repoMeta, manifestDigest, manifestMetadataMap, manifestBuck) - if err != nil { - err = fmt.Errorf("metadb: error while getting manifest data for digest %s %w", manifestDigest, err) - viewError = errors.Join(viewError, err) + repoMetaBlob, err = proto.Marshal(&repoMeta) + if err != nil { + return err + } - continue - } + return buck.Put([]byte(repo), repoMetaBlob) + }) - if filterFunc(repoMeta, manifestMeta) { - matchedManifests = append(matchedManifests, manifest) - manifestMetadataMap[manifestDigest] = manifestMeta - } - } + return err +} - if len(matchedManifests) > 0 { - indexContent.Manifests = matchedManifests +func (bdw *BoltDB) UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { + err := bdw.DB.Update(func(transaction *bbolt.Tx) error { + imgTrustStore := bdw.ImageTrustStore() - indexBlob, err := json.Marshal(indexContent) - if err != nil { - viewError = errors.Join(viewError, err) + if imgTrustStore == nil { + return nil + } - continue - } + // get ManifestData of signed manifest + imageDataBuck := transaction.Bucket([]byte(ImageDataBuck)) + idBlob := imageDataBuck.Get([]byte(manifestDigest)) - indexData.IndexBlob = indexBlob + if len(idBlob) == 0 { + // manifest meta not found, updating signatures with details about validity and author will not be performed + return nil + } - indexDataMap[indexDigest] = indexData - matchedTags[tag] = descriptor - } - default: - bdw.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + protoImageData := proto_go.ImageData{} - continue - } - } + err := proto.Unmarshal(idBlob, &protoImageData) + if err != nil { + return err + } - if len(matchedTags) == 0 { - continue - } + // update signatures with details about validity and author + repoBuck := transaction.Bucket([]byte(RepoMetaBuck)) + + repoMetaBlob := repoBuck.Get([]byte(repo)) + if repoMetaBlob == nil { + return zerr.ErrRepoMetaNotFound + } - repoMeta.Tags = matchedTags + protoRepoMeta := proto_go.RepoMeta{} - foundRepos = append(foundRepos, repoMeta) + err = proto.Unmarshal(repoMetaBlob, &protoRepoMeta) + if err != nil { + return err } - return viewError - }) + manifestSignatures := proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} + for sigType, sigs := range protoRepoMeta.Signatures[manifestDigest.String()].Map { + signaturesInfo := []*proto_go.SignatureInfo{} - return foundRepos, manifestMetadataMap, indexDataMap, err -} + for _, sigInfo := range sigs.List { + layersInfo := []*proto_go.LayersInfo{} -func (bdw *BoltDB) FilterRepos(ctx context.Context, filter mTypes.FilterRepoFunc) ( - []mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error, -) { - foundRepos := make([]mTypes.RepoMetadata, 0) + for _, layerInfo := range sigInfo.LayersInfo { + author, date, isTrusted, _ := imgTrustStore.VerifySignature(sigType, layerInfo.LayerContent, + layerInfo.SignatureKey, manifestDigest, mConvert.GetImageData(&protoImageData), repo) - err := bdw.DB.View(func(tx *bbolt.Tx) error { - var ( - buck = tx.Bucket([]byte(RepoMetadataBucket)) - cursor = buck.Cursor() - userBookmarks = getUserBookmarks(ctx, tx) - userStars = getUserStars(ctx, tx) - ) + if isTrusted { + layerInfo.Signer = author + } - for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - continue - } + if !date.IsZero() { + layerInfo.Signer = author + layerInfo.Date = timestamppb.New(date) + } - repoMeta := mTypes.RepoMetadata{} + layersInfo = append(layersInfo, layerInfo) + } - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err + signaturesInfo = append(signaturesInfo, &proto_go.SignatureInfo{ + SignatureManifestDigest: sigInfo.SignatureManifestDigest, + LayersInfo: layersInfo, + }) } - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) + manifestSignatures.Map[sigType] = &proto_go.SignaturesInfo{List: signaturesInfo} + } - if filter(repoMeta) { - foundRepos = append(foundRepos, repoMeta) - } + protoRepoMeta.Signatures[manifestDigest.String()] = &manifestSignatures + + repoMetaBlob, err = proto.Marshal(&protoRepoMeta) + if err != nil { + return err } - return nil + return repoBuck.Put([]byte(repo), repoMetaBlob) }) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, err - } - foundManifestMetadataMap, foundIndexDataMap, err := common.FetchDataForRepos(bdw, foundRepos) - - return foundRepos, foundManifestMetadataMap, foundIndexDataMap, err + return err } -func (bdw *BoltDB) SearchTags(ctx context.Context, searchText string, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - var ( - foundRepos = make([]mTypes.RepoMetadata, 0) - manifestMetadataMap = make(map[string]mTypes.ManifestMetadata) - indexDataMap = make(map[string]mTypes.IndexData) - ) - - searchedRepo, searchedTag, err := common.GetRepoTag(searchText) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("metadb: error while parsing search text, invalid format %w", err) - } - - err = bdw.DB.View(func(transaction *bbolt.Tx) error { - var ( - repoBuck = transaction.Bucket([]byte(RepoMetadataBucket)) - indexBuck = transaction.Bucket([]byte(IndexDataBucket)) - manifestBuck = transaction.Bucket([]byte(ManifestDataBucket)) - userBookmarks = getUserBookmarks(ctx, transaction) - userStars = getUserStars(ctx, transaction) - ) - - repoName, repoMetaBlob := repoBuck.Cursor().Seek([]byte(searchedRepo)) +func (bdw *BoltDB) RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest) error { + err := bdw.DB.Update(func(tx *bbolt.Tx) error { + buck := tx.Bucket([]byte(RepoMetaBuck)) + imageDataBuck := tx.Bucket([]byte(ImageDataBuck)) + repoBlobsBuck := tx.Bucket([]byte(RepoBlobsBuck)) - if string(repoName) != searchedRepo { + repoMetaBlob := buck.Get([]byte(repo)) + if repoMetaBlob == nil { return nil } - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { + repoMeta := &proto_go.RepoMeta{} + + err := proto.Unmarshal(repoMetaBlob, repoMeta) + if err != nil { return err } - repoMeta := mTypes.RepoMetadata{} - - err := json.Unmarshal(repoMetaBlob, &repoMeta) + imageData, err := fetchProtoImageData(imageDataBuck, manifestDigest.String()) if err != nil { + if errors.Is(err, zerr.ErrImageDataNotFound) { + return nil + } + return err } - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) + // Remove Referrers + if subject := mConvert.GetImageSubject(imageData); subject != nil { + referredDigest := subject.Digest.String() + refInfo := &proto_go.ReferrersInfo{} - matchedTags := make(map[string]mTypes.Descriptor) - - for tag, descriptor := range repoMeta.Tags { - if !strings.HasPrefix(tag, searchedTag) { - continue + if repoMeta.Referrers[referredDigest] != nil { + refInfo = repoMeta.Referrers[referredDigest] } - matchedTags[tag] = descriptor - - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest + referrers := refInfo.List - manifestMeta, err := fetchManifestMetaWithCheck(repoMeta, manifestDigest, manifestMetadataMap, manifestBuck) - if err != nil { - return fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", - manifestDigest, err) - } + for i := range referrers { + if referrers[i].Digest == manifestDigest.String() { + referrers[i].Count -= 1 - manifestMetadataMap[descriptor.Digest] = manifestMeta - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest + if referrers[i].Count == 0 || common.ReferenceIsDigest(reference) { + referrers = append(referrers[:i], referrers[i+1:]...) + } - indexData, err := fetchIndexDataWithCheck(indexDigest, indexDataMap, indexBuck) - if err != nil { - return fmt.Errorf("metadb: error fetching index data for index with digest %s %w", - indexDigest, err) + break } + } - var indexContent ispec.Index + refInfo.List = referrers - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return fmt.Errorf("metadb: error collecting filter data for index with digest %s %w", - indexDigest, err) + repoMeta.Referrers[referredDigest] = refInfo + } + + if !common.ReferenceIsDigest(reference) { + delete(repoMeta.Tags, reference) + } else { + // remove all tags pointing to this digest + for tag, desc := range repoMeta.Tags { + if desc.Digest == reference { + delete(repoMeta.Tags, tag) } + } + } - for _, manifest := range indexContent.Manifests { - manifestDigest := manifest.Digest.String() + /* try to find at least one tag pointing to manifestDigest + if not found then we can also remove everything related to this digest */ + var foundTag bool + for _, desc := range repoMeta.Tags { + if desc.Digest == manifestDigest.String() { + foundTag = true + } + } - manifestMeta, err := fetchManifestMetaWithCheck(repoMeta, manifestDigest, manifestMetadataMap, manifestBuck) - if err != nil { - return fmt.Errorf("metadb: error fetching from db manifest meta for manifest with digest %s %w", - manifestDigest, err) - } + if !foundTag { + delete(repoMeta.Statistics, manifestDigest.String()) + delete(repoMeta.Signatures, manifestDigest.String()) + delete(repoMeta.Referrers, manifestDigest.String()) + } - manifestMetadataMap[manifestDigest] = manifestMeta - } + repoBlobsBytes := repoBlobsBuck.Get([]byte(repoMeta.Name)) - indexDataMap[indexDigest] = indexData - default: - bdw.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + repoBlobs := &proto_go.RepoBlobs{} - continue + if repoBlobsBytes == nil { + repoBlobs.Blobs = map[string]*proto_go.BlobInfo{} + } else { + err := proto.Unmarshal(repoBlobsBytes, repoBlobs) + if err != nil { + return err } } - if len(matchedTags) == 0 { - return nil + repoMeta, repoBlobs, err = common.RemoveImageFromRepoMeta(repoMeta, repoBlobs, reference) + if err != nil { + return err } - repoMeta.Tags = matchedTags + repoBlobsBytes, err = proto.Marshal(repoBlobs) + if err != nil { + return err + } - foundRepos = append(foundRepos, repoMeta) + err = repoBlobsBuck.Put([]byte(repoMeta.Name), repoBlobsBytes) + if err != nil { + return err + } - return nil + repoMetaBlob, err = proto.Marshal(repoMeta) + if err != nil { + return err + } + + return buck.Put([]byte(repo), repoMetaBlob) }) - return foundRepos, manifestMetadataMap, indexDataMap, err + return err +} + +func (bdw *BoltDB) ImageTrustStore() mTypes.ImageTrustStore { + return bdw.imgTrustStore +} + +func (bdw *BoltDB) SetImageTrustStore(imgTrustStore mTypes.ImageTrustStore) { + bdw.imgTrustStore = imgTrustStore } func (bdw *BoltDB) ToggleStarRepo(ctx context.Context, repo string) (mTypes.ToggleState, error) { @@ -1549,16 +1381,16 @@ func (bdw *BoltDB) ToggleStarRepo(ctx context.Context, repo string) (mTypes.Togg return err } - repoBuck := tx.Bucket([]byte(RepoMetadataBucket)) + repoBuck := tx.Bucket([]byte(RepoMetaBuck)) repoMetaBlob := repoBuck.Get([]byte(repo)) if repoMetaBlob == nil { return zerr.ErrRepoMetaNotFound } - var repoMeta mTypes.RepoMetadata + repoMeta := &proto_go.RepoMeta{} - err = json.Unmarshal(repoMetaBlob, &repoMeta) + err = proto.Unmarshal(repoMetaBlob, repoMeta) if err != nil { return err } @@ -1570,7 +1402,7 @@ func (bdw *BoltDB) ToggleStarRepo(ctx context.Context, repo string) (mTypes.Togg repoMeta.Stars-- } - repoMetaBlob, err = json.Marshal(repoMeta) + repoMetaBlob, err = proto.Marshal(repoMeta) if err != nil { return err } @@ -2105,3 +1937,51 @@ func (bdw *BoltDB) DeleteUserData(ctx context.Context) error { return err } + +func (bdw *BoltDB) ResetDB() error { + err := bdw.DB.Update(func(transaction *bbolt.Tx) error { + err := resetBucket(transaction, RepoMetaBuck) + if err != nil { + return err + } + + err = resetBucket(transaction, ImageDataBuck) + if err != nil { + return err + } + + err = resetBucket(transaction, RepoBlobsBuck) + if err != nil { + return err + } + + err = resetBucket(transaction, UserAPIKeysBucket) + if err != nil { + return err + } + + err = resetBucket(transaction, UserDataBucket) + if err != nil { + return err + } + + return nil + }) + + return err +} + +func resetBucket(transaction *bbolt.Tx, bucketName string) error { + bucket := transaction.Bucket([]byte(bucketName)) + + if bucket != nil { + err := transaction.DeleteBucket([]byte(bucketName)) + if err != nil { + return err + } + } + + _, err := transaction.CreateBucketIfNotExists([]byte(bucketName)) + + return err +} diff --git a/pkg/meta/boltdb/boltdb_test.go b/pkg/meta/boltdb/boltdb_test.go index a7021388be..665cf72451 100644 --- a/pkg/meta/boltdb/boltdb_test.go +++ b/pkg/meta/boltdb/boltdb_test.go @@ -4,36 +4,25 @@ import ( "context" "crypto/rand" "encoding/base64" - "encoding/json" "math" "testing" "time" "github.com/opencontainers/go-digest" - ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" "go.etcd.io/bbolt" zerr "zotregistry.io/zot/errors" - zcommon "zotregistry.io/zot/pkg/common" "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta/boltdb" mTypes "zotregistry.io/zot/pkg/meta/types" reqCtx "zotregistry.io/zot/pkg/requestcontext" - . "zotregistry.io/zot/pkg/test/image-utils" ) type imgTrustStore struct{} func (its imgTrustStore) VerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest digest.Digest, manifestContent []byte, - repo string, -) (string, time.Time, bool, error) { - return "", time.Time{}, false, nil -} - -func (its imgTrustStore) ProtoVerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest digest.Digest, imageData mTypes.ImageData2, + signatureType string, rawSignature []byte, sigKey string, manifestDigest digest.Digest, imageData mTypes.ImageData, repo string, ) (string, time.Time, bool, error) { return "", time.Time{}, false, nil @@ -54,14 +43,6 @@ func TestWrapperErrors(t *testing.T) { boltdbWrapper.SetImageTrustStore(imgTrustStore{}) - repoMeta := mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - } - - repoMetaBlob, err := json.Marshal(repoMeta) - So(err, ShouldBeNil) - userAc := reqCtx.NewUserAccessControl() userAc.SetUsername("test") @@ -252,604 +233,6 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldNotBeNil) }) - Convey("GetManifestData", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - dataBuck := tx.Bucket([]byte(boltdb.ManifestDataBucket)) - - return dataBuck.Put([]byte("digest1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - _, err = boltdbWrapper.GetManifestData("digest1") - So(err, ShouldNotBeNil) - - _, err = boltdbWrapper.GetManifestMeta("repo1", "digest1") - So(err, ShouldNotBeNil) - }) - - Convey("SetManifestMeta", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - dataBuck := tx.Bucket([]byte(boltdb.ManifestDataBucket)) - - err := dataBuck.Put([]byte("digest1"), repoMetaBlob) - if err != nil { - return err - } - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.SetManifestMeta("repo1", "digest1", mTypes.ManifestMetadata{}) - So(err, ShouldNotBeNil) - - _, err = boltdbWrapper.GetManifestMeta("repo1", "digest1") - So(err, ShouldNotBeNil) - }) - - Convey("FilterRepos", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - err := buck.Put([]byte("badRepo"), []byte("bad repo")) - So(err, ShouldBeNil) - - return nil - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.FilterRepos(context.Background(), - func(repoMeta mTypes.RepoMetadata) bool { return true }) - So(err, ShouldNotBeNil) - }) - - Convey("SetReferrer", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.SetReferrer("repo", "ref", mTypes.ReferrerInfo{}) - So(err, ShouldNotBeNil) - }) - - Convey("DeleteReferrer", func() { - Convey("RepoMeta not found", func() { - err := boltdbWrapper.DeleteReferrer("r", "dig", "dig") - So(err, ShouldNotBeNil) - }) - - Convey("bad repo meta blob", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.DeleteReferrer("repo", "dig", "dig") - So(err, ShouldNotBeNil) - }) - }) - - Convey("SetRepoReference", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.SetRepoReference("repo1", "tag", "digest", ispec.MediaTypeImageManifest) - So(err, ShouldNotBeNil) - }) - - Convey("GetRepoMeta", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - _, err = boltdbWrapper.GetRepoMeta("repo1") - So(err, ShouldNotBeNil) - }) - - Convey("DeleteRepoTag", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.DeleteRepoTag("repo1", "tag") - So(err, ShouldNotBeNil) - }) - - Convey("GetReferrersInfo", func() { - _, err = boltdbWrapper.GetReferrersInfo("repo1", "tag", nil) - So(err, ShouldNotBeNil) - - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - _, err = boltdbWrapper.GetReferrersInfo("repo1", "tag", nil) - So(err, ShouldNotBeNil) - }) - - Convey("IncrementRepoStars", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.IncrementRepoStars("repo2") - So(err, ShouldNotBeNil) - - err = boltdbWrapper.IncrementRepoStars("repo1") - So(err, ShouldNotBeNil) - }) - - Convey("DecrementRepoStars", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.DecrementRepoStars("repo2") - So(err, ShouldNotBeNil) - - err = boltdbWrapper.DecrementRepoStars("repo1") - So(err, ShouldNotBeNil) - }) - - Convey("GetRepoStars", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - _, err = boltdbWrapper.GetRepoStars("repo1") - So(err, ShouldNotBeNil) - }) - - Convey("GetMultipleRepoMeta", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - _, err = boltdbWrapper.GetMultipleRepoMeta(context.TODO(), func(repoMeta mTypes.RepoMetadata) bool { - return true - }) - So(err, ShouldNotBeNil) - }) - - Convey("IncrementImageDownloads", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.IncrementImageDownloads("repo2", "tag") - So(err, ShouldNotBeNil) - - err = boltdbWrapper.IncrementImageDownloads("repo1", "tag") - So(err, ShouldNotBeNil) - - err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), repoMetaBlob) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.IncrementImageDownloads("repo1", "tag") - So(err, ShouldNotBeNil) - }) - - Convey("AddManifestSignature", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.AddManifestSignature("repo1", digest.FromString("dig"), - mTypes.SignatureMetadata{}) - So(err, ShouldNotBeNil) - - err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), repoMetaBlob) - }) - So(err, ShouldBeNil) - - // signatures not found - err = boltdbWrapper.AddManifestSignature("repo1", digest.FromString("dig"), - mTypes.SignatureMetadata{}) - So(err, ShouldBeNil) - - // - err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - repoMeta := mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{}, - Signatures: map[string]mTypes.ManifestSignatures{ - "digest1": { - "cosgin": {{}}, - }, - "digest2": { - "notation": {{}}, - }, - }, - } - - repoMetaBlob, err := json.Marshal(repoMeta) - So(err, ShouldBeNil) - - return repoBuck.Put([]byte("repo1"), repoMetaBlob) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.AddManifestSignature("repo1", digest.FromString("dig"), - mTypes.SignatureMetadata{ - SignatureType: "cosign", - SignatureDigest: "digest1", - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.AddManifestSignature("repo1", digest.FromString("dig"), - mTypes.SignatureMetadata{ - SignatureType: "cosign", - SignatureDigest: "digest2", - }) - So(err, ShouldBeNil) - - repoData, err := boltdbWrapper.GetRepoMeta("repo1") - So(err, ShouldBeNil) - So(len(repoData.Signatures[string(digest.FromString("dig"))][zcommon.CosignSignature]), - ShouldEqual, 1) - So(repoData.Signatures[string(digest.FromString("dig"))][zcommon.CosignSignature][0].SignatureManifestDigest, - ShouldEqual, "digest2") - - err = boltdbWrapper.AddManifestSignature("repo1", digest.FromString("dig"), - mTypes.SignatureMetadata{ - SignatureType: "notation", - SignatureDigest: "digest2", - }) - So(err, ShouldBeNil) - }) - - Convey("DeleteSignature", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.DeleteSignature("repo2", digest.FromString("dig"), - mTypes.SignatureMetadata{}) - So(err, ShouldNotBeNil) - - err = boltdbWrapper.DeleteSignature("repo1", digest.FromString("dig"), - mTypes.SignatureMetadata{}) - So(err, ShouldNotBeNil) - - err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - repoMeta := mTypes.RepoMetadata{ - Tags: map[string]mTypes.Descriptor{}, - Signatures: map[string]mTypes.ManifestSignatures{ - "digest1": { - "cosgin": []mTypes.SignatureInfo{ - { - SignatureManifestDigest: "sigDigest1", - }, - { - SignatureManifestDigest: "sigDigest2", - }, - }, - }, - "digest2": { - "notation": {{}}, - }, - }, - } - - repoMetaBlob, err := json.Marshal(repoMeta) - So(err, ShouldBeNil) - - return repoBuck.Put([]byte("repo1"), repoMetaBlob) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.DeleteSignature("repo1", "digest1", - mTypes.SignatureMetadata{ - SignatureType: "cosgin", - SignatureDigest: "sigDigest2", - }) - So(err, ShouldBeNil) - }) - - Convey("SearchRepos", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchRepos(context.Background(), "") - So(err, ShouldNotBeNil) - - err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - dataBuck := tx.Bucket([]byte(boltdb.ManifestDataBucket)) - - err := dataBuck.Put([]byte("dig1"), []byte("wrong json")) - if err != nil { - return err - } - - repoMeta := mTypes.RepoMetadata{ - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageManifest}, - }, - Signatures: map[string]mTypes.ManifestSignatures{}, - } - repoMetaBlob, err := json.Marshal(repoMeta) - So(err, ShouldBeNil) - - err = repoBuck.Put([]byte("repo1"), repoMetaBlob) - if err != nil { - return err - } - - repoMeta = mTypes.RepoMetadata{ - Name: "repo2", - Tags: map[string]mTypes.Descriptor{ - "tag2": {Digest: "dig2", MediaType: ispec.MediaTypeImageManifest}, - }, - Signatures: map[string]mTypes.ManifestSignatures{}, - } - repoMetaBlob, err = json.Marshal(repoMeta) - So(err, ShouldBeNil) - - return repoBuck.Put([]byte("repo2"), repoMetaBlob) - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchRepos(context.Background(), "repo1") - So(err, ShouldNotBeNil) - - _, _, _, err = boltdbWrapper.SearchRepos(context.Background(), "repo2") - So(err, ShouldNotBeNil) - }) - - Convey("Index Errors", func() { - Convey("Bad index data", func() { - indexDigest := digest.FromString("indexDigest") - - err := boltdbWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = setBadIndexData(boltdbWrapper.DB, indexDigest.String()) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchRepos(ctx, "") - So(err, ShouldNotBeNil) - - _, _, _, err = boltdbWrapper.SearchTags(ctx, "repo:") - So(err, ShouldNotBeNil) - }) - - Convey("Bad indexBlob in IndexData", func() { - indexDigest := digest.FromString("indexDigest") - - err := boltdbWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = boltdbWrapper.SetIndexData(indexDigest, mTypes.IndexData{ - IndexBlob: []byte("bad json"), - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchRepos(ctx, "") - So(err, ShouldNotBeNil) - - _, _, _, err = boltdbWrapper.SearchTags(ctx, "repo:") - So(err, ShouldNotBeNil) - }) - }) - - Convey("SearchTags", func() { - ctx := context.Background() - - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchTags(ctx, "") - So(err, ShouldNotBeNil) - - _, _, _, err = boltdbWrapper.SearchTags(ctx, "repo1:") - So(err, ShouldNotBeNil) - - err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - dataBuck := tx.Bucket([]byte(boltdb.ManifestDataBucket)) - - manifestMeta := mTypes.ManifestMetadata{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("wrong json"), - Signatures: mTypes.ManifestSignatures{}, - } - - manifestMetaBlob, err := json.Marshal(manifestMeta) - if err != nil { - return err - } - - err = dataBuck.Put([]byte("dig1"), manifestMetaBlob) - if err != nil { - return err - } - - err = dataBuck.Put([]byte("wrongManifestData"), []byte("wrong json")) - if err != nil { - return err - } - - // manifest data doesn't exist - repoMeta = mTypes.RepoMetadata{ - Name: "repo1", - Tags: map[string]mTypes.Descriptor{ - "tag2": {Digest: "dig2", MediaType: ispec.MediaTypeImageManifest}, - }, - Signatures: map[string]mTypes.ManifestSignatures{}, - } - repoMetaBlob, err = json.Marshal(repoMeta) - So(err, ShouldBeNil) - - err = repoBuck.Put([]byte("repo1"), repoMetaBlob) - if err != nil { - return err - } - - // manifest data is wrong - repoMeta = mTypes.RepoMetadata{ - Name: "repo2", - Tags: map[string]mTypes.Descriptor{ - "tag2": {Digest: "wrongManifestData", MediaType: ispec.MediaTypeImageManifest}, - }, - Signatures: map[string]mTypes.ManifestSignatures{}, - } - repoMetaBlob, err = json.Marshal(repoMeta) - So(err, ShouldBeNil) - - err = repoBuck.Put([]byte("repo2"), repoMetaBlob) - if err != nil { - return err - } - - repoMeta = mTypes.RepoMetadata{ - Name: "repo3", - Tags: map[string]mTypes.Descriptor{ - "tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageManifest}, - }, - Signatures: map[string]mTypes.ManifestSignatures{}, - } - repoMetaBlob, err = json.Marshal(repoMeta) - So(err, ShouldBeNil) - - return repoBuck.Put([]byte("repo3"), repoMetaBlob) - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchTags(ctx, "repo1:") - So(err, ShouldNotBeNil) - - _, _, _, err = boltdbWrapper.SearchTags(ctx, "repo2:") - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags Index errors", func() { - Convey("FilterTags bad IndexData", func() { - indexDigest := digest.FromString("indexDigest") - - err := boltdbWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = setBadIndexData(boltdbWrapper.DB, indexDigest.String()) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags bad indexBlob in IndexData", func() { - indexDigest := digest.FromString("indexDigest") - - err := boltdbWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = boltdbWrapper.SetIndexData(indexDigest, mTypes.IndexData{ - IndexBlob: []byte("bad json"), - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags didn't match any index manifest", func() { - var ( - indexDigest = digest.FromString("indexDigest") - manifestDigestFromIndex1 = digest.FromString("manifestDigestFromIndex1") - manifestDigestFromIndex2 = digest.FromString("manifestDigestFromIndex2") - ) - - err := boltdbWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - indexBlob, err := GetIndexBlobWithManifests([]digest.Digest{ - manifestDigestFromIndex1, manifestDigestFromIndex2, - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.SetIndexData(indexDigest, mTypes.IndexData{ - IndexBlob: indexBlob, - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.SetManifestData(manifestDigestFromIndex1, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.SetManifestData(manifestDigestFromIndex2, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return false }) - So(err, ShouldBeNil) - }) - }) - Convey("ToggleStarRepo bad context errors", func() { uacKey := reqCtx.GetContextKey() ctx := context.WithValue(context.Background(), uacKey, "bad context") @@ -867,7 +250,7 @@ func TestWrapperErrors(t *testing.T) { ctx := userAc.DeriveContext(context.Background()) err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) + repoBuck := tx.Bucket([]byte(boltdb.RepoMetaBuck)) err := repoBuck.Put([]byte("repo"), []byte("bad repo")) So(err, ShouldBeNil) @@ -983,23 +366,6 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldNotBeNil) }) - Convey("Unsuported type", func() { - digest := digest.FromString("digest") - - err := boltdbWrapper.SetRepoReference("repo", "tag1", digest, "invalid type") //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchRepos(ctx, "") - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.SearchTags(ctx, "repo:") - So(err, ShouldBeNil) - - _, _, _, err = boltdbWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) - So(err, ShouldBeNil) - }) - Convey("GetUserRepoMeta unmarshal error", func() { userAc := reqCtx.NewUserAccessControl() userAc.SetUsername("username") @@ -1009,7 +375,7 @@ func TestWrapperErrors(t *testing.T) { ctx := userAc.DeriveContext(context.Background()) err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) + repoBuck := tx.Bucket([]byte(boltdb.RepoMetaBuck)) err := repoBuck.Put([]byte("repo"), []byte("bad repo")) So(err, ShouldBeNil) @@ -1018,78 +384,8 @@ func TestWrapperErrors(t *testing.T) { }) So(err, ShouldBeNil) - _, err := boltdbWrapper.GetUserRepoMeta(ctx, "repo") + _, err := boltdbWrapper.GetRepoMeta(ctx, "repo") So(err, ShouldNotBeNil) }) - - Convey("UpdateSignaturesValidity", func() { - Convey("manifestMeta of signed manifest not found", func() { - err := boltdbWrapper.UpdateSignaturesValidity("repo", digest.FromString("dig")) - So(err, ShouldBeNil) - }) - - Convey("repoMeta of signed manifest not found", func() { - // repo Meta not found - err := boltdbWrapper.SetManifestData(digest.FromString("dig"), mTypes.ManifestData{ - ManifestBlob: []byte("Bad Manifest"), - ConfigBlob: []byte("Bad Manifest"), - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.UpdateSignaturesValidity("repo", digest.FromString("dig")) - So(err, ShouldNotBeNil) - }) - - Convey("manifest - bad content", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - dataBuck := tx.Bucket([]byte(boltdb.ManifestDataBucket)) - - return dataBuck.Put([]byte("digest1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.UpdateSignaturesValidity("repo1", "digest1") - So(err, ShouldNotBeNil) - }) - - Convey("index - bad content", func() { - err := boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - dataBuck := tx.Bucket([]byte(boltdb.IndexDataBucket)) - - return dataBuck.Put([]byte("digest1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.UpdateSignaturesValidity("repo1", "digest1") - So(err, ShouldNotBeNil) - }) - - Convey("repo - bad content", func() { - // repo Meta not found - err := boltdbWrapper.SetManifestData(digest.FromString("dig"), mTypes.ManifestData{ - ManifestBlob: []byte("Bad Manifest"), - ConfigBlob: []byte("Bad Manifest"), - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(boltdb.RepoMetadataBucket)) - - return repoBuck.Put([]byte("repo1"), []byte("wrong json")) - }) - So(err, ShouldBeNil) - - err = boltdbWrapper.UpdateSignaturesValidity("repo1", digest.FromString("dig")) - So(err, ShouldNotBeNil) - }) - }) - }) -} - -func setBadIndexData(dB *bbolt.DB, digest string) error { - return dB.Update(func(tx *bbolt.Tx) error { - indexDataBuck := tx.Bucket([]byte(boltdb.IndexDataBucket)) - - return indexDataBuck.Put([]byte(digest), []byte("bad json")) }) } diff --git a/pkg/meta/boltdb/buckets.go b/pkg/meta/boltdb/buckets.go index aa9475072c..e15b7059f4 100644 --- a/pkg/meta/boltdb/buckets.go +++ b/pkg/meta/boltdb/buckets.go @@ -2,16 +2,10 @@ package boltdb // MetadataDB. const ( - ManifestDataBucket = "ManifestData" - IndexDataBucket = "IndexData" - RepoMetadataBucket = "RepoMetadata" - UserDataBucket = "UserData" - VersionBucket = "Version" - UserAPIKeysBucket = "UserAPIKeys" -) - -const ( - ProtoImageDataBuck = "ProtoImageData" - ProtoRepoMetaBuck = "ProtoRepoMeta" - ProtoRepoBlobsBuck = "ProtoRepoBlobsMeta" + ImageDataBuck = "ImageData" + RepoMetaBuck = "RepoMeta" + RepoBlobsBuck = "RepoBlobsMeta" + UserDataBucket = "UserData" + VersionBucket = "Version" + UserAPIKeysBucket = "UserAPIKeys" ) diff --git a/pkg/meta/boltdb/proto_boltdb.go b/pkg/meta/boltdb/proto_boltdb.go deleted file mode 100644 index fcfc3bc7fc..0000000000 --- a/pkg/meta/boltdb/proto_boltdb.go +++ /dev/null @@ -1,1109 +0,0 @@ -package boltdb - -import ( - "context" - "encoding/json" - "fmt" - "strings" - - godigest "github.com/opencontainers/go-digest" - ispec "github.com/opencontainers/image-spec/specs-go/v1" - "go.etcd.io/bbolt" - "google.golang.org/protobuf/proto" - "google.golang.org/protobuf/types/known/timestamppb" - - zerr "zotregistry.io/zot/errors" - zcommon "zotregistry.io/zot/pkg/common" - "zotregistry.io/zot/pkg/meta/common" - mConvert "zotregistry.io/zot/pkg/meta/convert" - "zotregistry.io/zot/pkg/meta/proto_go" - mTypes "zotregistry.io/zot/pkg/meta/types" - reqCtx "zotregistry.io/zot/pkg/requestcontext" -) - -func (bdw *BoltDB) ProtoSetImageData(digest godigest.Digest, imageData mTypes.ImageData2) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoImageDataBuck)) - - protoImageData := &proto_go.ImageData{} - - switch imageData.MediaType { - case ispec.MediaTypeImageManifest: - manifest := imageData.Manifests[0] - - protoImageData = mConvert.GetProtoImageManifestData(manifest.Manifest, manifest.ConfigContent, - manifest.Size, manifest.Digest.String()) - case ispec.MediaTypeImageIndex: - protoImageData = mConvert.GetProtoImageIndexData(*imageData.Index, imageData.Size, imageData.Digest.String()) - } - - pImageDataBlob, err := proto.Marshal(protoImageData) - if err != nil { - return fmt.Errorf("metadb: error while calculating blob for manifest with digest %s %w", digest, err) - } - - err = buck.Put([]byte(digest), pImageDataBlob) - if err != nil { - return fmt.Errorf("metadb: error while setting manifest data with for digest %s %w", digest, err) - } - - return nil - }) - - return err -} - -func (bdw *BoltDB) ProtoSetRepoReference(repo string, reference string, imageData mTypes.ImageData2, -) error { - if err := common.ValidateRepoReferenceInput(repo, reference, imageData.Digest); err != nil { - return err - } - - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - repoBuck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - repoBlobsBuck := tx.Bucket([]byte(ProtoRepoBlobsBuck)) - imageBuck := tx.Bucket([]byte(ProtoImageDataBuck)) - - // 1. Add image data to db if needed - - protoImageData := mConvert.GetProtoImageData(imageData) - - imageDataBlob, err := proto.Marshal(protoImageData) - if err != nil { - return err - } - - imageBuck.Put([]byte(imageData.Digest), imageDataBlob) - - repoMetaBlob := repoBuck.Get([]byte(repo)) - - repoMeta := &proto_go.ProtoRepoMeta{ - Name: repo, - Tags: map[string]*proto_go.TagDescriptor{"": {}}, // This is done so Protobuff can insitialize a non-nill map - Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, - Signatures: map[string]*proto_go.ManifestSignatures{"": {Map: map[string]*proto_go.SignaturesInfo{"": {}}}}, - Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, - } - - if len(repoMetaBlob) > 0 { - err := proto.Unmarshal(repoMetaBlob, repoMeta) - if err != nil { - return err - } - } - - // 2. Referrers - if protoImageData.Subject != nil { - refList := repoMeta.Referrers[protoImageData.Subject.Digest].List - newRefList := append(refList, &proto_go.ReferrerInfo{ - Digest: protoImageData.Digest, - MediaType: protoImageData.MediaType, - ArtifactType: protoImageData.ArtifacType, - Size: protoImageData.Size, - Annotations: protoImageData.Annotations, - }) - repoMeta.Referrers[protoImageData.Subject.Digest].List = newRefList - } - - // 3. Update tag - if !common.ReferenceIsDigest(reference) { - repoMeta.Tags[reference] = &proto_go.TagDescriptor{ - Digest: imageData.Digest.String(), - MediaType: imageData.MediaType, - } - } - - if _, ok := repoMeta.Statistics[imageData.Digest.String()]; !ok { - repoMeta.Statistics[imageData.Digest.String()] = &proto_go.DescriptorStatistics{DownloadCount: 0} - } - - if _, ok := repoMeta.Signatures[imageData.Digest.String()]; !ok { - repoMeta.Signatures[imageData.Digest.String()] = &proto_go.ManifestSignatures{ - Map: map[string]*proto_go.SignaturesInfo{"": {}}, - } - } - - if _, ok := repoMeta.Referrers[imageData.Digest.String()]; !ok { - repoMeta.Referrers[imageData.Digest.String()] = &proto_go.ReferrersInfo{ - List: []*proto_go.ReferrerInfo{}, - } - } - - // 4. Blobs - repoBlobsBytes := repoBlobsBuck.Get([]byte(repoMeta.Name)) - - repoBlobs := &proto_go.RepoBlobs{} - - if repoBlobsBytes == nil { - repoBlobs.Blobs = map[string]*proto_go.BlobInfo{} - } else { - err := proto.Unmarshal(repoBlobsBytes, repoBlobs) - if err != nil { - return nil - } - } - - repoMeta, repoBlobs, err = common.GetUpdatedRepoMeta(repoMeta, repoBlobs, reference, imageData) - if err != nil { - return err - } - - repoBlobsBytes, err = proto.Marshal(repoBlobs) - if err != nil { - return err - } - - repoBlobsBuck.Put([]byte(repoMeta.Name), repoBlobsBytes) - - repoMetaBlob, err = proto.Marshal(repoMeta) - if err != nil { - return err - } - - return repoBuck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoFilterImageData(ctx context.Context, digests []string, -) (map[string]mTypes.ImageData2, error) { - imageDataMap := map[string]mTypes.ImageData2{} - - err := bdw.DB.View(func(transaction *bbolt.Tx) error { - imageBuck := transaction.Bucket([]byte(ProtoImageDataBuck)) - - for _, digest := range digests { - protoImageData, err := fetchProtoImageData(imageBuck, digest) - if err != nil { - return err - } - - if protoImageData.MediaType == ispec.MediaTypeImageIndex { - for i, manifest := range protoImageData.Manifests { - manifestDigest := manifest.Digest - - imageManifestData, err := fetchProtoImageData(imageBuck, manifestDigest) - if err != nil { - return err - } - - protoImageData.Manifests[i] = imageManifestData.Manifests[0] - } - } - - imageDataMap[digest] = mConvert.GetImageData(protoImageData) - } - - return nil - }) - - return imageDataMap, err -} - -// TODO -func (bdw *BoltDB) ProtoSearchRepos(ctx context.Context, searchText string, -) ([]mTypes.FullRepoMetadata, error) { - repos := []mTypes.FullRepoMetadata{} - - err := bdw.DB.View(func(transaction *bbolt.Tx) error { - var ( - repoBuck = transaction.Bucket([]byte(ProtoRepoMetaBuck)) - userBookmarks = getUserBookmarks(ctx, transaction) - userStars = getUserStars(ctx, transaction) - ) - - cursor := repoBuck.Cursor() - - for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - continue - } - - rank := common.RankRepoName(searchText, string(repoName)) - if rank == -1 { - continue - } - - var protoRepoMeta proto_go.ProtoRepoMeta - - err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) - if err != nil { - return err - } - - delete(protoRepoMeta.Tags, "") - - protoRepoMeta.Rank = int32(rank) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) - - repos = append(repos, mConvert.GetFullRepoMeta(&protoRepoMeta)) - } - - return nil - }) - - return repos, err -} - -func fetchProtoImageData(imageBuck *bbolt.Bucket, digest string) (*proto_go.ImageData, error) { - imageDataBlob := imageBuck.Get([]byte(digest)) - // TODO: check empty imageDataBlob - - imageData := proto_go.ImageData{} - - err := proto.Unmarshal(imageDataBlob, &imageData) - if err != nil { - return nil, err - } - - return &imageData, nil -} - -func (bdw *BoltDB) ProtoSearchTags(ctx context.Context, searchText string, -) ([]mTypes.FullImageData, error) { - images := []mTypes.FullImageData{} - - searchedRepo, searchedTag, err := common.GetRepoTag(searchText) - if err != nil { - return []mTypes.FullImageData{}, - fmt.Errorf("metadb: error while parsing search text, invalid format %w", err) - } - - err = bdw.DB.View(func(transaction *bbolt.Tx) error { - var ( - repoBuck = transaction.Bucket([]byte(ProtoRepoMetaBuck)) - imageBuck = transaction.Bucket([]byte(ProtoImageDataBuck)) - userBookmarks = getUserBookmarks(ctx, transaction) - userStars = getUserStars(ctx, transaction) - ) - - repoName, repoMetaBlob := repoBuck.Cursor().Seek([]byte(searchedRepo)) - - if string(repoName) != searchedRepo { - return nil - } - - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - return err - } - - protoRepoMeta := &proto_go.ProtoRepoMeta{} - - err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) - if err != nil { - return err - } - - delete(protoRepoMeta.Tags, "") - - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) - - for tag, descriptor := range protoRepoMeta.Tags { - if !strings.HasPrefix(tag, searchedTag) || tag == "" { - continue - } - - protoImageData := &proto_go.ImageData{} - - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest - - imageManifestData, err := fetchProtoImageData(imageBuck, manifestDigest) - if err != nil { - return fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", - manifestDigest, err) - } - - imageManifestData.Repo = ref(string(repoName)) - imageManifestData.Tag = &tag - - protoImageData = imageManifestData - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest - - imageIndexData, err := fetchProtoImageData(imageBuck, indexDigest) - if err != nil { - return fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", - indexDigest, err) - } - - for i, manifest := range imageIndexData.Manifests { - manifestDigest := manifest.Digest - - imageManifestData, err := fetchProtoImageData(imageBuck, manifestDigest) - if err != nil { - return err - } - - imageIndexData.Manifests[i] = imageManifestData.Manifests[0] - } - - imageIndexData.Repo = ref(string(repoName)) - imageIndexData.Tag = &tag - - protoImageData = imageIndexData - default: - bdw.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") - - continue - } - - // fullImageData := getFullImageData(tag, protoRepoMeta, protoImageData) - images = append(images, mConvert.GetFullImageData(tag, protoRepoMeta, protoImageData)) - } - - return nil - }) - - return images, err -} - -func (bdw *BoltDB) ProtoFilterTags(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterProtoFunc, -) ([]mTypes.FullImageData, error) { - images := []mTypes.FullImageData{} - - err := bdw.DB.View(func(transaction *bbolt.Tx) error { - var ( - repoBuck = transaction.Bucket([]byte(ProtoRepoMetaBuck)) - imageDataBuck = transaction.Bucket([]byte(ProtoImageDataBuck)) - userBookmarks = getUserBookmarks(ctx, transaction) - userStars = getUserStars(ctx, transaction) - ) - - cursor := repoBuck.Cursor() - repoName, repoMetaBlob := cursor.First() - - for ; repoName != nil; repoName, repoMetaBlob = cursor.Next() { - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - continue - } - - protoRepoMeta := &proto_go.ProtoRepoMeta{} - - err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) - if err != nil { - return err - } - - delete(protoRepoMeta.Tags, "") - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) - repoMeta := mConvert.GetRepoMeta(protoRepoMeta) - - for tag, descriptor := range protoRepoMeta.Tags { - if !filterRepoTag(string(repoName), tag) { - continue - } - - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest - - imageManifestData, err := fetchProtoImageData(imageDataBuck, manifestDigest) - if err != nil { - return fmt.Errorf("metadb: error while unmashaling manifest metadata for digest %s %w", manifestDigest, err) - } - - imageManifestData.Repo = ref(repoMeta.Name) - imageManifestData.Tag = ref(tag) - - imageData := mConvert.GetImageData(imageManifestData) - - if filterFunc(repoMeta, imageData) { - images = append(images, mConvert.GetFullImageData(tag, protoRepoMeta, imageManifestData)) - } - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest - - imageIndexData, err := fetchProtoImageData(imageDataBuck, indexDigest) - if err != nil { - return fmt.Errorf("metadb: error while getting index data for digest %s %w", indexDigest, err) - } - - matchedManifests := []*proto_go.ManifestData{} - - for _, manifest := range imageIndexData.Manifests { - manifestDigest := manifest.Digest - - imageManifestData, err := fetchProtoImageData(imageDataBuck, manifestDigest) - if err != nil { - return fmt.Errorf("metadb: error while getting manifest data for digest %s %w", manifestDigest, err) - } - - imageData := mConvert.GetImageData(imageManifestData) - - if filterFunc(repoMeta, imageData) { - matchedManifests = append(matchedManifests, imageManifestData.Manifests[0]) - } - } - - if len(matchedManifests) > 0 { - imageIndexData.Manifests = matchedManifests - - images = append(images, mConvert.GetFullImageData(tag, protoRepoMeta, imageIndexData)) - } - default: - bdw.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") - - continue - } - } - } - - return nil - }) - - return images, err -} - -func (bdw *BoltDB) ProtoFilterRepos(ctx context.Context, rankName mTypes.FilterRepoNameFunc, filter mTypes.FilterFullRepoFunc, -) ([]mTypes.FullRepoMetadata, error) { - repos := []mTypes.FullRepoMetadata{} - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - var ( - buck = tx.Bucket([]byte(ProtoRepoMetaBuck)) - cursor = buck.Cursor() - userBookmarks = getUserBookmarks(ctx, tx) - userStars = getUserStars(ctx, tx) - ) - - for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - continue - } - - rank := rankName(string(repoName)) - if rank < 0 { - continue - } - - repoMeta := proto_go.ProtoRepoMeta{} - - err := json.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - repoMeta.Rank = int32(rank) - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) - - fullRepoMeta := mConvert.GetFullRepoMeta(&repoMeta) - - if filter(fullRepoMeta) { - repos = append(repos, fullRepoMeta) - } - } - - return nil - }) - if err != nil { - return []mTypes.FullRepoMetadata{}, err - } - - return repos, err -} - -func (bdw *BoltDB) ProtoGetRepoMeta(repo string) (mTypes.RepoMetadata2, error) { - var protoRepoMeta proto_go.ProtoRepoMeta - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - - // object not found - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - // object found - err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) - if err != nil { - return err - } - - return nil - }) - - return mConvert.GetRepoMeta(&protoRepoMeta), err -} - -func (bdw *BoltDB) ProtoGetFullRepoMeta(ctx context.Context, repo string) (mTypes.FullRepoMetadata, error) { - var protoRepoMeta proto_go.ProtoRepoMeta - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - userBookmarks := getUserBookmarks(ctx, tx) - userStars := getUserStars(ctx, tx) - - repoMetaBlob := buck.Get([]byte(repo)) - - // object not found - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - // object found - err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) - if err != nil { - return err - } - - delete(protoRepoMeta.Tags, "") - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repo) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, repo) - - return nil - }) - - return mConvert.GetFullRepoMeta(&protoRepoMeta), err -} - -func (bdw *BoltDB) ProtoGetImageData(digest godigest.Digest) (mTypes.ImageData2, error) { - imageData := mTypes.ImageData2{} - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - imageBuck := tx.Bucket([]byte(ProtoImageDataBuck)) - - protoImageData, err := fetchProtoImageData(imageBuck, digest.String()) - if err != nil { - return err - } - - if protoImageData.MediaType == ispec.MediaTypeImageIndex { - for i, manifest := range protoImageData.Manifests { - manifestDigest := manifest.Digest - - imageManifestData, err := fetchProtoImageData(imageBuck, manifestDigest) - if err != nil { - return err - } - - protoImageData.Manifests[i] = imageManifestData.Manifests[0] - } - } - - imageData = mConvert.GetImageData(protoImageData) - - return nil - }) - - return imageData, err -} - -func (bdw *BoltDB) ProtoGetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata2) bool, -) ([]mTypes.RepoMetadata2, error) { - foundRepos := []mTypes.RepoMetadata2{} - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - cursor := buck.Cursor() - - for repoName, repoMetaBlob := cursor.First(); repoName != nil; repoName, repoMetaBlob = cursor.Next() { - if ok, err := reqCtx.RepoIsUserAvailable(ctx, string(repoName)); !ok || err != nil { - continue - } - - protoRepoMeta := proto_go.ProtoRepoMeta{} - - err := proto.Unmarshal(repoMetaBlob, &protoRepoMeta) - if err != nil { - return err - } - - repoMeta := mConvert.GetRepoMeta(&protoRepoMeta) - - if filter(repoMeta) { - foundRepos = append(foundRepos, repoMeta) - } - } - - return nil - }) - - return foundRepos, err -} - -func (bdw *BoltDB) ProtoAddManifestSignature(repo string, signedManifestDigest godigest.Digest, - sygMeta mTypes.SignatureMetadata, -) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - - if len(repoMetaBlob) == 0 { - var err error - // create a new object - repoMeta := proto_go.ProtoRepoMeta{ - Name: repo, - Tags: map[string]*proto_go.TagDescriptor{}, - Signatures: map[string]*proto_go.ManifestSignatures{ - signedManifestDigest.String(): { - Map: map[string]*proto_go.SignaturesInfo{ - sygMeta.SignatureType: { - List: []*proto_go.SignatureInfo{ - { - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), - }, - }, - }, - }, - }, - }, - Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, - } - - repoMetaBlob, err = proto.Marshal(&repoMeta) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - } - - protoRepoMeta := &proto_go.ProtoRepoMeta{} - - err := proto.Unmarshal(repoMetaBlob, protoRepoMeta) - if err != nil { - return err - } - - var ( - manifestSignatures *proto_go.ManifestSignatures - found bool - ) - - if manifestSignatures, found = protoRepoMeta.Signatures[signedManifestDigest.String()]; !found { - manifestSignatures = &proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} - } - - signatureSlice := &proto_go.SignaturesInfo{List: []*proto_go.SignatureInfo{}} - if sigSlice, found := manifestSignatures.Map[sygMeta.SignatureType]; found { - signatureSlice = sigSlice - } - - if !common.ProtoSignatureAlreadyExists(signatureSlice.List, sygMeta) { - switch sygMeta.SignatureType { - case zcommon.NotationSignature: - signatureSlice.List = append(signatureSlice.List, &proto_go.SignatureInfo{ - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), - }) - case zcommon.CosignSignature: - signatureSlice.List = []*proto_go.SignatureInfo{{ - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), - }} - } - } - - manifestSignatures.Map[sygMeta.SignatureType] = signatureSlice - protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures - - repoMetaBlob, err = proto.Marshal(protoRepoMeta) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoDeleteSignature(repo string, signedManifestDigest godigest.Digest, sigMeta mTypes.SignatureMetadata) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrManifestMetaNotFound - } - - protoRepoMeta := proto_go.ProtoRepoMeta{} - - err := json.Unmarshal(repoMetaBlob, &protoRepoMeta) - if err != nil { - return err - } - - manifestSignatures, found := protoRepoMeta.Signatures[signedManifestDigest.String()] - if !found { - return zerr.ErrManifestMetaNotFound - } - - signatureSlice := manifestSignatures.Map[sigMeta.SignatureType] - - newSignatureSlice := make([]*proto_go.SignatureInfo, 0, len(signatureSlice.List)) - - for _, sigInfo := range signatureSlice.List { - if sigInfo.SignatureManifestDigest != sigMeta.SignatureDigest { - newSignatureSlice = append(newSignatureSlice, sigInfo) - } - } - - manifestSignatures.Map[sigMeta.SignatureType] = &proto_go.SignaturesInfo{List: newSignatureSlice} - - protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures - - repoMetaBlob, err = proto.Marshal(&protoRepoMeta) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoIncrementRepoStars(repo string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - var repoMeta proto_go.ProtoRepoMeta - - err := proto.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - repoMeta.Stars++ - - repoMetaBlob, err = proto.Marshal(&repoMeta) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoDecrementRepoStars(repo string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - var repoMeta proto_go.ProtoRepoMeta - - err := proto.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - if repoMeta.Stars == 0 { - return nil - } - - repoMeta.Stars-- - - repoMetaBlob, err = proto.Marshal(&repoMeta) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoGetRepoStars(repo string) (int, error) { - stars := 0 - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - buck.Get([]byte(repo)) - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - var repoMeta proto_go.ProtoRepoMeta - - err := proto.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - stars = int(repoMeta.Stars) - - return nil - }) - - return stars, err -} - -func (bdw *BoltDB) ProtoDeleteRepoTag(repo string, tag string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - - // object not found - if repoMetaBlob == nil { - return nil - } - - // object found - var repoMeta proto_go.ProtoRepoMeta - - err := proto.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - delete(repoMeta.Tags, tag) - - repoMetaBlob, err = proto.Marshal(&repoMeta) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoGetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMetadata2, error) { - var repoMeta proto_go.ProtoRepoMeta - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - userBookmarks := getUserBookmarks(ctx, tx) - userStars := getUserStars(ctx, tx) - - repoMetaBlob := buck.Get([]byte(repo)) - - // object not found - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - // object found - err := proto.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repo) - repoMeta.IsStarred = zcommon.Contains(userStars, repo) - - return nil - }) - - return mConvert.GetRepoMeta(&repoMeta), err -} - -func (bdw *BoltDB) ProtoSetRepoMeta(repo string, repoMeta mTypes.RepoMetadata2) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMeta.Name = repo - - repoMetaBlob, err := proto.Marshal(mConvert.GetProtoRepoMeta(repoMeta)) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoGetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string) ([]mTypes.ReferrerInfo, error) { - referrersInfoResult := []mTypes.ReferrerInfo{} - - err := bdw.DB.View(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - if len(repoMetaBlob) == 0 { - return zerr.ErrRepoMetaNotFound - } - - var repoMeta proto_go.ProtoRepoMeta - - err := proto.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - referrersInfo := repoMeta.Referrers[referredDigest.String()].List - - for i := range referrersInfo { - if !common.MatchesArtifactTypes(referrersInfo[i].ArtifactType, artifactTypes) { - continue - } - - referrersInfoResult = append(referrersInfoResult, mTypes.ReferrerInfo{ - Digest: referrersInfo[i].Digest, - MediaType: referrersInfo[i].MediaType, - ArtifactType: referrersInfo[i].ArtifactType, - Size: int(referrersInfo[i].Size), - Annotations: referrersInfo[i].Annotations, - }) - } - - return nil - }) - - return referrersInfoResult, err -} - -func (bdw *BoltDB) ProtoIncrementImageDownloads(repo string, reference string) error { - err := bdw.DB.Update(func(tx *bbolt.Tx) error { - buck := tx.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := buck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrManifestMetaNotFound - } - - var repoMeta proto_go.ProtoRepoMeta - - err := proto.Unmarshal(repoMetaBlob, &repoMeta) - if err != nil { - return err - } - - manifestDigest := reference - - if !common.ReferenceIsDigest(reference) { - // search digest for tag - descriptor, found := repoMeta.Tags[reference] - - if !found { - return zerr.ErrManifestMetaNotFound - } - - manifestDigest = descriptor.Digest - } - - manifestStatistics := repoMeta.Statistics[manifestDigest] - manifestStatistics.DownloadCount++ - repoMeta.Statistics[manifestDigest] = manifestStatistics - - repoMetaBlob, err = proto.Marshal(&repoMeta) - if err != nil { - return err - } - - return buck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func (bdw *BoltDB) ProtoUpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { - err := bdw.DB.Update(func(transaction *bbolt.Tx) error { - imgTrustStore := bdw.ImageTrustStore() - - if imgTrustStore == nil { - return nil - } - - // get ManifestData of signed manifest - imageDataBuck := transaction.Bucket([]byte(ProtoImageDataBuck)) - idBlob := imageDataBuck.Get([]byte(manifestDigest)) - - if len(idBlob) != 0 { - // manifest meta not found, updating signatures with details about validity and author will not be performed - return nil - } - - protoImageData := proto_go.ImageData{} - - err := proto.Unmarshal(idBlob, &protoImageData) - if err != nil { - return err - } - - // update signatures with details about validity and author - repoBuck := transaction.Bucket([]byte(ProtoRepoMetaBuck)) - - repoMetaBlob := repoBuck.Get([]byte(repo)) - if repoMetaBlob == nil { - return zerr.ErrRepoMetaNotFound - } - - protoRepoMeta := proto_go.ProtoRepoMeta{} - - err = proto.Unmarshal(repoMetaBlob, &protoRepoMeta) - if err != nil { - return err - } - - manifestSignatures := proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} - for sigType, sigs := range protoRepoMeta.Signatures[manifestDigest.String()].Map { - signaturesInfo := []*proto_go.SignatureInfo{} - - for _, sigInfo := range sigs.List { - layersInfo := []*proto_go.LayersInfo{} - - for _, layerInfo := range sigInfo.LayersInfo { - author, date, isTrusted, _ := imgTrustStore.ProtoVerifySignature(sigType, layerInfo.LayerContent, - layerInfo.SignatureKey, manifestDigest, mConvert.GetImageData(&protoImageData), repo) - - if isTrusted { - layerInfo.Signer = author - } - - if !date.IsZero() { - layerInfo.Signer = author - layerInfo.Date = timestamppb.New(date) - } - - layersInfo = append(layersInfo, layerInfo) - } - - signaturesInfo = append(signaturesInfo, &proto_go.SignatureInfo{ - SignatureManifestDigest: sigInfo.SignatureManifestDigest, - LayersInfo: layersInfo, - }) - } - - manifestSignatures.Map[sigType] = &proto_go.SignaturesInfo{List: signaturesInfo} - } - - protoRepoMeta.Signatures[manifestDigest.String()] = &manifestSignatures - - repoMetaBlob, err = proto.Marshal(&protoRepoMeta) - if err != nil { - return err - } - - return repoBuck.Put([]byte(repo), repoMetaBlob) - }) - - return err -} - -func ref[T any](input T) *T { - ref := input - - return &ref -} diff --git a/pkg/meta/common/common.go b/pkg/meta/common/common.go index 2c989b7bad..cd8a4e62a5 100644 --- a/pkg/meta/common/common.go +++ b/pkg/meta/common/common.go @@ -1,8 +1,6 @@ package common import ( - "encoding/json" - "fmt" "strings" "time" @@ -16,24 +14,6 @@ import ( mTypes "zotregistry.io/zot/pkg/meta/types" ) -func UpdateManifestMeta(repoMeta mTypes.RepoMetadata, manifestDigest godigest.Digest, - manifestMeta mTypes.ManifestMetadata, -) mTypes.RepoMetadata { - updatedRepoMeta := repoMeta - - updatedStatistics := repoMeta.Statistics[manifestDigest.String()] - updatedStatistics.DownloadCount = manifestMeta.DownloadCount - updatedRepoMeta.Statistics[manifestDigest.String()] = updatedStatistics - - if manifestMeta.Signatures == nil { - manifestMeta.Signatures = mTypes.ManifestSignatures{} - } - - updatedRepoMeta.Signatures[manifestDigest.String()] = manifestMeta.Signatures - - return updatedRepoMeta -} - func SignatureAlreadyExists(signatureSlice []mTypes.SignatureInfo, sm mTypes.SignatureMetadata) bool { for _, sigInfo := range signatureSlice { if sm.SignatureDigest == sigInfo.SignatureManifestDigest { @@ -87,7 +67,7 @@ const ( ) // RankRepoName associates a rank to a given repoName given a searchText. -// The imporance of the value grows inversly proportional to the int value it has. +// The importance of the value grows inversely proportional to the int value it has. // For example: rank(1) > rank(10) > rank(100)... func RankRepoName(searchText string, repoName string) int { searchText = strings.Trim(searchText, "/") @@ -102,7 +82,7 @@ func RankRepoName(searchText string, repoName string) int { return perfectMatchPriority } - // searchText containst just 1 diretory name + // searchText contains just 1 directory name if len(searchTextSlice) == 1 { lastNameInRepoPath := repoNameSlice[len(repoNameSlice)-1] @@ -170,21 +150,6 @@ func GetRepoTag(searchText string) (string, string, error) { return repo, tag, nil } -func GetReferredSubject(descriptorBlob []byte) (godigest.Digest, bool) { - var manifest ispec.Manifest - - err := json.Unmarshal(descriptorBlob, &manifest) - if err != nil { - return "", false - } - - if manifest.Subject == nil || manifest.Subject.Digest.String() == "" { - return "", false - } - - return manifest.Subject.Digest, true -} - func MatchesArtifactTypes(descriptorMediaType string, artifactTypes []string) bool { if len(artifactTypes) == 0 { return true @@ -221,149 +186,13 @@ func CheckImageLastUpdated(repoLastUpdated time.Time, isSigned bool, noImageChec return repoLastUpdated, noImageChecked, isSigned } -func FilterDataByRepo(foundRepos []mTypes.RepoMetadata, manifestMetadataMap map[string]mTypes.ManifestMetadata, - indexDataMap map[string]mTypes.IndexData, -) (map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - var ( - foundManifestMetadataMap = make(map[string]mTypes.ManifestMetadata) - foundindexDataMap = make(map[string]mTypes.IndexData) - ) - - // keep just the manifestMeta we need - for _, repoMeta := range foundRepos { - for _, descriptor := range repoMeta.Tags { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - foundManifestMetadataMap[descriptor.Digest] = manifestMetadataMap[descriptor.Digest] - case ispec.MediaTypeImageIndex: - indexData := indexDataMap[descriptor.Digest] - - var indexContent ispec.Index - - err := json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("metadb: error while getting manifest data for digest %s %w", descriptor.Digest, err) - } - - for _, manifestDescriptor := range indexContent.Manifests { - manifestDigest := manifestDescriptor.Digest.String() - - foundManifestMetadataMap[manifestDigest] = manifestMetadataMap[manifestDigest] - } - - foundindexDataMap[descriptor.Digest] = indexData - default: - continue - } - } - } - - return foundManifestMetadataMap, foundindexDataMap, nil -} - -func FetchDataForRepos(metaDB mTypes.MetaDB, foundRepos []mTypes.RepoMetadata, -) (map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - foundManifestMetadataMap := map[string]mTypes.ManifestMetadata{} - foundIndexDataMap := map[string]mTypes.IndexData{} - - for idx := range foundRepos { - for _, descriptor := range foundRepos[idx].Tags { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestData, err := metaDB.GetManifestData(godigest.Digest(descriptor.Digest)) - if err != nil { - return map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, err - } - - foundManifestMetadataMap[descriptor.Digest] = mTypes.ManifestMetadata{ - ManifestBlob: manifestData.ManifestBlob, - ConfigBlob: manifestData.ConfigBlob, - } - case ispec.MediaTypeImageIndex: - indexData, err := metaDB.GetIndexData(godigest.Digest(descriptor.Digest)) - if err != nil { - return map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, err - } - - var indexContent ispec.Index - - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{}, - fmt.Errorf("metadb: error while getting index data for digest %s %w", descriptor.Digest, err) - } - - for _, manifestDescriptor := range indexContent.Manifests { - manifestDigest := manifestDescriptor.Digest.String() - - manifestData, err := metaDB.GetManifestData(manifestDescriptor.Digest) - if err != nil { - return map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, err - } - - foundManifestMetadataMap[manifestDigest] = mTypes.ManifestMetadata{ - ManifestBlob: manifestData.ManifestBlob, - ConfigBlob: manifestData.ConfigBlob, - } - } - - foundIndexDataMap[descriptor.Digest] = indexData - } - } - } - - return foundManifestMetadataMap, foundIndexDataMap, nil -} - -// FindMediaTypeForDigest will look into the buckets for a certain digest. Depending on which bucket that -// digest is found the corresponding mediatype is returned. -func FindMediaTypeForDigest(metaDB mTypes.MetaDB, digest godigest.Digest) (bool, string) { - _, err := metaDB.GetManifestData(digest) - if err == nil { - return true, ispec.MediaTypeImageManifest - } - - _, err = metaDB.GetIndexData(digest) - if err == nil { - return true, ispec.MediaTypeImageIndex - } - - return false, "" -} - -func GetImageDescriptor(metaDB mTypes.MetaDB, repo, tag string) (mTypes.Descriptor, error) { - repoMeta, err := metaDB.GetRepoMeta(repo) - if err != nil { - return mTypes.Descriptor{}, err - } - - imageDescriptor, ok := repoMeta.Tags[tag] - if !ok { - return mTypes.Descriptor{}, zerr.ErrTagMetaNotFound - } - - return imageDescriptor, nil -} - -func InitializeImageConfig(blob []byte) ispec.Image { - var configContent ispec.Image - - err := json.Unmarshal(blob, &configContent) - if err != nil { - return ispec.Image{} - } - - return configContent -} - -func GetUpdatedRepoMeta(repoMeta *proto_go.ProtoRepoMeta, repoBlobs *proto_go.RepoBlobs, ref string, - imageData mTypes.ImageData2, -) (*proto_go.ProtoRepoMeta, *proto_go.RepoBlobs, error) { +func AddImageDataToRepoMeta(repoMeta *proto_go.RepoMeta, repoBlobs *proto_go.RepoBlobs, reference string, + imageData mTypes.ImageData, +) (*proto_go.RepoMeta, *proto_go.RepoBlobs, error) { switch imageData.MediaType { case ispec.MediaTypeImageManifest: manifestData := imageData.Manifests[0] + vendor := GetVendor(manifestData.Manifest.Annotations) if vendor == "" { vendor = GetVendor(manifestData.Manifest.Annotations) @@ -374,31 +203,33 @@ func GetUpdatedRepoMeta(repoMeta *proto_go.ProtoRepoMeta, repoBlobs *proto_go.Re vendors = append(vendors, vendor) } - platforms := []*proto_go.Platform{getProtoPlatform(&imageData.Manifests[0].ConfigContent.Platform)} - if platforms[0].Os == "" && platforms[0].Arch == "" { + platforms := []*proto_go.Platform{getProtoPlatform(&manifestData.Config.Platform)} + if platforms[0].OS == "" && platforms[0].Architecture == "" { platforms = []*proto_go.Platform{} } - subBlobs := []string{manifestData.Config.Digest.String()} - repoBlobs.Blobs[manifestData.Config.Digest.String()] = &proto_go.BlobInfo{ - Size: manifestData.Config.Size, + subBlobs := []string{manifestData.Manifest.Config.Digest.String()} + repoBlobs.Blobs[manifestData.Manifest.Config.Digest.String()] = &proto_go.BlobInfo{ + Size: manifestData.Manifest.Config.Size, } - for _, layer := range manifestData.Layers { + for _, layer := range manifestData.Manifest.Layers { subBlobs = append(subBlobs, layer.Digest.String()) repoBlobs.Blobs[layer.Digest.String()] = &proto_go.BlobInfo{Size: layer.Size} } + lastUpdated := zcommon.GetImageLastUpdated(manifestData.Config) + repoBlobs.Blobs[imageData.Digest.String()] = &proto_go.BlobInfo{ Size: imageData.Size, Vendors: vendors, Platforms: platforms, SubBlobs: subBlobs, - LastUpdated: mConvert.GetProtoTime(manifestData.ConfigContent.Created), + LastUpdated: mConvert.GetProtoTime(&lastUpdated), } case ispec.MediaTypeImageIndex: subBlobs := []string{} - for _, manifest := range imageData.Manifests { + for _, manifest := range imageData.Index.Manifests { subBlobs = append(subBlobs, manifest.Digest.String()) } @@ -408,27 +239,83 @@ func GetUpdatedRepoMeta(repoMeta *proto_go.ProtoRepoMeta, repoBlobs *proto_go.Re } } + // update info only when a tag is added + if zcommon.IsDigest(reference) { + return repoMeta, repoBlobs, nil + } + + size, platforms, vendors := recalculateAggregateFields(repoMeta, repoBlobs) + repoMeta.Vendors = vendors + repoMeta.Platforms = platforms + repoMeta.Size = int32(size) + imageBlobInfo := repoBlobs.Blobs[imageData.Digest.String()] + repoMeta.LastUpdatedImage = mConvert.GetProtoEarlierUpdatedImage(repoMeta.LastUpdatedImage, + &proto_go.RepoLastUpdatedImage{ + LastUpdated: imageBlobInfo.LastUpdated, + MediaType: imageData.MediaType, + Digest: imageData.Digest.String(), + Tag: reference, + }) + + return repoMeta, repoBlobs, nil +} + +func RemoveImageFromRepoMeta(repoMeta *proto_go.RepoMeta, repoBlobs *proto_go.RepoBlobs, ref string, +) (*proto_go.RepoMeta, *proto_go.RepoBlobs, error) { + var updatedLastImage *proto_go.RepoLastUpdatedImage - repoMeta.Vendors = mConvert.AddVendors(repoMeta.Vendors, imageBlobInfo.Vendors) - repoMeta.Platforms = mConvert.AddProtoPlatforms(repoMeta.Platforms, imageBlobInfo.Platforms) - repoMeta.Size = int32(getRepoSize(repoMeta, repoBlobs)) - - if zcommon.IsTag(ref) { - repoMeta.LastUpdatedImage = mConvert.GetProtoEarlierUpdatedImage(repoMeta.LastUpdatedImage, - &proto_go.RepoLastUpdatedImage{ - LastUpdated: imageBlobInfo.LastUpdated, - MediaType: imageData.MediaType, - Digest: imageData.Digest.String(), - Tag: ref, - }) + updatedBlobs := map[string]*proto_go.BlobInfo{} + updatedSize := int64(0) + updatedVendors := []string{} + updatedPlatforms := []*proto_go.Platform{} + + for tag, descriptor := range repoMeta.Tags { + if descriptor.Digest == "" { + continue + } + + queue := []string{descriptor.Digest} + + mConvert.GetProtoEarlierUpdatedImage(updatedLastImage, &proto_go.RepoLastUpdatedImage{ + LastUpdated: repoBlobs.Blobs[descriptor.Digest].LastUpdated, + MediaType: descriptor.MediaType, + Digest: descriptor.Digest, + Tag: tag, + }) + + for len(queue) > 0 { + currentBlob := queue[0] + queue = queue[1:] + + if _, found := updatedBlobs[currentBlob]; !found { + blobInfo := repoBlobs.Blobs[currentBlob] + + updatedBlobs[currentBlob] = blobInfo + updatedSize += blobInfo.Size + updatedVendors = mConvert.AddVendors(updatedVendors, blobInfo.Vendors) + updatedPlatforms = mConvert.AddProtoPlatforms(updatedPlatforms, blobInfo.Platforms) + + queue = append(queue, blobInfo.SubBlobs...) + } + } } + repoMeta.Size = int32(updatedSize) + repoMeta.Vendors = updatedVendors + repoMeta.Platforms = updatedPlatforms + repoMeta.LastUpdatedImage = updatedLastImage + + repoBlobs.Blobs = updatedBlobs + return repoMeta, repoBlobs, nil } -func getRepoSize(repoMeta *proto_go.ProtoRepoMeta, repoBlobs *proto_go.RepoBlobs) int64 { +func recalculateAggregateFields(repoMeta *proto_go.RepoMeta, repoBlobs *proto_go.RepoBlobs, +) (int64, []*proto_go.Platform, []string) { size := int64(0) + platforms := []*proto_go.Platform{} + vendors := []string{} blobsMap := map[string]struct{}{} for _, descriptor := range repoMeta.Tags { @@ -444,15 +331,21 @@ func getRepoSize(repoMeta *proto_go.ProtoRepoMeta, repoBlobs *proto_go.RepoBlobs if _, found := blobsMap[currentBlob]; !found { blobInfo := repoBlobs.Blobs[currentBlob] + if blobInfo == nil { + continue + } + blobsMap[currentBlob] = struct{}{} size += blobInfo.Size + vendors = mConvert.AddVendors(vendors, blobInfo.Vendors) + platforms = mConvert.AddProtoPlatforms(platforms, blobInfo.Platforms) queue = append(queue, blobInfo.SubBlobs...) } } } - return size + return size, platforms, vendors } func getProtoPlatform(platform *ispec.Platform) *proto_go.Platform { @@ -461,8 +354,8 @@ func getProtoPlatform(platform *ispec.Platform) *proto_go.Platform { } return &proto_go.Platform{ - Arch: getArch(platform.Architecture, platform.Variant), - Os: platform.OS, + Architecture: getArch(platform.Architecture, platform.Variant), + OS: platform.OS, } } diff --git a/pkg/meta/common/common_test.go b/pkg/meta/common/common_test.go index 6c7c9efe94..01ffbf3978 100644 --- a/pkg/meta/common/common_test.go +++ b/pkg/meta/common/common_test.go @@ -5,23 +5,15 @@ import ( "testing" "time" - "github.com/opencontainers/go-digest" - ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" "zotregistry.io/zot/pkg/meta/common" mTypes "zotregistry.io/zot/pkg/meta/types" - "zotregistry.io/zot/pkg/test/mocks" ) var ErrTestError = errors.New("test error") func TestUtils(t *testing.T) { - Convey("GetReferredSubject", t, func() { - _, err := common.GetReferredSubject([]byte("bad json")) - So(err, ShouldNotBeNil) - }) - Convey("MatchesArtifactTypes", t, func() { res := common.MatchesArtifactTypes("", nil) So(res, ShouldBeTrue) @@ -116,145 +108,4 @@ func TestUtils(t *testing.T) { So(res, ShouldEqual, false) }) - - Convey("FilterDataByRepo", t, func() { - Convey("Functionality", func() { - _, _, err := common.FilterDataByRepo( - []mTypes.RepoMetadata{{ - Tags: map[string]mTypes.Descriptor{ - "manifest": { - Digest: "manifestDigest", - MediaType: ispec.MediaTypeImageManifest, - }, - "index": { - Digest: "indexDigest", - MediaType: ispec.MediaTypeImageIndex, - }, - "rand": { - Digest: "randDigest", - MediaType: "rand", - }, - }, - }}, - map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{ - "indexDigest": { - IndexBlob: []byte(`{ - "manifests": [ - { - "digest": "manifestDigest" - } - ] - }`), - }, - }, - ) - So(err, ShouldBeNil) - }) - Convey("Errors", func() { - // Unmarshal index data error - _, _, err := common.FilterDataByRepo( - []mTypes.RepoMetadata{{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: "indexDigest", - MediaType: ispec.MediaTypeImageIndex, - }, - }, - }}, - map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{ - "indexDigest": { - IndexBlob: []byte("bad blob"), - }, - }, - ) - - So(err, ShouldNotBeNil) - }) - }) - - Convey("FetchDataForRepos", t, func() { - Convey("Errors", func() { - // Unmarshal index data error - _, _, err := common.FetchDataForRepos( - mocks.MetaDBMock{ - GetIndexDataFn: func(indexDigest digest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{ - IndexBlob: []byte("bad blob"), - }, nil - }, - }, - []mTypes.RepoMetadata{{ - Tags: map[string]mTypes.Descriptor{ - "tag": { - Digest: "indexDigest", - MediaType: ispec.MediaTypeImageIndex, - }, - }, - }}, - ) - So(err, ShouldNotBeNil) - }) - }) -} - -func TestFetchDataForRepos(t *testing.T) { - Convey("GetReferredSubject", t, func() { - mockMetaDB := mocks.MetaDBMock{} - - Convey("GetManifestData errors", func() { - mockMetaDB.GetManifestDataFn = func(manifestDigest digest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, ErrTestError - } - - _, _, err := common.FetchDataForRepos(mockMetaDB, []mTypes.RepoMetadata{ - { - Tags: map[string]mTypes.Descriptor{ - "tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageManifest}, - }, - }, - }) - So(err, ShouldNotBeNil) - }) - - Convey("GetIndexData errors", func() { - mockMetaDB.GetIndexDataFn = func(indexDigest digest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{}, ErrTestError - } - - _, _, err := common.FetchDataForRepos(mockMetaDB, []mTypes.RepoMetadata{ - { - Tags: map[string]mTypes.Descriptor{ - "tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageIndex}, - }, - }, - }) - So(err, ShouldNotBeNil) - }) - - Convey("GetIndexData ok, GetManifestData errors", func() { - mockMetaDB.GetIndexDataFn = func(indexDigest digest.Digest) (mTypes.IndexData, error) { - return mTypes.IndexData{ - IndexBlob: []byte(`{ - "manifests": [ - {"digest": "dig1"} - ] - }`), - }, nil - } - mockMetaDB.GetManifestDataFn = func(manifestDigest digest.Digest) (mTypes.ManifestData, error) { - return mTypes.ManifestData{}, ErrTestError - } - - _, _, err := common.FetchDataForRepos(mockMetaDB, []mTypes.RepoMetadata{ - { - Tags: map[string]mTypes.Descriptor{ - "tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageIndex}, - }, - }, - }) - So(err, ShouldNotBeNil) - }) - }) } diff --git a/pkg/meta/convert/convert.go b/pkg/meta/convert/convert.go index 5637657384..50bb8f2b68 100644 --- a/pkg/meta/convert/convert.go +++ b/pkg/meta/convert/convert.go @@ -4,171 +4,362 @@ import ( "time" godigest "github.com/opencontainers/go-digest" + "github.com/opencontainers/image-spec/specs-go" ispec "github.com/opencontainers/image-spec/specs-go/v1" "google.golang.org/protobuf/types/known/timestamppb" + "zotregistry.io/zot/pkg/common" "zotregistry.io/zot/pkg/meta/proto_go" - mTypes "zotregistry.io/zot/pkg/meta/types" ) -func GetImageManifestData(manifestContent ispec.Manifest, configContent ispec.Image, size int64, digest godigest.Digest) mTypes.ImageData2 { - return mTypes.ImageData2{ - MediaType: ispec.MediaTypeImageManifest, - Digest: digest, - Size: size, - Manifests: []mTypes.ManifestData2{ - { - Digest: digest, - Size: size, - ConfigContent: configContent, - Manifest: manifestContent, - }, - }, +func GetHistory(history []*proto_go.History) []ispec.History { + if history == nil { + return nil + } + + results := make([]ispec.History, 0, len(history)) + + for _, his := range history { + results = append(results, ispec.History{ + Created: ref(his.Created.AsTime()), + CreatedBy: deref(his.CreatedBy, ""), + Author: deref(his.Author, ""), + Comment: deref(his.Comment, ""), + EmptyLayer: deref(his.EmptyLayer, false), + }) + } + + return results +} + +func GetImageArtifactType(imageData *proto_go.ImageData) string { + switch imageData.MediaType { + case ispec.MediaTypeImageManifest: + manifestArtifactType := deref(imageData.Manifests[0].Manifest.ArtifactType, "") + if manifestArtifactType != "" { + return manifestArtifactType + } + + return imageData.Manifests[0].Manifest.Config.MediaType + case ispec.MediaTypeImageIndex: + return deref(imageData.Index.Index.ArtifactType, "") + default: + return "" } } -func GetImageIndexData(indexContent ispec.Index, size int64, digest godigest.Digest) mTypes.ImageData2 { - return mTypes.ImageData2{ - MediaType: ispec.MediaTypeImageIndex, - Index: &indexContent, - Manifests: getManifests(indexContent.Manifests), - Size: size, - Digest: digest, +func GetImageManifestSize(imageData *proto_go.ImageData) int64 { + switch imageData.MediaType { + case ispec.MediaTypeImageManifest: + return imageData.Manifests[0].Size + case ispec.MediaTypeImageIndex: + return imageData.Index.Size + default: + return 0 + } +} + +func GetImageDigest(imageData *proto_go.ImageData) godigest.Digest { + switch imageData.MediaType { + case ispec.MediaTypeImageManifest: + return godigest.Digest(imageData.Manifests[0].Digest) + case ispec.MediaTypeImageIndex: + return godigest.Digest(imageData.Index.Digest) + default: + return "" + } +} + +func GetImageDigestStr(imageData *proto_go.ImageData) string { + switch imageData.MediaType { + case ispec.MediaTypeImageManifest: + return imageData.Manifests[0].Digest + case ispec.MediaTypeImageIndex: + return imageData.Index.Digest + default: + return "" + } +} + +func GetImageAnnotations(imageData *proto_go.ImageData) map[string]string { + switch imageData.MediaType { + case ispec.MediaTypeImageManifest: + return imageData.Manifests[0].Manifest.Annotations + case ispec.MediaTypeImageIndex: + return imageData.Index.Index.Annotations + default: + return map[string]string{} } } -func GetProtoImageData(imageData mTypes.ImageData2) *proto_go.ImageData { +func GetImageSubject(imageData *proto_go.ImageData) *ispec.Descriptor { switch imageData.MediaType { case ispec.MediaTypeImageManifest: - if len(imageData.Manifests) == 0 { + if imageData.Manifests[0].Manifest.Subject == nil { return nil } - return GetProtoImageManifestData(imageData.Manifests[0].Manifest, imageData.Manifests[0].ConfigContent, - imageData.Manifests[0].Size, imageData.Manifests[0].Digest.String()) + return GetDescriptorRef(imageData.Manifests[0].Manifest.Subject) case ispec.MediaTypeImageIndex: - if imageData.Index == nil { + if imageData.Index.Index.Subject == nil { return nil } - return GetProtoImageIndexData(*imageData.Index, imageData.Size, imageData.Digest.String()) + return GetDescriptorRef(imageData.Index.Index.Subject) default: + return &ispec.Descriptor{} + } +} + +func GetDescriptorRef(descriptor *proto_go.Descriptor) *ispec.Descriptor { + if descriptor == nil { return nil } + + var platform *ispec.Platform + if descriptor.Platform != nil { + platform = ref(GetPlatform(descriptor.Platform)) + } + + return &ispec.Descriptor{ + MediaType: descriptor.MediaType, + Digest: godigest.Digest(descriptor.Digest), + Size: descriptor.Size, + URLs: descriptor.URLs, + Data: descriptor.Data, + Platform: platform, + ArtifactType: deref(descriptor.ArtifactType, ""), + Annotations: descriptor.Annotations, + } } -func GetProtoImageIndexData(indexContent ispec.Index, size int64, digest string) *proto_go.ImageData { - return &proto_go.ImageData{ - Versioned: &proto_go.Versioned{Schemaversion: int32(indexContent.SchemaVersion)}, - MediaType: ispec.MediaTypeImageIndex, - ArtifacType: common.GetIndexArtifactType(indexContent), - Manifests: getProtoManifests(indexContent.Manifests), - Subject: getProtoDesc(indexContent.Subject), - Annotations: indexContent.Annotations, - Size: size, - Digest: digest, +func GetPlatform(platform *proto_go.Platform) ispec.Platform { + if platform == nil { + return ispec.Platform{} + } + + return ispec.Platform{ + Architecture: platform.Architecture, + OS: platform.OS, } } -func getProtoManifests(descriptors []ispec.Descriptor) []*proto_go.ManifestData { - manifestList := []*proto_go.ManifestData{} +func GetLayers(descriptors []*proto_go.Descriptor) []ispec.Descriptor { + results := make([]ispec.Descriptor, 0, len(descriptors)) - for _, manifest := range descriptors { - manifestList = append(manifestList, &proto_go.ManifestData{ - MediaType: ref(ispec.MediaTypeImageManifest), - Digest: manifest.Digest.String(), + for _, desc := range descriptors { + results = append(results, ispec.Descriptor{ + MediaType: desc.MediaType, + Digest: godigest.Digest(desc.Digest), + Size: desc.Size, }) } - return manifestList + return results } -func getManifests(descriptors []ispec.Descriptor) []mTypes.ManifestData2 { - manifestList := []mTypes.ManifestData2{} +func GetSubject(subj *proto_go.Descriptor) *ispec.Descriptor { + if subj == nil { + return nil + } - for _, manifest := range descriptors { - manifestList = append(manifestList, mTypes.ManifestData2{ - Digest: manifest.Digest, - Size: manifest.Size, + return &ispec.Descriptor{ + MediaType: subj.MediaType, + Digest: godigest.Digest(subj.Digest), + Size: subj.Size, + } +} + +func GetReferrers(refs map[string]*proto_go.ReferrersInfo) map[string][]mTypes.ReferrerInfo { + results := map[string][]mTypes.ReferrerInfo{} + + for digest, ref := range refs { + referrers := []mTypes.ReferrerInfo{} + + for _, dbRef := range ref.List { + referrers = append(referrers, mTypes.ReferrerInfo{ + Digest: dbRef.Digest, + MediaType: dbRef.MediaType, + ArtifactType: dbRef.ArtifactType, + Size: int(dbRef.Size), + Annotations: dbRef.Annotations, + }) + } + + results[digest] = referrers + } + + return results +} + +func GetImageReferrers(refs *proto_go.ReferrersInfo) []mTypes.ReferrerInfo { + if refs == nil { + return []mTypes.ReferrerInfo{} + } + + results := []mTypes.ReferrerInfo{} + + for _, dbRef := range refs.List { + results = append(results, mTypes.ReferrerInfo{ + Digest: dbRef.Digest, + MediaType: dbRef.MediaType, + ArtifactType: dbRef.ArtifactType, + Size: int(dbRef.Size), + Annotations: dbRef.Annotations, }) } - return manifestList + return results +} + +func GetSignatures(sigs map[string]*proto_go.ManifestSignatures) map[string]mTypes.ManifestSignatures { + results := map[string]mTypes.ManifestSignatures{} + + for digest, dbSignatures := range sigs { + imageSignatures := mTypes.ManifestSignatures{} + + for signatureName, signatureInfo := range dbSignatures.Map { + imageSignatures[signatureName] = GetSignaturesInfo(signatureInfo.List) + } + + results[digest] = imageSignatures + } + + return results +} + +func GetImageSignatures(sigs *proto_go.ManifestSignatures) mTypes.ManifestSignatures { + if sigs == nil { + return mTypes.ManifestSignatures{} + } + + results := mTypes.ManifestSignatures{} + + for signatureName, signatureInfo := range sigs.Map { + results[signatureName] = GetSignaturesInfo(signatureInfo.List) + } + + return results } -func GetProtoImageManifestData(manifestContent ispec.Manifest, configContent ispec.Image, size int64, digest string, -) *proto_go.ImageData { - return &proto_go.ImageData{ - Versioned: &proto_go.Versioned{Schemaversion: int32(manifestContent.SchemaVersion)}, - MediaType: ispec.MediaTypeImageManifest, - Manifests: []*proto_go.ManifestData{GetProtoManifestData(manifestContent, configContent, size, digest)}, - ArtifacType: common.GetManifestArtifactType(manifestContent), - Subject: getProtoDesc(manifestContent.Subject), - Annotations: manifestContent.Annotations, - Size: size, - Digest: digest, +func GetSignaturesInfo(sigsInfo []*proto_go.SignatureInfo) []mTypes.SignatureInfo { + results := []mTypes.SignatureInfo{} + + for _, siginfo := range sigsInfo { + results = append(results, mTypes.SignatureInfo{ + SignatureManifestDigest: siginfo.SignatureManifestDigest, + LayersInfo: GetLayersInfo(siginfo.LayersInfo), + }) } + + return results } -func GetProtoManifestData(manifestContent ispec.Manifest, configContent ispec.Image, size int64, digest string, -) *proto_go.ManifestData { - return &proto_go.ManifestData{ - Versioned: &proto_go.Versioned{Schemaversion: int32(manifestContent.SchemaVersion)}, +func GetLayersInfo(layersInfo []*proto_go.LayersInfo) []mTypes.LayerInfo { + results := []mTypes.LayerInfo{} + + for _, layerInfo := range layersInfo { + date := time.Time{} + + if layerInfo.Date != nil { + date = layerInfo.Date.AsTime() + } + + results = append(results, mTypes.LayerInfo{ + LayerDigest: layerInfo.LayerDigest, + LayerContent: layerInfo.LayerContent, + SignatureKey: layerInfo.SignatureKey, + Signer: layerInfo.Signer, + Date: date, + }) + } + + return results +} + +func GetStatisticsMap(stats map[string]*proto_go.DescriptorStatistics) map[string]mTypes.DescriptorStatistics { + results := map[string]mTypes.DescriptorStatistics{} + + for digest, stat := range stats { + results[digest] = mTypes.DescriptorStatistics{ + DownloadCount: int(stat.DownloadCount), + } + } + + return results +} + +func GetImageStatistics(stats *proto_go.DescriptorStatistics) mTypes.DescriptorStatistics { + if stats == nil { + return mTypes.DescriptorStatistics{} + } + + return mTypes.DescriptorStatistics{ + DownloadCount: int(stats.DownloadCount), + } +} + +func GetImageManifestData(manifestContent ispec.Manifest, configContent ispec.Image, size int64, + digest godigest.Digest, +) mTypes.ImageData { + return mTypes.ImageData{ + MediaType: ispec.MediaTypeImageManifest, Digest: digest, - MediaType: ref(ispec.MediaTypeImageManifest), - Config: &proto_go.ConfigData{ - Digest: manifestContent.Config.Digest.String(), - Size: manifestContent.Config.Size, - Created: GetProtoTime(configContent.Created), - Author: &configContent.Author, - Platform: &proto_go.Platform{ - Arch: getArch(configContent.Architecture, configContent.Variant), - Os: configContent.OS, - // Osversion: &configContent.OSVersion, - // Osfeatures: configContent.OSFeatures, - // Variant: &configContent.Variant, - }, - Config: &proto_go.ImageConfig{ - User: configContent.Config.User, - Exposedports: getProtoExposedports(configContent.Config.ExposedPorts), - Env: configContent.Config.Env, - Entrypoint: configContent.Config.Entrypoint, - Cmd: configContent.Config.Cmd, - Volumes: getProtoConfigVolumes(configContent.Config.Volumes), - Workingdir: &configContent.Config.WorkingDir, - Labels: configContent.Config.Labels, - Stopsignal: &configContent.Config.StopSignal, - Argsescaped: configContent.Config.ArgsEscaped, - }, - Rootfs: &proto_go.RootFS{ - Type: configContent.RootFS.Type, - Diffids: getProtoDiffIDs(configContent.RootFS.DiffIDs), + Size: size, + Manifests: []mTypes.ManifestData{ + { + Digest: digest, + Size: size, + Config: configContent, + Manifest: manifestContent, }, - History: getProtoHistory(configContent.History), }, - Size: size, - ArtifactType: ref(common.GetManifestArtifactType(manifestContent)), - Layers: getProtoManifestLayers(manifestContent.Layers), - Subject: getProtoDesc(manifestContent.Subject), - Annotations: manifestContent.Annotations, } } -func getArch(arch string, variant string) string { - if variant != "" { - arch = arch + "/" + variant +func GetImageIndexData(indexContent ispec.Index, size int64, digest godigest.Digest) mTypes.ImageData { + return mTypes.ImageData{ + MediaType: ispec.MediaTypeImageIndex, + Index: &indexContent, + Manifests: GetManifests(indexContent.Manifests), + Size: size, + Digest: digest, } +} - return arch +func GetTags(tags map[string]*proto_go.TagDescriptor) map[string]mTypes.Descriptor { + resultMap := map[string]mTypes.Descriptor{} + + for tag, tagDescriptor := range tags { + resultMap[tag] = mTypes.Descriptor{ + Digest: tagDescriptor.Digest, + MediaType: tagDescriptor.MediaType, + } + } + + return resultMap } -func GetProtoTime(time *time.Time) *timestamppb.Timestamp { - if time == nil { - return nil +func GetManifests(descriptors []ispec.Descriptor) []mTypes.ManifestData { + manifestList := []mTypes.ManifestData{} + + for _, manifest := range descriptors { + manifestList = append(manifestList, mTypes.ManifestData{ + Digest: manifest.Digest, + Size: manifest.Size, + }) + } + + return manifestList +} + +func GetArch(arch string, variant string) string { + if variant != "" { + arch = arch + "/" + variant } - return timestamppb.New(*time) + return arch } func GetTime(time *timestamppb.Timestamp) *time.Time { @@ -179,139 +370,257 @@ func GetTime(time *timestamppb.Timestamp) *time.Time { return ref(time.AsTime()) } -func getProtoManifestLayers(layers []ispec.Descriptor) []*proto_go.Descriptor { - protoLayers := []*proto_go.Descriptor{} +func GetImageMetaFromProto(tag string, protoRepoMeta *proto_go.RepoMeta, protoImageData *proto_go.ImageData, +) mTypes.ImageMeta { + imageData := GetImageData(protoImageData) + imageDigest := imageData.Digest.String() - for _, layer := range layers { - layer := layer + return mTypes.ImageMeta{ + Repo: protoRepoMeta.Name, + Tag: tag, + MediaType: imageData.MediaType, + Digest: imageData.Digest, + Size: imageData.Size, + Index: imageData.Index, + Manifests: GetFullManifestData(protoRepoMeta, imageData.Manifests), + IsStarred: protoRepoMeta.IsStarred, + IsBookmarked: protoRepoMeta.IsBookmarked, - protoLayers = append(protoLayers, getProtoDesc(&layer)) + Referrers: GetImageReferrers(protoRepoMeta.Referrers[imageDigest]), + Statistics: GetImageStatistics(protoRepoMeta.Statistics[imageDigest]), + Signatures: GetImageSignatures(protoRepoMeta.Signatures[imageDigest]), } - - return protoLayers } -func getProtoDesc(descriptor *ispec.Descriptor) *proto_go.Descriptor { - if descriptor == nil { - return nil - } +func GetFullManifestData(protoRepoMeta *proto_go.RepoMeta, manifestData []mTypes.ManifestData, +) []mTypes.ManifestMeta { + results := []mTypes.ManifestMeta{} - return &proto_go.Descriptor{ - MediaType: descriptor.MediaType, - Digest: descriptor.Digest.String(), - Size: descriptor.Size, - Urls: descriptor.URLs, - Annotations: descriptor.Annotations, - Data: descriptor.Data, - Platform: getProtoPlatform(descriptor.Platform), - ArtifactType: &descriptor.ArtifactType, + for i := range manifestData { + results = append(results, mTypes.ManifestMeta{ + ManifestData: manifestData[i], + Referrers: GetImageReferrers(protoRepoMeta.Referrers[manifestData[i].Digest.String()]), + Statistics: GetImageStatistics(protoRepoMeta.Statistics[manifestData[i].Digest.String()]), + Signatures: GetImageSignatures(protoRepoMeta.Signatures[manifestData[i].Digest.String()]), + }) } + + return results } -func getProtoPlatform(platform *ispec.Platform) *proto_go.Platform { - if platform == nil { - return nil +func GetRepoMeta(protoRepoMeta *proto_go.RepoMeta) mTypes.RepoMeta { + repoDownloads := int32(0) + + for _, descriptor := range protoRepoMeta.Tags { + if statistic := protoRepoMeta.Statistics[descriptor.Digest]; statistic != nil { + repoDownloads += statistic.DownloadCount + } } - return &proto_go.Platform{ - Arch: getArch(platform.Architecture, platform.Variant), - Os: platform.OS, - // Osversion: &platform.OSVersion, - // Osfeatures: platform.OSFeatures, - // Variant: &platform.Variant, + return mTypes.RepoMeta{ + Name: protoRepoMeta.Name, + Tags: GetTags(protoRepoMeta.Tags), + Rank: int(protoRepoMeta.Rank), + Size: int64(protoRepoMeta.Size), + Platforms: GetPlatforms(protoRepoMeta.Platforms), + Vendors: protoRepoMeta.Vendors, + IsStarred: protoRepoMeta.IsStarred, + IsBookmarked: protoRepoMeta.IsBookmarked, + StarCount: int(protoRepoMeta.Stars), + DownloadCount: int(repoDownloads), + LastUpdatedImage: GetLastUpdatedImage(protoRepoMeta.LastUpdatedImage), + Statistics: GetStatisticsMap(protoRepoMeta.Statistics), + Signatures: GetSignatures(protoRepoMeta.Signatures), + Referrers: GetReferrers(protoRepoMeta.Referrers), } } -func getProtoHistory(historySlice []ispec.History) []*proto_go.History { - protoHistory := []*proto_go.History{} - - for _, history := range historySlice { - history := history +func GetPlatforms(platforms []*proto_go.Platform) []ispec.Platform { + result := []ispec.Platform{} - protoHistory = append(protoHistory, &proto_go.History{ - Created: GetProtoTime(history.Created), - Createdby: &history.CreatedBy, - Author: &history.Author, - Comment: &history.Comment, - Emptylayer: &history.EmptyLayer, + for i := range platforms { + result = append(result, ispec.Platform{ + OS: platforms[i].OS, + Architecture: platforms[i].Architecture, }) } - return protoHistory + return result } -func getProtoDiffIDs(digests []godigest.Digest) []string { - digestsStr := []string{} +func GetLastUpdated(timestamp *timestamppb.Timestamp) *time.Time { + if timestamp == nil { + return nil + } - for _, digest := range digests { - digestsStr = append(digestsStr, digest.String()) + return ref(timestamp.AsTime()) +} + +func AddProtoPlatforms(platforms []*proto_go.Platform, newPlatforms []*proto_go.Platform) []*proto_go.Platform { + for _, newPlatform := range newPlatforms { + if !ContainsProtoPlatform(platforms, newPlatform) { + platforms = append(platforms, newPlatform) + } } - return digestsStr + return platforms } -func getProtoExposedports(exposedPorts map[string]struct{}) map[string]*proto_go.EmptyMessage { - protoPorts := map[string]*proto_go.EmptyMessage{} +func ContainsProtoPlatform(platforms []*proto_go.Platform, platform *proto_go.Platform) bool { + for i := range platforms { + if platforms[i].OS == platform.OS && platforms[i].Architecture == platform.Architecture { + return true + } + } + + return false +} - for i := range exposedPorts { - protoPorts[i] = &proto_go.EmptyMessage{} +func AddVendors(vendors []string, newVendors []string) []string { + for _, newVendor := range newVendors { + if !common.Contains(vendors, newVendor) { + vendors = append(vendors, newVendor) + } } - return protoPorts + return vendors } -func getProtoConfigVolumes(volumes map[string]struct{}) map[string]*proto_go.EmptyMessage { - protoVolumes := map[string]*proto_go.EmptyMessage{} +func GetLastUpdatedImage(protoLastUpdated *proto_go.RepoLastUpdatedImage) *mTypes.LastUpdatedImage { + if protoLastUpdated == nil { + return nil + } - for i := range volumes { - protoVolumes[i] = &proto_go.EmptyMessage{} + return &mTypes.LastUpdatedImage{ + Descriptor: mTypes.Descriptor{ + Digest: protoLastUpdated.Digest, + MediaType: protoLastUpdated.MediaType, + }, + Tag: protoLastUpdated.Tag, + LastUpdated: GetTime(protoLastUpdated.LastUpdated), } +} + +func GetImageData(dbImageData *proto_go.ImageData) mTypes.ImageData { + imageData := mTypes.ImageData{ + MediaType: dbImageData.MediaType, + Size: GetImageManifestSize(dbImageData), + Digest: GetImageDigest(dbImageData), + } + + if dbImageData.MediaType == ispec.MediaTypeImageIndex { + var subject *ispec.Descriptor + + manifests := make([]ispec.Descriptor, 0, len(dbImageData.Manifests)) + + for _, manifest := range dbImageData.Manifests { + manifests = append(manifests, ispec.Descriptor{ + MediaType: deref(manifest.Manifest.MediaType, ""), + Digest: godigest.Digest(manifest.Digest), + Size: manifest.Size, + }) + } - return protoVolumes + if imgSubject := GetImageSubject(dbImageData); imgSubject != nil { + subject = &ispec.Descriptor{ + MediaType: imgSubject.MediaType, + Digest: imgSubject.Digest, + Size: imgSubject.Size, + } + } + + imageData.Index = &ispec.Index{ + Versioned: specs.Versioned{SchemaVersion: int(dbImageData.Index.Index.Versioned.GetSchemaVersion())}, + MediaType: ispec.MediaTypeImageIndex, + Manifests: manifests, + Subject: subject, + ArtifactType: GetImageArtifactType(dbImageData), + Annotations: GetImageAnnotations(dbImageData), + } + } + + manifestDataList := make([]mTypes.ManifestData, 0, len(dbImageData.Manifests)) + + for _, manifest := range dbImageData.Manifests { + manifestDataList = append(manifestDataList, mTypes.ManifestData{ + Size: manifest.Size, + Digest: godigest.Digest(manifest.Digest), + Manifest: ispec.Manifest{ + Versioned: specs.Versioned{SchemaVersion: int(manifest.Manifest.Versioned.GetSchemaVersion())}, + MediaType: deref(manifest.Manifest.MediaType, ""), + ArtifactType: deref(manifest.Manifest.ArtifactType, ""), + Config: ispec.Descriptor{ + MediaType: manifest.Manifest.Config.MediaType, + Size: manifest.Manifest.Config.Size, + Digest: godigest.Digest(manifest.Manifest.Config.Digest), + }, + Layers: GetLayers(manifest.Manifest.Layers), + Subject: GetSubject(manifest.Manifest.Subject), + Annotations: manifest.Manifest.Annotations, + }, + Config: ispec.Image{ + Created: GetTime(manifest.Config.Created), + Author: deref(manifest.Config.Author, ""), + Platform: GetPlatform(manifest.Config.Platform), + Config: ispec.ImageConfig{ + User: manifest.Config.Config.User, + ExposedPorts: GetExposedPorts(manifest.Config.Config.ExposedPorts), + Env: manifest.Config.Config.Env, + Entrypoint: manifest.Config.Config.Entrypoint, + Cmd: manifest.Config.Config.Cmd, + Volumes: GetConfigVolumes(manifest.Config.Config.Volumes), + WorkingDir: deref(manifest.Config.Config.WorkingDir, ""), + Labels: manifest.Config.Config.Labels, + StopSignal: deref(manifest.Config.Config.StopSignal, ""), + }, + RootFS: ispec.RootFS{ + Type: manifest.Config.RootFS.Type, + DiffIDs: GetDiffIDs(manifest.Config.RootFS.DiffIDs), + }, + History: GetHistory(manifest.Config.History), + }, + }) + } + + imageData.Manifests = manifestDataList + + return imageData } -func GetProtoRefferrerInfo(referrer mTypes.ReferrerInfo) *proto_go.ReferrerInfo { - return &proto_go.ReferrerInfo{ - Digest: referrer.Digest, - MediaType: referrer.MediaType, - ArtifactType: referrer.ArtifactType, - Size: int64(referrer.Size), - Annotations: referrer.Annotations, +func GetExposedPorts(exposedPorts map[string]*proto_go.EmptyMessage) map[string]struct{} { + if exposedPorts == nil { + return nil } + + result := map[string]struct{}{} + + for key := range exposedPorts { + result[key] = struct{}{} + } + + return result } -func GetFullImageData(tag string, protoRepoMeta *proto_go.ProtoRepoMeta, protoImageData *proto_go.ImageData, -) mTypes.FullImageData { - imageData := GetImageData(protoImageData) - imageDigest := imageData.Digest.String() +func GetConfigVolumes(configVolumes map[string]*proto_go.EmptyMessage) map[string]struct{} { + if configVolumes == nil { + return nil + } - return mTypes.FullImageData{ - Repo: protoRepoMeta.Name, - Tag: tag, - MediaType: imageData.MediaType, - Digest: imageData.Digest, - Size: imageData.Size, - Index: imageData.Index, - Manifests: GetFullManifestData(protoRepoMeta, imageData.Manifests), + result := map[string]struct{}{} - Referrers: GetImageReferrers(protoRepoMeta.Referrers[imageDigest]), - Statistics: GetImageStatistics(protoRepoMeta.Statistics[imageDigest]), - Signatures: GetImageSignatures(protoRepoMeta.Signatures[imageDigest]), + for key := range configVolumes { + result[key] = struct{}{} } + + return result } -func GetFullManifestData(protoRepoMeta *proto_go.ProtoRepoMeta, manifestData []mTypes.ManifestData2, -) []mTypes.FullManifestData { - results := []mTypes.FullManifestData{} +func GetDiffIDs(diffIDs []string) []godigest.Digest { + result := make([]godigest.Digest, 0, len(diffIDs)) - for i := range manifestData { - results = append(results, mTypes.FullManifestData{ - ManifestData2: manifestData[i], - Referrers: GetImageReferrers(protoRepoMeta.Referrers[manifestData[i].Digest.String()]), - Statistics: GetImageStatistics(protoRepoMeta.Statistics[manifestData[i].Digest.String()]), - Signatures: GetImageSignatures(protoRepoMeta.Signatures[manifestData[i].Digest.String()]), - }) + for i := range diffIDs { + result = append(result, godigest.Digest(diffIDs[i])) } - return results + return result } diff --git a/pkg/meta/convert/convert2.go b/pkg/meta/convert/convert2.go deleted file mode 100644 index 1f3ed27a96..0000000000 --- a/pkg/meta/convert/convert2.go +++ /dev/null @@ -1,555 +0,0 @@ -package convert - -import ( - "time" - - godigest "github.com/opencontainers/go-digest" - "github.com/opencontainers/image-spec/specs-go" - ispec "github.com/opencontainers/image-spec/specs-go/v1" - "google.golang.org/protobuf/types/known/timestamppb" - zcommon "zotregistry.io/zot/pkg/common" - "zotregistry.io/zot/pkg/meta/proto_go" - - mTypes "zotregistry.io/zot/pkg/meta/types" -) - -func GetRepoMeta(repo *proto_go.ProtoRepoMeta) mTypes.RepoMetadata2 { - return mTypes.RepoMetadata2{ - Name: repo.Name, - Tags: getTags(repo.Tags), - Statistics: GetStatisticsMap(repo.Statistics), - Signatures: GetSignatures(repo.Signatures), - Referrers: GetReferrers(repo.Referrers), - } -} - -func GetProtoRepoMeta(repo mTypes.RepoMetadata2) *proto_go.ProtoRepoMeta { - return &proto_go.ProtoRepoMeta{ - Name: repo.Name, - Tags: getProtoTags(repo.Tags), - Statistics: getProtoStatistics(repo.Statistics), - Signatures: getProtoSignatures(repo.Signatures), - Referrers: getProtoReferrers(repo.Referrers), - } -} - -func GetImageData(dbImageData *proto_go.ImageData) mTypes.ImageData2 { - imageData := mTypes.ImageData2{ - MediaType: dbImageData.MediaType, - Size: dbImageData.Size, - Digest: godigest.Digest(dbImageData.Digest), - } - - if dbImageData.MediaType == ispec.MediaTypeImageIndex { - manifests := make([]ispec.Descriptor, 0, len(dbImageData.Manifests)) - var subject *ispec.Descriptor - - for _, manifest := range dbImageData.Manifests { - manifests = append(manifests, ispec.Descriptor{ - MediaType: deref(manifest.MediaType, ""), - Digest: godigest.Digest(manifest.Digest), - Size: manifest.Size, - }) - } - - if dbImageData.Subject != nil { - subject = &ispec.Descriptor{ - MediaType: dbImageData.Subject.MediaType, - Digest: godigest.Digest(dbImageData.Subject.Digest), - Size: dbImageData.Subject.Size, - } - } - - imageData.Index = &ispec.Index{ - Versioned: specs.Versioned{SchemaVersion: int(dbImageData.Versioned.GetSchemaversion())}, - MediaType: ispec.MediaTypeImageIndex, - ArtifactType: dbImageData.ArtifacType, - Manifests: manifests, - Subject: subject, - Annotations: dbImageData.Annotations, - } - } - - manifestDataList := make([]mTypes.ManifestData2, 0, len(dbImageData.Manifests)) - - for _, manifest := range dbImageData.Manifests { - manifestDataList = append(manifestDataList, mTypes.ManifestData2{ - Size: manifest.Size, - Digest: godigest.Digest(manifest.Digest), - Manifest: ispec.Manifest{ - Versioned: specs.Versioned{SchemaVersion: int(manifest.Versioned.GetSchemaversion())}, - MediaType: deref(manifest.MediaType, ""), - ArtifactType: deref(manifest.ArtifactType, ""), - Config: ispec.Descriptor{ - MediaType: ispec.MediaTypeImageConfig, // TODO add config media type to manifest data in DB - Size: manifest.Config.Size, - Digest: godigest.Digest(manifest.Config.Digest), - }, - Layers: getLayers(manifest.Layers), - Subject: getSubject(manifest.Subject), - Annotations: manifest.Annotations, - }, - ConfigContent: ispec.Image{ - Created: getTime(manifest.Config.Created), - Author: deref(manifest.Config.Author, ""), - Platform: getPlatform(manifest.Config.Platform), - Config: ispec.ImageConfig{}, - RootFS: ispec.RootFS{}, - History: getHistory(manifest.Config.History), - }, - }) - } - - imageData.Manifests = manifestDataList - - return imageData -} - -func getTime(time *timestamppb.Timestamp) *time.Time { - if time == nil { - return nil - } - - return ref(time.AsTime()) -} - -func getHistory(history []*proto_go.History) []ispec.History { - results := make([]ispec.History, 0, len(history)) - - for _, his := range history { - results = append(results, ispec.History{ - Created: ref(his.Created.AsTime()), - CreatedBy: deref(his.Createdby, ""), - Author: deref(his.Author, ""), - Comment: deref(his.Comment, ""), - EmptyLayer: deref(his.Emptylayer, false), - }) - } - - return results -} - -func ref[T any](input T) *T { - ref := input - - return &ref -} - -func deref[T any](pointer *T, defaultVal T) T { - if pointer != nil { - return *pointer - } - - return defaultVal -} - -func getPlatform(platform *proto_go.Platform) ispec.Platform { - if platform == nil { - return ispec.Platform{} - } - - return ispec.Platform{ - Architecture: platform.Arch, - OS: platform.Os, - // OSVersion: deref(platform.Osversion, ""), - // OSFeatures: platform.Osfeatures, - // Variant: deref(platform.Variant, ""), - } -} - -func getLayers(descriptors []*proto_go.Descriptor) []ispec.Descriptor { - results := make([]ispec.Descriptor, 0, len(descriptors)) - - for _, desc := range descriptors { - results = append(results, ispec.Descriptor{ - MediaType: desc.MediaType, - Digest: godigest.Digest(desc.Digest), - Size: desc.Size, - }) - } - - return results -} - -func getSubject(subj *proto_go.Descriptor) *ispec.Descriptor { - if subj == nil { - return nil - } - - return &ispec.Descriptor{ - MediaType: subj.MediaType, - Digest: godigest.Digest(subj.Digest), - Size: subj.Size, - } -} - -func GetStatisticsMap(stats map[string]*proto_go.DescriptorStatistics) map[string]mTypes.DescriptorStatistics { - results := map[string]mTypes.DescriptorStatistics{} - - for digest, stat := range stats { - results[digest] = mTypes.DescriptorStatistics{ - DownloadCount: int(stat.DownloadCount), - } - } - - return results -} - -func GetImageStatistics(stats *proto_go.DescriptorStatistics) mTypes.DescriptorStatistics { - return mTypes.DescriptorStatistics{ - DownloadCount: int(stats.DownloadCount), - } -} - -func getProtoStatistics(stats map[string]mTypes.DescriptorStatistics) map[string]*proto_go.DescriptorStatistics { - results := map[string]*proto_go.DescriptorStatistics{} - - for digest, stat := range stats { - results[digest] = &proto_go.DescriptorStatistics{ - DownloadCount: int32(stat.DownloadCount), - } - } - - return results -} - -func GetReferrers(refs map[string]*proto_go.ReferrersInfo) map[string][]mTypes.ReferrerInfo { - results := map[string][]mTypes.ReferrerInfo{} - - for digest, ref := range refs { - referrers := []mTypes.ReferrerInfo{} - - for _, dbRef := range ref.List { - referrers = append(referrers, mTypes.ReferrerInfo{ - Digest: dbRef.Digest, - MediaType: dbRef.MediaType, - ArtifactType: dbRef.ArtifactType, - Size: int(dbRef.Size), - Annotations: dbRef.Annotations, - }) - } - - results[digest] = referrers - } - - return results -} - -func GetImageReferrers(refs *proto_go.ReferrersInfo) []mTypes.ReferrerInfo { - results := []mTypes.ReferrerInfo{} - - for _, dbRef := range refs.List { - results = append(results, mTypes.ReferrerInfo{ - Digest: dbRef.Digest, - MediaType: dbRef.MediaType, - ArtifactType: dbRef.ArtifactType, - Size: int(dbRef.Size), - Annotations: dbRef.Annotations, - }) - } - - return results -} - -func getProtoReferrers(refs map[string][]mTypes.ReferrerInfo) map[string]*proto_go.ReferrersInfo { - results := map[string]*proto_go.ReferrersInfo{} - - for digest, ref := range refs { - referrersInfoList := []*proto_go.ReferrerInfo{} - - for _, dbRef := range ref { - referrersInfoList = append(referrersInfoList, &proto_go.ReferrerInfo{ - Digest: dbRef.Digest, - MediaType: dbRef.MediaType, - ArtifactType: dbRef.ArtifactType, - Size: int64(dbRef.Size), - Annotations: dbRef.Annotations, - }) - } - - results[digest] = &proto_go.ReferrersInfo{List: referrersInfoList} - } - - return results -} - -func GetSignatures(sigs map[string]*proto_go.ManifestSignatures) map[string]mTypes.ManifestSignatures { - results := map[string]mTypes.ManifestSignatures{} - - for digest, dbSignatures := range sigs { - imageSignatures := mTypes.ManifestSignatures{} - - for signatureName, signatureInfo := range dbSignatures.Map { - imageSignatures[signatureName] = getSignaturesInfo(signatureInfo.List) - } - - results[digest] = imageSignatures - } - - return results -} - -func GetImageSignatures(sigs *proto_go.ManifestSignatures) mTypes.ManifestSignatures { - results := mTypes.ManifestSignatures{} - - for signatureName, signatureInfo := range sigs.Map { - results[signatureName] = getSignaturesInfo(signatureInfo.List) - } - - return results -} - -func getProtoSignatures(sigs map[string]mTypes.ManifestSignatures) map[string]*proto_go.ManifestSignatures { - results := map[string]*proto_go.ManifestSignatures{} - - for digest, dbSignatures := range sigs { - imageSignatures := &proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{}} - - for signatureName, signatureInfo := range dbSignatures { - imageSignatures.Map[signatureName] = &proto_go.SignaturesInfo{List: getProtoSignaturesInfo(signatureInfo)} - } - - results[digest] = imageSignatures - } - - return results -} - -func getSignaturesInfo(sigsInfo []*proto_go.SignatureInfo) []mTypes.SignatureInfo { - results := []mTypes.SignatureInfo{} - - for _, siginfo := range sigsInfo { - results = append(results, mTypes.SignatureInfo{ - SignatureManifestDigest: siginfo.SignatureManifestDigest, - LayersInfo: getLayersInfo(siginfo.LayersInfo), - }) - } - - return results -} - -func getProtoSignaturesInfo(sigsInfo []mTypes.SignatureInfo) []*proto_go.SignatureInfo { - results := []*proto_go.SignatureInfo{} - - for _, sigInfo := range sigsInfo { - results = append(results, &proto_go.SignatureInfo{ - SignatureManifestDigest: sigInfo.SignatureManifestDigest, - LayersInfo: GetProtoLayersInfo(sigInfo.LayersInfo), - }) - } - - return results -} - -func GetProtoLayersInfo(layersInfo []mTypes.LayerInfo) []*proto_go.LayersInfo { - result := make([]*proto_go.LayersInfo, len(layersInfo)) - - for _, layerInfo := range layersInfo { - result = append(result, &proto_go.LayersInfo{ - LayerDigest: layerInfo.LayerDigest, - LayerContent: layerInfo.LayerContent, - SignatureKey: layerInfo.SignatureKey, - Signer: layerInfo.Signer, - Date: timestamppb.New(layerInfo.Date), - }) - } - - return result -} - -func getLayersInfo(layersInfo []*proto_go.LayersInfo) []mTypes.LayerInfo { - results := []mTypes.LayerInfo{} - - for _, layerInfo := range layersInfo { - date := time.Time{} - - if layerInfo.Date != nil { - date = layerInfo.Date.AsTime() - } - - results = append(results, mTypes.LayerInfo{ - LayerDigest: layerInfo.LayerDigest, - LayerContent: layerInfo.LayerContent, - SignatureKey: layerInfo.SignatureKey, - Signer: layerInfo.Signer, - Date: date, - }) - } - - return results -} - -func getTags(tags map[string]*proto_go.TagDescriptor) map[string]mTypes.Descriptor { - resultMap := map[string]mTypes.Descriptor{} - - for tag, tagDescriptor := range tags { - resultMap[tag] = mTypes.Descriptor{ - Digest: tagDescriptor.Digest, - MediaType: tagDescriptor.MediaType, - } - } - - return resultMap -} - -func getProtoTags(tags map[string]mTypes.Descriptor) map[string]*proto_go.TagDescriptor { - resultMap := map[string]*proto_go.TagDescriptor{} - - for tag, tagDescriptor := range tags { - resultMap[tag] = &proto_go.TagDescriptor{ - Digest: tagDescriptor.Digest, - MediaType: tagDescriptor.MediaType, - } - } - - return resultMap -} - -func GetFullRepoMeta(protoRepoMeta *proto_go.ProtoRepoMeta) mTypes.FullRepoMetadata { - repoDownloads := int32(0) - - for _, descriptor := range protoRepoMeta.Tags { - repoDownloads += protoRepoMeta.Statistics[descriptor.Digest].DownloadCount - } - - return mTypes.FullRepoMetadata{ - Name: protoRepoMeta.Name, - Tags: getTags(protoRepoMeta.Tags), - Rank: int(protoRepoMeta.Rank), - Size: int64(protoRepoMeta.Size), - Platforms: GetPlatforms(protoRepoMeta.Platforms), - Vendors: protoRepoMeta.Vendors, - IsStarred: protoRepoMeta.IsStarred, - IsBookmarked: protoRepoMeta.IsBookmarked, - StarCount: int(protoRepoMeta.Stars), - DownloadCount: int(repoDownloads), - LastUpdatedImage: GetLastUpdatedImage(protoRepoMeta.LastUpdatedImage), - Statistics: GetStatisticsMap(protoRepoMeta.Statistics), - Signatures: GetSignatures(protoRepoMeta.Signatures), - Referrers: GetReferrers(protoRepoMeta.Referrers), - } - -} - -func GetPlatforms(platforms []*proto_go.Platform) []ispec.Platform { - result := []ispec.Platform{} - - for i := range platforms { - result = append(result, ispec.Platform{ - OS: platforms[i].Os, - Architecture: platforms[i].Arch, - }) - } - - return result -} - -func GetLastUpdated(timestamp *timestamppb.Timestamp) *time.Time { - if timestamp == nil { - return nil - } - - return ref(timestamp.AsTime()) -} - -func AddPlatforms(platforms []ispec.Platform, newPlatforms []ispec.Platform) []ispec.Platform { - for _, newPlatform := range newPlatforms { - if !ContainsPlatform(platforms, newPlatform) { - platforms = append(platforms, newPlatform) - } - } - - return platforms -} - -func ContainsPlatform(platforms []ispec.Platform, platform ispec.Platform) bool { - for i := range platforms { - if platforms[i].OS == platform.OS || platforms[i].Architecture == platform.Architecture { - return true - } - } - - return false -} - -func AddProtoPlatforms(platforms []*proto_go.Platform, newPlatforms []*proto_go.Platform) []*proto_go.Platform { - for _, newPlatform := range newPlatforms { - if !ContainsProtoPlatform(platforms, newPlatform) { - platforms = append(platforms, newPlatform) - } - } - - return platforms -} - -func ContainsProtoPlatform(platforms []*proto_go.Platform, platform *proto_go.Platform) bool { - for i := range platforms { - if platforms[i].Os == platform.Os || platforms[i].Arch == platform.Arch { - return true - } - } - - return false -} - -func AddVendors(vendors []string, newVendors []string) []string { - for _, newVendor := range newVendors { - if !zcommon.Contains(vendors, newVendor) { - vendors = append(vendors, newVendor) - } - } - - return vendors -} - -func GetEarlierLastUpdated(lastUpdated *time.Time, newLastUpdated *time.Time) *time.Time { - if newLastUpdated == nil { - return lastUpdated - } - - if lastUpdated == nil { - return newLastUpdated - } - - if lastUpdated.Before(*newLastUpdated) { - return newLastUpdated - } - - return lastUpdated -} - -func GetLastUpdatedImage(protoLastUpdated *proto_go.RepoLastUpdatedImage) *mTypes.LastUpdatedImage { - if protoLastUpdated == nil { - return nil - } - - return &mTypes.LastUpdatedImage{ - Descriptor: mTypes.Descriptor{ - Digest: protoLastUpdated.Digest, - MediaType: protoLastUpdated.MediaType, - }, - Tag: protoLastUpdated.Tag, - LastUpdated: getTime(protoLastUpdated.LastUpdated), - } -} - -func GetProtoEarlierUpdatedImage(repoLastImage *proto_go.RepoLastUpdatedImage, newLastImage *proto_go.RepoLastUpdatedImage, -) *proto_go.RepoLastUpdatedImage { - if repoLastImage == nil { - return newLastImage - } - - if newLastImage == nil || newLastImage.LastUpdated == nil { - return repoLastImage - } - - if repoLastImage.LastUpdated == nil { - return newLastImage - } - - if repoLastImage.LastUpdated.AsTime().Before(newLastImage.LastUpdated.AsTime()) { - return newLastImage - } - - return repoLastImage -} diff --git a/pkg/meta/convert/convert_full.go b/pkg/meta/convert/convert_full.go deleted file mode 100644 index 3a85f2ff86..0000000000 --- a/pkg/meta/convert/convert_full.go +++ /dev/null @@ -1,18 +0,0 @@ -package convert - -// import ( -// "zotregistry.io/zot/pkg/meta/proto_go" - -// mTypes "zotregistry.io/zot/pkg/meta/types" -// ) - -// // func GetFullRepoMeta(repo *proto_go.ProtoRepoMeta) mTypes.FullRepoMetadata { - -// // return mTypes.FullRepoMetadata{ -// // Name: repo.Name, -// // Tags: getTags(repo.Tags), -// // Statistics: getStatistics(repo.Statistics), -// // Signatures: getSignatures(repo.Signatures), -// // Referrers: getReferrers(repo.Referrers), -// // } -// // } diff --git a/pkg/meta/convert/convert_proto.go b/pkg/meta/convert/convert_proto.go new file mode 100644 index 0000000000..59c3be8720 --- /dev/null +++ b/pkg/meta/convert/convert_proto.go @@ -0,0 +1,392 @@ +package convert + +import ( + "time" + + godigest "github.com/opencontainers/go-digest" + ispec "github.com/opencontainers/image-spec/specs-go/v1" + "google.golang.org/protobuf/types/known/timestamppb" + + "zotregistry.io/zot/pkg/common" + "zotregistry.io/zot/pkg/meta/proto_go" + mTypes "zotregistry.io/zot/pkg/meta/types" +) + +func GetProtoRepoMeta(repo mTypes.RepoMeta) *proto_go.RepoMeta { + return &proto_go.RepoMeta{ + Name: repo.Name, + Tags: GetProtoTags(repo.Tags), + Statistics: GetProtoStatistics(repo.Statistics), + Signatures: GetProtoSignatures(repo.Signatures), + Referrers: GetProtoReferrers(repo.Referrers), + Size: int32(repo.Size), + Vendors: repo.Vendors, + Platforms: GetProtoPlatforms(repo.Platforms), + LastUpdatedImage: GetProtoLastUpdatedImage(repo.LastUpdatedImage), + } +} + +func GetProtoImageData(imageData mTypes.ImageData) *proto_go.ImageData { + switch imageData.MediaType { + case ispec.MediaTypeImageManifest: + if len(imageData.Manifests) == 0 { + return nil + } + manifestData := imageData.Manifests[0] + + return GetProtoImageManifestData(manifestData.Manifest, manifestData.Config, manifestData.Size, + manifestData.Digest.String()) + case ispec.MediaTypeImageIndex: + if imageData.Index == nil { + return nil + } + + return GetProtoImageIndexData(*imageData.Index, imageData.Size, imageData.Digest.String()) + default: + return nil + } +} + +func GetProtoImageManifestData(manifestContent ispec.Manifest, configContent ispec.Image, size int64, digest string, +) *proto_go.ImageData { + return &proto_go.ImageData{ + MediaType: ispec.MediaTypeImageManifest, + Manifests: []*proto_go.ManifestData{GetProtoManifestData(manifestContent, configContent, size, digest)}, + Index: nil, + } +} + +func GetProtoManifestData(manifestContent ispec.Manifest, configContent ispec.Image, size int64, digest string, +) *proto_go.ManifestData { + return &proto_go.ManifestData{ + Digest: digest, + Size: size, + Manifest: &proto_go.Manifest{ + Versioned: &proto_go.Versioned{SchemaVersion: int32(manifestContent.SchemaVersion)}, + Config: &proto_go.Descriptor{ + Digest: manifestContent.Config.Digest.String(), + Size: manifestContent.Config.Size, + MediaType: manifestContent.Config.MediaType, + }, + MediaType: ref(ispec.MediaTypeImageManifest), + ArtifactType: &manifestContent.ArtifactType, + Layers: getProtoManifestLayers(manifestContent.Layers), + Subject: getProtoDesc(manifestContent.Subject), + Annotations: manifestContent.Annotations, + }, + Config: &proto_go.Image{ + Created: GetProtoTime(configContent.Created), + Author: &configContent.Author, + Platform: &proto_go.Platform{ + Architecture: GetArch(configContent.Architecture, configContent.Variant), + OS: configContent.OS, + }, + Config: &proto_go.ImageConfig{ + User: configContent.Config.User, + ExposedPorts: getProtoExposedPorts(configContent.Config.ExposedPorts), + Env: configContent.Config.Env, + Entrypoint: configContent.Config.Entrypoint, + Cmd: configContent.Config.Cmd, + Volumes: getProtoConfigVolumes(configContent.Config.Volumes), + WorkingDir: &configContent.Config.WorkingDir, + Labels: configContent.Config.Labels, + StopSignal: &configContent.Config.StopSignal, + }, + RootFS: &proto_go.RootFS{ + Type: configContent.RootFS.Type, + DiffIDs: getProtoDiffIDs(configContent.RootFS.DiffIDs), + }, + History: getProtoHistory(configContent.History), + }, + } +} + +func GetProtoImageIndexData(indexContent ispec.Index, size int64, digest string) *proto_go.ImageData { + return &proto_go.ImageData{ + MediaType: ispec.MediaTypeImageIndex, + Index: &proto_go.IndexData{ + Size: size, + Digest: digest, + Index: &proto_go.Index{ + Versioned: &proto_go.Versioned{SchemaVersion: int32(indexContent.Versioned.SchemaVersion)}, + MediaType: ref(ispec.MediaTypeImageIndex), + ArtifactType: ref(common.GetIndexArtifactType(indexContent)), + Manifests: getProtoManifestList(indexContent.Manifests), + Subject: getProtoDesc(indexContent.Subject), + Annotations: indexContent.Annotations, + }, + }, + } +} + +func GetProtoStatistics(stats map[string]mTypes.DescriptorStatistics) map[string]*proto_go.DescriptorStatistics { + results := map[string]*proto_go.DescriptorStatistics{} + + for digest, stat := range stats { + results[digest] = &proto_go.DescriptorStatistics{ + DownloadCount: int32(stat.DownloadCount), + } + } + + return results +} + +func GetProtoPlatforms(platforms []ispec.Platform) []*proto_go.Platform { + result := []*proto_go.Platform{} + + for i := range platforms { + result = append(result, &proto_go.Platform{ + OS: platforms[i].OS, + Architecture: platforms[i].Architecture, + }) + } + + return result +} + +func GetProtoReferrers(refs map[string][]mTypes.ReferrerInfo) map[string]*proto_go.ReferrersInfo { + results := map[string]*proto_go.ReferrersInfo{} + + for digest, ref := range refs { + referrersInfoList := []*proto_go.ReferrerInfo{} + + for _, dbRef := range ref { + referrersInfoList = append(referrersInfoList, GetProtoReferrerInfo(dbRef)) + } + + results[digest] = &proto_go.ReferrersInfo{List: referrersInfoList} + } + + return results +} + +func GetProtoSignatures(sigs map[string]mTypes.ManifestSignatures) map[string]*proto_go.ManifestSignatures { + results := map[string]*proto_go.ManifestSignatures{} + + for digest, dbSignatures := range sigs { + imageSignatures := &proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{}} + + for signatureName, signatureInfo := range dbSignatures { + imageSignatures.Map[signatureName] = &proto_go.SignaturesInfo{List: GetProtoSignaturesInfo(signatureInfo)} + } + + results[digest] = imageSignatures + } + + return results +} + +func GetProtoSignaturesInfo(sigsInfo []mTypes.SignatureInfo) []*proto_go.SignatureInfo { + results := []*proto_go.SignatureInfo{} + + for _, sigInfo := range sigsInfo { + results = append(results, &proto_go.SignatureInfo{ + SignatureManifestDigest: sigInfo.SignatureManifestDigest, + LayersInfo: GetProtoLayersInfo(sigInfo.LayersInfo), + }) + } + + return results +} + +func GetProtoLayersInfo(layersInfo []mTypes.LayerInfo) []*proto_go.LayersInfo { + result := make([]*proto_go.LayersInfo, 0, len(layersInfo)) + + for _, layerInfo := range layersInfo { + result = append(result, &proto_go.LayersInfo{ + LayerDigest: layerInfo.LayerDigest, + LayerContent: layerInfo.LayerContent, + SignatureKey: layerInfo.SignatureKey, + Signer: layerInfo.Signer, + Date: timestamppb.New(layerInfo.Date), + }) + } + + return result +} + +func getProtoManifestLayers(layers []ispec.Descriptor) []*proto_go.Descriptor { + protoLayers := []*proto_go.Descriptor{} + + for _, layer := range layers { + layer := layer + + protoLayers = append(protoLayers, getProtoDesc(&layer)) + } + + return protoLayers +} + +func getProtoDesc(descriptor *ispec.Descriptor) *proto_go.Descriptor { + if descriptor == nil { + return nil + } + + return &proto_go.Descriptor{ + MediaType: descriptor.MediaType, + Digest: descriptor.Digest.String(), + Size: descriptor.Size, + URLs: descriptor.URLs, + Annotations: descriptor.Annotations, + Data: descriptor.Data, + Platform: getProtoPlatform(descriptor.Platform), + ArtifactType: &descriptor.ArtifactType, + } +} + +func getProtoManifestList(manifests []ispec.Descriptor) []*proto_go.Descriptor { + result := make([]*proto_go.Descriptor, 0, len(manifests)) + + for _, manifest := range manifests { + result = append(result, &proto_go.Descriptor{ + MediaType: manifest.MediaType, + Digest: manifest.Digest.String(), + Size: manifest.Size, + URLs: manifest.URLs, + Annotations: manifest.Annotations, + Data: manifest.Data, + Platform: getProtoPlatform(manifest.Platform), + ArtifactType: ref(manifest.ArtifactType), + }) + } + + return result +} + +func getProtoPlatform(platform *ispec.Platform) *proto_go.Platform { + if platform == nil { + return nil + } + + return &proto_go.Platform{ + Architecture: GetArch(platform.Architecture, platform.Variant), + OS: platform.OS, + } +} + +func getProtoHistory(historySlice []ispec.History) []*proto_go.History { + protoHistory := []*proto_go.History{} + + for _, history := range historySlice { + history := history + + protoHistory = append(protoHistory, &proto_go.History{ + Created: GetProtoTime(history.Created), + CreatedBy: &history.CreatedBy, + Author: &history.Author, + Comment: &history.Comment, + EmptyLayer: &history.EmptyLayer, + }) + } + + return protoHistory +} + +func getProtoDiffIDs(digests []godigest.Digest) []string { + digestsStr := []string{} + + for _, digest := range digests { + digestsStr = append(digestsStr, digest.String()) + } + + return digestsStr +} + +func getProtoExposedPorts(exposedPorts map[string]struct{}) map[string]*proto_go.EmptyMessage { + protoPorts := map[string]*proto_go.EmptyMessage{} + + for i := range exposedPorts { + protoPorts[i] = &proto_go.EmptyMessage{} + } + + return protoPorts +} + +func getProtoConfigVolumes(volumes map[string]struct{}) map[string]*proto_go.EmptyMessage { + protoVolumes := map[string]*proto_go.EmptyMessage{} + + for i := range volumes { + protoVolumes[i] = &proto_go.EmptyMessage{} + } + + return protoVolumes +} + +func GetProtoReferrerInfo(referrer mTypes.ReferrerInfo) *proto_go.ReferrerInfo { + return &proto_go.ReferrerInfo{ + Digest: referrer.Digest, + MediaType: referrer.MediaType, + ArtifactType: referrer.ArtifactType, + Size: int64(referrer.Size), + Annotations: referrer.Annotations, + } +} + +func GetProtoTime(time *time.Time) *timestamppb.Timestamp { + if time == nil { + return nil + } + + return timestamppb.New(*time) +} + +func GetProtoTags(tags map[string]mTypes.Descriptor) map[string]*proto_go.TagDescriptor { + resultMap := map[string]*proto_go.TagDescriptor{} + + for tag, tagDescriptor := range tags { + resultMap[tag] = &proto_go.TagDescriptor{ + Digest: tagDescriptor.Digest, + MediaType: tagDescriptor.MediaType, + } + } + + return resultMap +} + +func GetProtoLastUpdatedImage(lastUpdatedImage *mTypes.LastUpdatedImage) *proto_go.RepoLastUpdatedImage { + if lastUpdatedImage == nil { + return nil + } + + return &proto_go.RepoLastUpdatedImage{ + LastUpdated: GetProtoTime(lastUpdatedImage.LastUpdated), + MediaType: lastUpdatedImage.MediaType, + Digest: lastUpdatedImage.Digest, + Tag: lastUpdatedImage.Tag, + } +} + +func GetProtoEarlierUpdatedImage(repoLastImage *proto_go.RepoLastUpdatedImage, lastImage *proto_go.RepoLastUpdatedImage, +) *proto_go.RepoLastUpdatedImage { + if repoLastImage == nil { + return lastImage + } + + if lastImage == nil || lastImage.LastUpdated == nil { + return repoLastImage + } + + if repoLastImage.LastUpdated == nil { + return lastImage + } + + if repoLastImage.LastUpdated.AsTime().Before(lastImage.LastUpdated.AsTime()) { + return lastImage + } + + return repoLastImage +} + +func ref[T any](input T) *T { + ref := input + + return &ref +} + +func deref[T any](pointer *T, defaultVal T) T { + if pointer != nil { + return *pointer + } + + return defaultVal +} diff --git a/pkg/meta/dynamodb/dynamodb.go b/pkg/meta/dynamodb/dynamodb.go index e486fe1fb9..042fe06727 100644 --- a/pkg/meta/dynamodb/dynamodb.go +++ b/pkg/meta/dynamodb/dynamodb.go @@ -2,7 +2,6 @@ package dynamodb import ( "context" - "encoding/json" "errors" "fmt" "strings" @@ -29,41 +28,33 @@ import ( reqCtx "zotregistry.io/zot/pkg/requestcontext" ) -var errMetaDB = errors.New("metadb: error while constructing manifest meta") - type DynamoDB struct { - Client *dynamodb.Client - APIKeyTablename string - RepoMetaTablename string - ProtoRepoMetaTablename string - ProtoRepoBlobsTablename string - ProtoImageDataTablename string - IndexDataTablename string - ManifestDataTablename string - UserDataTablename string - VersionTablename string - Patches []func(client *dynamodb.Client, tableNames map[string]string) error - imgTrustStore mTypes.ImageTrustStore - Log log.Logger + Client *dynamodb.Client + APIKeyTablename string + RepoMetaTablename string + RepoBlobsTablename string + ImageDataTablename string + UserDataTablename string + VersionTablename string + Patches []func(client *dynamodb.Client, tableNames map[string]string) error + imgTrustStore mTypes.ImageTrustStore + Log log.Logger } func New( client *dynamodb.Client, params DBDriverParameters, log log.Logger, ) (*DynamoDB, error) { dynamoWrapper := DynamoDB{ - Client: client, - RepoMetaTablename: params.RepoMetaTablename, - ManifestDataTablename: params.ManifestDataTablename, - IndexDataTablename: params.IndexDataTablename, - VersionTablename: params.VersionTablename, - UserDataTablename: params.UserDataTablename, - APIKeyTablename: params.APIKeyTablename, - ProtoRepoMetaTablename: "Proto" + params.RepoMetaTablename, // TODO CHANGE - ProtoImageDataTablename: "Proto" + params.ManifestDataTablename, // TODO CHANGE - ProtoRepoBlobsTablename: "ProtoBlobs" + params.RepoMetaTablename, - Patches: version.GetDynamoDBPatches(), - imgTrustStore: nil, - Log: log, + Client: client, + VersionTablename: params.VersionTablename, + UserDataTablename: params.UserDataTablename, + APIKeyTablename: params.APIKeyTablename, + RepoMetaTablename: params.RepoMetaTablename, + ImageDataTablename: params.ImageDataTablename, + RepoBlobsTablename: params.RepoBlobsInfoTablename, + Patches: version.GetDynamoDBPatches(), + imgTrustStore: nil, + Log: log, } err := dynamoWrapper.createVersionTable() @@ -71,27 +62,27 @@ func New( return nil, err } - err = dynamoWrapper.createProtoRepoMetaTable() + err = dynamoWrapper.createTable(dynamoWrapper.RepoMetaTablename) if err != nil { return nil, err } - err = dynamoWrapper.createProtoRepoBlobsTable() + err = dynamoWrapper.createTable(dynamoWrapper.RepoBlobsTablename) if err != nil { return nil, err } - err = dynamoWrapper.createProtoImageDataTable() + err = dynamoWrapper.createTable(dynamoWrapper.ImageDataTablename) if err != nil { return nil, err } - err = dynamoWrapper.createUserDataTable() + err = dynamoWrapper.createTable(dynamoWrapper.UserDataTablename) if err != nil { return nil, err } - err = dynamoWrapper.createAPIKeyTable() + err = dynamoWrapper.createTable(dynamoWrapper.APIKeyTablename) if err != nil { return nil, err } @@ -108,1125 +99,693 @@ func (dwr *DynamoDB) SetImageTrustStore(imgTrustStore mTypes.ImageTrustStore) { dwr.imgTrustStore = imgTrustStore } -func (dwr *DynamoDB) SetManifestData(manifestDigest godigest.Digest, manifestData mTypes.ManifestData) error { - mdAttributeValue, err := attributevalue.Marshal(manifestData) +func (dwr *DynamoDB) SetProtoImageData(digest godigest.Digest, protoImageData *proto_go.ImageData) error { + bytes, err := proto.Marshal(protoImageData) + if err != nil { + return err + } + + mdAttributeValue, err := attributevalue.Marshal(bytes) if err != nil { return err } _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ ExpressionAttributeNames: map[string]string{ - "#MD": "ManifestData", + "#ID": "ImageData", }, ExpressionAttributeValues: map[string]types.AttributeValue{ - ":ManifestData": mdAttributeValue, + ":ImageData": mdAttributeValue, }, Key: map[string]types.AttributeValue{ - "Digest": &types.AttributeValueMemberS{ - Value: manifestDigest.String(), + "Key": &types.AttributeValueMemberS{ + Value: mConvert.GetImageDigestStr(protoImageData), }, }, - TableName: aws.String(dwr.ManifestDataTablename), - UpdateExpression: aws.String("SET #MD = :ManifestData"), + TableName: aws.String(dwr.ImageDataTablename), + UpdateExpression: aws.String("SET #ID = :ImageData"), }) return err } -func (dwr *DynamoDB) GetManifestData(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - resp, err := dwr.Client.GetItem(context.Background(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.ManifestDataTablename), +func (dwr *DynamoDB) SetImageData(digest godigest.Digest, imageData mTypes.ImageData) error { + return dwr.SetProtoImageData(imageData.Digest, mConvert.GetProtoImageData(imageData)) +} + +func (dwr *DynamoDB) GetProtoImageData(ctx context.Context, digest godigest.Digest) (*proto_go.ImageData, error) { + resp, err := dwr.Client.GetItem(ctx, &dynamodb.GetItemInput{ + TableName: aws.String(dwr.ImageDataTablename), Key: map[string]types.AttributeValue{ - "Digest": &types.AttributeValueMemberS{Value: manifestDigest.String()}, + "Key": &types.AttributeValueMemberS{Value: digest.String()}, }, }) if err != nil { - return mTypes.ManifestData{}, err - } - - if resp.Item == nil { - return mTypes.ManifestData{}, zerr.ErrManifestDataNotFound - } - - var manifestData mTypes.ManifestData - - err = attributevalue.Unmarshal(resp.Item["ManifestData"], &manifestData) - if err != nil { - return mTypes.ManifestData{}, err - } - - return manifestData, nil -} - -func (dwr *DynamoDB) SetManifestMeta(repo string, manifestDigest godigest.Digest, manifestMeta mTypes.ManifestMetadata, -) error { - if manifestMeta.Signatures == nil { - manifestMeta.Signatures = mTypes.ManifestSignatures{} + return nil, err } - repoMeta, err := dwr.GetRepoMeta(repo) - if err != nil { - if !errors.Is(err, zerr.ErrRepoMetaNotFound) { - return err - } - - repoMeta = mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - } - } + blob := []byte{} - err = dwr.SetManifestData(manifestDigest, mTypes.ManifestData{ - ManifestBlob: manifestMeta.ManifestBlob, - ConfigBlob: manifestMeta.ConfigBlob, - }) - if err != nil { - return err + if resp.Item == nil { + return nil, zerr.ErrImageDataNotFound } - updatedRepoMeta := common.UpdateManifestMeta(repoMeta, manifestDigest, manifestMeta) - - err = dwr.SetRepoMeta(repo, updatedRepoMeta) + err = attributevalue.Unmarshal(resp.Item["ImageData"], &blob) if err != nil { - return err + return nil, err } - return err -} - -func (dwr *DynamoDB) GetManifestMeta(repo string, manifestDigest godigest.Digest, -) (mTypes.ManifestMetadata, error) { //nolint:contextcheck - manifestData, err := dwr.GetManifestData(manifestDigest) - if err != nil { - if errors.Is(err, zerr.ErrManifestDataNotFound) { - return mTypes.ManifestMetadata{}, zerr.ErrManifestMetaNotFound - } - - return mTypes.ManifestMetadata{}, - fmt.Errorf("%w for manifest '%s' from repo '%s'", errMetaDB, manifestDigest, repo) - } + imageData := &proto_go.ImageData{} - repoMeta, err := dwr.GetRepoMeta(repo) + err = proto.Unmarshal(blob, imageData) if err != nil { - if errors.Is(err, zerr.ErrRepoMetaNotFound) { - return mTypes.ManifestMetadata{}, zerr.ErrManifestMetaNotFound - } - - return mTypes.ManifestMetadata{}, - fmt.Errorf("%w for manifest '%s' from repo '%s'", errMetaDB, manifestDigest, repo) - } - - manifestMetadata := mTypes.ManifestMetadata{} - - manifestMetadata.ManifestBlob = manifestData.ManifestBlob - manifestMetadata.ConfigBlob = manifestData.ConfigBlob - manifestMetadata.DownloadCount = repoMeta.Statistics[manifestDigest.String()].DownloadCount - - manifestMetadata.Signatures = mTypes.ManifestSignatures{} - - if repoMeta.Signatures[manifestDigest.String()] != nil { - manifestMetadata.Signatures = repoMeta.Signatures[manifestDigest.String()] + return nil, err } - return manifestMetadata, nil + return imageData, nil } -func (dwr *DynamoDB) IncrementRepoStars(repo string) error { - repoMeta, err := dwr.GetRepoMeta(repo) - if err != nil { - return err - } - - repoMeta.Stars++ - - err = dwr.SetRepoMeta(repo, repoMeta) - - return err -} +func (dwr *DynamoDB) setProtoRepoMeta(repo string, repoMeta *proto_go.RepoMeta) error { + repoMeta.Name = repo -func (dwr *DynamoDB) DecrementRepoStars(repo string) error { - repoMeta, err := dwr.GetRepoMeta(repo) + repoMetaBlob, err := proto.Marshal(repoMeta) if err != nil { return err } - if repoMeta.Stars > 0 { - repoMeta.Stars-- - } - - err = dwr.SetRepoMeta(repo, repoMeta) - - return err -} - -func (dwr *DynamoDB) GetRepoStars(repo string) (int, error) { - repoMeta, err := dwr.GetRepoMeta(repo) - if err != nil { - return 0, err - } - - return repoMeta.Stars, nil -} - -func (dwr *DynamoDB) SetIndexData(indexDigest godigest.Digest, indexData mTypes.IndexData) error { - indexAttributeValue, err := attributevalue.Marshal(indexData) + repoAttributeValue, err := attributevalue.Marshal(repoMetaBlob) if err != nil { return err } _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ ExpressionAttributeNames: map[string]string{ - "#ID": "IndexData", + "#RM": "RepoMetadata", }, ExpressionAttributeValues: map[string]types.AttributeValue{ - ":IndexData": indexAttributeValue, + ":RepoMetadata": repoAttributeValue, }, Key: map[string]types.AttributeValue{ - "IndexDigest": &types.AttributeValueMemberS{ - Value: indexDigest.String(), + "Key": &types.AttributeValueMemberS{ + Value: repo, }, }, - TableName: aws.String(dwr.IndexDataTablename), - UpdateExpression: aws.String("SET #ID = :IndexData"), + TableName: aws.String(dwr.RepoMetaTablename), + UpdateExpression: aws.String("SET #RM = :RepoMetadata"), }) return err } -func (dwr *DynamoDB) GetIndexData(indexDigest godigest.Digest) (mTypes.IndexData, error) { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.IndexDataTablename), +func (dwr *DynamoDB) getProtoRepoMeta(ctx context.Context, repo string) (*proto_go.RepoMeta, error) { + resp, err := dwr.Client.GetItem(ctx, &dynamodb.GetItemInput{ + TableName: aws.String(dwr.RepoMetaTablename), Key: map[string]types.AttributeValue{ - "IndexDigest": &types.AttributeValueMemberS{ - Value: indexDigest.String(), - }, + "Key": &types.AttributeValueMemberS{Value: repo}, }, }) if err != nil { - return mTypes.IndexData{}, err + return nil, err } if resp.Item == nil { - return mTypes.IndexData{}, zerr.ErrRepoMetaNotFound + return nil, zerr.ErrRepoMetaNotFound } - var indexData mTypes.IndexData + blob := []byte{} - err = attributevalue.Unmarshal(resp.Item["IndexData"], &indexData) + err = attributevalue.Unmarshal(resp.Item["RepoMetadata"], &blob) if err != nil { - return mTypes.IndexData{}, err + return nil, err } - return indexData, nil -} + repoMeta := &proto_go.RepoMeta{} -func (dwr DynamoDB) SetReferrer(repo string, referredDigest godigest.Digest, referrer mTypes.ReferrerInfo) error { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) + err = proto.Unmarshal(blob, repoMeta) if err != nil { + return nil, err + } + + return repoMeta, nil +} + +func (dwr *DynamoDB) SetRepoReference(repo string, reference string, imageData mTypes.ImageData) error { + if err := common.ValidateRepoReferenceInput(repo, reference, imageData.Digest); err != nil { return err } - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, + // 1. Add image data to db if needed + protoImageData := mConvert.GetProtoImageData(imageData) + + err := dwr.SetProtoImageData(imageData.Digest, protoImageData) + if err != nil { + return err } - if resp.Item != nil { - err := attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { + repoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) + if err != nil { + if !errors.Is(err, zerr.ErrRepoMetaNotFound) { return err } - } - - referrers := repoMeta.Referrers[referredDigest.String()] - for i := range referrers { - if referrers[i].Digest == referrer.Digest { - return nil + repoMeta = &proto_go.RepoMeta{ + Name: repo, + Tags: map[string]*proto_go.TagDescriptor{"": {}}, + Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, + Signatures: map[string]*proto_go.ManifestSignatures{"": {Map: map[string]*proto_go.SignaturesInfo{"": {}}}}, + Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, } } - referrers = append(referrers, referrer) - - repoMeta.Referrers[referredDigest.String()] = referrers + // 2. Referrers + if subject := mConvert.GetImageSubject(protoImageData); subject != nil { + refInfo := &proto_go.ReferrersInfo{} + if repoMeta.Referrers[subject.Digest.String()] != nil { + refInfo = repoMeta.Referrers[subject.Digest.String()] + } - return dwr.SetRepoMeta(repo, repoMeta) -} + foundReferrer := false -func (dwr DynamoDB) GetReferrers(repo string, referredDigest godigest.Digest) ([]mTypes.ReferrerInfo, error) { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return []mTypes.ReferrerInfo{}, err - } + for i := range refInfo.List { + if refInfo.List[i].Digest == mConvert.GetImageDigestStr(protoImageData) { + foundReferrer = true + refInfo.List[i].Count += 1 - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - } + break + } + } - if resp.Item != nil { - err := attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { - return []mTypes.ReferrerInfo{}, err + if !foundReferrer { + refInfo.List = append(refInfo.List, &proto_go.ReferrerInfo{ + Count: 1, + MediaType: protoImageData.MediaType, + Digest: mConvert.GetImageDigestStr(protoImageData), + ArtifactType: mConvert.GetImageArtifactType(protoImageData), + Size: mConvert.GetImageManifestSize(protoImageData), + Annotations: mConvert.GetImageAnnotations(protoImageData), + }) } - } - return repoMeta.Referrers[referredDigest.String()], nil -} + repoMeta.Referrers[subject.Digest.String()] = refInfo + } -func (dwr DynamoDB) DeleteReferrer(repo string, referredDigest godigest.Digest, - referrerDigest godigest.Digest, -) error { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return err + // 3. Update tag + if !common.ReferenceIsDigest(reference) { + repoMeta.Tags[reference] = &proto_go.TagDescriptor{ + Digest: imageData.Digest.String(), + MediaType: imageData.MediaType, + } } - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, + if _, ok := repoMeta.Statistics[imageData.Digest.String()]; !ok { + repoMeta.Statistics[imageData.Digest.String()] = &proto_go.DescriptorStatistics{DownloadCount: 0} } - if resp.Item != nil { - err := attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { - return err + if _, ok := repoMeta.Signatures[imageData.Digest.String()]; !ok { + repoMeta.Signatures[imageData.Digest.String()] = &proto_go.ManifestSignatures{ + Map: map[string]*proto_go.SignaturesInfo{"": {}}, } } - referrers := repoMeta.Referrers[referredDigest.String()] - - for i := range referrers { - if referrers[i].Digest == referrerDigest.String() { - referrers = append(referrers[:i], referrers[i+1:]...) - - break + if _, ok := repoMeta.Referrers[imageData.Digest.String()]; !ok { + repoMeta.Referrers[imageData.Digest.String()] = &proto_go.ReferrersInfo{ + List: []*proto_go.ReferrerInfo{}, } } - repoMeta.Referrers[referredDigest.String()] = referrers - - return dwr.SetRepoMeta(repo, repoMeta) -} - -func (dwr DynamoDB) GetReferrersInfo(repo string, referredDigest godigest.Digest, - artifactTypes []string, -) ([]mTypes.ReferrerInfo, error) { - referrersInfo, err := dwr.GetReferrers(repo, referredDigest) + // 4. Blobs + repoBlobs, err := dwr.getRepoBlobsInfo(repo) if err != nil { - return nil, err + return err } - filteredResults := make([]mTypes.ReferrerInfo, 0, len(referrersInfo)) - - for _, referrerInfo := range referrersInfo { - if !common.MatchesArtifactTypes(referrerInfo.ArtifactType, artifactTypes) { - continue - } + repoMeta, repoBlobs, err = common.AddImageDataToRepoMeta(repoMeta, repoBlobs, reference, imageData) + if err != nil { + return err + } - filteredResults = append(filteredResults, referrerInfo) + err = dwr.setRepoBlobsInfo(repo, repoBlobs) + if err != nil { + return err } - return filteredResults, nil + return dwr.setProtoRepoMeta(repo, repoMeta) } -/* - RemoveRepoReference removes the tag from RepoMetadata if the reference is a tag, - -it also removes its corresponding digest from Statistics, Signatures and Referrers if there are no tags -pointing to it. -If the reference is a digest then it will remove the digest from Statistics, Signatures and Referrers only -if there are no tags pointing to the digest, otherwise it's noop. -*/ -func (dwr *DynamoDB) RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest, -) error { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), +func (dwr *DynamoDB) getRepoBlobsInfo(repo string) (*proto_go.RepoBlobs, error) { + resp, err := dwr.Client.GetItem(context.Background(), &dynamodb.GetItemInput{ + TableName: aws.String(dwr.RepoBlobsTablename), Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, + "Key": &types.AttributeValueMemberS{Value: repo}, }, }) if err != nil { - return err + return nil, err } - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, + if resp.Item == nil { + return &proto_go.RepoBlobs{Name: repo, Blobs: map[string]*proto_go.BlobInfo{"": {}}}, nil } - if resp.Item != nil { - err := attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { - return err - } + repoBlobsBytes := []byte{} + + err = attributevalue.Unmarshal(resp.Item["RepoBlobsInfo"], &repoBlobsBytes) + if err != nil { + return nil, err } - if !common.ReferenceIsDigest(reference) { - delete(repoMeta.Tags, reference) + repoBlobs := &proto_go.RepoBlobs{} + if repoBlobsBytes == nil { + repoBlobs.Blobs = map[string]*proto_go.BlobInfo{} } else { - // find all tags pointing to this digest - tags := []string{} - for tag, desc := range repoMeta.Tags { - if desc.Digest == reference { - tags = append(tags, tag) - } - } - - // remove all tags - for _, tag := range tags { - delete(repoMeta.Tags, tag) + err := proto.Unmarshal(repoBlobsBytes, repoBlobs) + if err != nil { + return nil, err } } - /* try to find at least one tag pointing to manifestDigest - if not found then we can also remove everything related to this digest */ - var foundTag bool - - for _, desc := range repoMeta.Tags { - if desc.Digest == manifestDigest.String() { - foundTag = true - } - } + return repoBlobs, nil +} - if !foundTag { - delete(repoMeta.Statistics, manifestDigest.String()) - delete(repoMeta.Signatures, manifestDigest.String()) - delete(repoMeta.Referrers, manifestDigest.String()) - } - - err = dwr.SetRepoMeta(repo, repoMeta) - - return err -} - -func (dwr *DynamoDB) SetRepoReference(repo string, reference string, manifestDigest godigest.Digest, - mediaType string, -) error { - if err := common.ValidateRepoReferenceInput(repo, reference, manifestDigest); err != nil { - return err - } - - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return err - } - - repoMeta := mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Referrers: map[string][]mTypes.ReferrerInfo{}, - } - - if resp.Item != nil { - err := attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { - return err - } - } - - if !common.ReferenceIsDigest(reference) { - repoMeta.Tags[reference] = mTypes.Descriptor{ - Digest: manifestDigest.String(), - MediaType: mediaType, - } - } - - if _, ok := repoMeta.Statistics[manifestDigest.String()]; !ok { - repoMeta.Statistics[manifestDigest.String()] = mTypes.DescriptorStatistics{DownloadCount: 0} - } - - if _, ok := repoMeta.Signatures[manifestDigest.String()]; !ok { - repoMeta.Signatures[manifestDigest.String()] = mTypes.ManifestSignatures{} - } - - if _, ok := repoMeta.Referrers[manifestDigest.String()]; !ok { - repoMeta.Referrers[manifestDigest.String()] = []mTypes.ReferrerInfo{} - } - - err = dwr.SetRepoMeta(repo, repoMeta) - - return err -} - -func (dwr *DynamoDB) DeleteRepoTag(repo string, tag string) error { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return err - } - - if resp.Item == nil { - return nil - } - - var repoMeta mTypes.RepoMetadata - - err = attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) +func (dwr *DynamoDB) setRepoBlobsInfo(repo string, repoBlobs *proto_go.RepoBlobs) error { + bytes, err := proto.Marshal(repoBlobs) if err != nil { return err } - delete(repoMeta.Tags, tag) - - repoAttributeValue, err := attributevalue.Marshal(repoMeta) + mdAttributeValue, err := attributevalue.Marshal(bytes) if err != nil { return err } _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ ExpressionAttributeNames: map[string]string{ - "#RM": "RepoMetadata", + "#RBI": "RepoBlobsInfo", }, ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoMetadata": repoAttributeValue, + ":RepoBlobsInfo": mdAttributeValue, }, Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ + "Key": &types.AttributeValueMemberS{ Value: repo, }, }, - TableName: aws.String(dwr.RepoMetaTablename), - UpdateExpression: aws.String("SET #RM = :RepoMetadata"), + TableName: aws.String(dwr.RepoBlobsTablename), + UpdateExpression: aws.String("SET #RBI = :RepoBlobsInfo"), }) return err } -func (dwr *DynamoDB) GetRepoMeta(repo string) (mTypes.RepoMetadata, error) { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return mTypes.RepoMetadata{}, err - } +func (dwr *DynamoDB) SearchRepos(ctx context.Context, searchText string) ([]mTypes.RepoMeta, error) { + repos := []mTypes.RepoMeta{} - if resp.Item == nil { - return mTypes.RepoMetadata{}, zerr.ErrRepoMetaNotFound - } + userBookmarks := getUserBookmarks(ctx, dwr) + userStars := getUserStars(ctx, dwr) - var repoMeta mTypes.RepoMetadata + repoMetaAttributeIterator := NewBaseDynamoAttributesIterator( + dwr.Client, dwr.RepoMetaTablename, "RepoMetadata", 0, dwr.Log, + ) - err = attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { - return mTypes.RepoMetadata{}, err - } + repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) - return repoMeta, nil -} + for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { + if err != nil { + return []mTypes.RepoMeta{}, err + } -func (dwr *DynamoDB) GetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMetadata, error) { - resp, err := dwr.Client.GetItem(ctx, &dynamodb.GetItemInput{ - TableName: aws.String(dwr.RepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return mTypes.RepoMetadata{}, err - } + repoMetaBlob := []byte{} - if resp.Item == nil { - return mTypes.RepoMetadata{}, zerr.ErrRepoMetaNotFound - } + err := attributevalue.Unmarshal(repoMetaAttribute, &repoMetaBlob) + if err != nil { + return []mTypes.RepoMeta{}, err + } - var repoMeta mTypes.RepoMetadata + protoRepoMeta := &proto_go.RepoMeta{} - err = attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { - return mTypes.RepoMetadata{}, err - } + err = proto.Unmarshal(repoMetaBlob, protoRepoMeta) + if err != nil { + return []mTypes.RepoMeta{}, err + } - userData, err := dwr.GetUserData(ctx) - if err != nil { - return mTypes.RepoMetadata{}, err - } + if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { + continue + } - repoMeta.IsBookmarked = zcommon.Contains(userData.BookmarkedRepos, repo) - repoMeta.IsStarred = zcommon.Contains(userData.StarredRepos, repo) + rank := common.RankRepoName(searchText, protoRepoMeta.Name) + if rank == -1 { + continue + } - return repoMeta, nil + protoRepoMeta.Rank = int32(rank) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) + + repos = append(repos, mConvert.GetRepoMeta(protoRepoMeta)) + } + + return repos, nil } -func (dwr *DynamoDB) IncrementImageDownloads(repo string, reference string) error { - repoMeta, err := dwr.GetRepoMeta(repo) +func (dwr *DynamoDB) SearchTags(ctx context.Context, searchText string) ([]mTypes.ImageMeta, error) { + images := []mTypes.ImageMeta{} + userBookmarks := getUserBookmarks(ctx, dwr) + userStars := getUserStars(ctx, dwr) + + searchedRepo, searchedTag, err := common.GetRepoTag(searchText) if err != nil { - return err + return []mTypes.ImageMeta{}, + fmt.Errorf("metadb: error while parsing search text, invalid format %w", err) } - descriptorDigest := reference - - if !common.ReferenceIsDigest(reference) { - // search digest for tag - descriptor, found := repoMeta.Tags[reference] + if ok, err := reqCtx.RepoIsUserAvailable(ctx, searchedRepo); !ok || err != nil { + return []mTypes.ImageMeta{}, err + } - if !found { - return zerr.ErrManifestMetaNotFound + protoRepoMeta, err := dwr.getProtoRepoMeta(ctx, searchedRepo) + if err != nil { + if errors.Is(err, zerr.ErrRepoMetaNotFound) { + return []mTypes.ImageMeta{}, nil } - descriptorDigest = descriptor.Digest + return nil, err } - manifestStatistics := repoMeta.Statistics[descriptorDigest] - manifestStatistics.DownloadCount++ - repoMeta.Statistics[descriptorDigest] = manifestStatistics + delete(protoRepoMeta.Tags, "") - return dwr.SetRepoMeta(repo, repoMeta) -} + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, searchedRepo) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, searchedRepo) -func (dwr *DynamoDB) UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { - imgTrustStore := dwr.ImageTrustStore() + for tag, descriptor := range protoRepoMeta.Tags { + if !strings.HasPrefix(tag, searchedTag) { + continue + } - if imgTrustStore == nil { - return nil - } + var protoImageData *proto_go.ImageData - // get ManifestData of signed manifest - var blob []byte + switch descriptor.MediaType { + case ispec.MediaTypeImageManifest: + manifestDigest := descriptor.Digest - manifestData, err := dwr.GetManifestData(manifestDigest) - if err != nil { - if errors.Is(err, zerr.ErrManifestDataNotFound) { - indexData, err := dwr.GetIndexData(manifestDigest) + imageManifestData, err := dwr.GetProtoImageData(ctx, godigest.Digest(manifestDigest)) if err != nil { - return nil //nolint: nilerr + return []mTypes.ImageMeta{}, + fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", manifestDigest, err) } - blob = indexData.IndexBlob - } else { - return fmt.Errorf("%w for manifest '%s' from repo '%s'", errMetaDB, manifestDigest, repo) - } - } else { - blob = manifestData.ManifestBlob - } - - // update signatures with details about validity and author - repoMeta, err := dwr.GetRepoMeta(repo) - if err != nil { - return err - } - - manifestSignatures := mTypes.ManifestSignatures{} - - for sigType, sigs := range repoMeta.Signatures[manifestDigest.String()] { - signaturesInfo := []mTypes.SignatureInfo{} + protoImageData = imageManifestData + case ispec.MediaTypeImageIndex: + indexDigest := godigest.Digest(descriptor.Digest) - for _, sigInfo := range sigs { - layersInfo := []mTypes.LayerInfo{} + imageIndexData, err := dwr.GetProtoImageData(ctx, indexDigest) + if err != nil { + return []mTypes.ImageMeta{}, + fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", indexDigest, err) + } - for _, layerInfo := range sigInfo.LayersInfo { - author, date, isTrusted, _ := imgTrustStore.VerifySignature(sigType, layerInfo.LayerContent, layerInfo.SignatureKey, - manifestDigest, blob, repo) + manifestDataList := make([]*proto_go.ManifestData, 0, len(imageIndexData.Index.Index.Manifests)) - if isTrusted { - layerInfo.Signer = author - } + for _, manifest := range imageIndexData.Index.Index.Manifests { + manifestDigest := godigest.Digest(manifest.Digest) - if !date.IsZero() { - layerInfo.Signer = author - layerInfo.Date = date + imageManifestData, err := dwr.GetProtoImageData(ctx, manifestDigest) + if err != nil { + return []mTypes.ImageMeta{}, err } - layersInfo = append(layersInfo, layerInfo) + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) } - signaturesInfo = append(signaturesInfo, mTypes.SignatureInfo{ - SignatureManifestDigest: sigInfo.SignatureManifestDigest, - LayersInfo: layersInfo, - }) + imageIndexData.Manifests = manifestDataList + + protoImageData = imageIndexData + default: + dwr.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + + continue } - manifestSignatures[sigType] = signaturesInfo + images = append(images, mConvert.GetImageMetaFromProto(tag, protoRepoMeta, protoImageData)) } - repoMeta.Signatures[manifestDigest.String()] = manifestSignatures - - return dwr.SetRepoMeta(repoMeta.Name, repoMeta) + return images, err } -func (dwr *DynamoDB) AddManifestSignature(repo string, signedManifestDigest godigest.Digest, - sygMeta mTypes.SignatureMetadata, -) error { - repoMeta, err := dwr.GetRepoMeta(repo) - if err != nil { - if errors.Is(err, zerr.ErrRepoMetaNotFound) { - repoMeta = mTypes.RepoMetadata{ - Name: repo, - Tags: map[string]mTypes.Descriptor{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - Signatures: map[string]mTypes.ManifestSignatures{ - signedManifestDigest.String(): { - sygMeta.SignatureType: []mTypes.SignatureInfo{ - { - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: sygMeta.LayersInfo, - }, - }, - }, - }, - Referrers: map[string][]mTypes.ReferrerInfo{}, - } - - return dwr.SetRepoMeta(repo, repoMeta) - } +func (dwr *DynamoDB) FilterTags(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, + filterFunc mTypes.FilterFunc, +) ([]mTypes.ImageMeta, error) { + images := []mTypes.ImageMeta{} + userBookmarks := getUserBookmarks(ctx, dwr) + userStars := getUserStars(ctx, dwr) - return err - } + var viewError error - var ( - manifestSignatures mTypes.ManifestSignatures - found bool + repoMetaAttributeIterator := NewBaseDynamoAttributesIterator( + dwr.Client, dwr.RepoMetaTablename, "RepoMetadata", 0, dwr.Log, ) - if manifestSignatures, found = repoMeta.Signatures[signedManifestDigest.String()]; !found { - manifestSignatures = mTypes.ManifestSignatures{} - } + repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) - signatureSlice := manifestSignatures[sygMeta.SignatureType] - if !common.SignatureAlreadyExists(signatureSlice, sygMeta) { - if sygMeta.SignatureType == zcommon.NotationSignature { - signatureSlice = append(signatureSlice, mTypes.SignatureInfo{ - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: sygMeta.LayersInfo, - }) - } else if sygMeta.SignatureType == zcommon.CosignSignature { - signatureSlice = []mTypes.SignatureInfo{{ - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: sygMeta.LayersInfo, - }} + for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { + if err != nil { + viewError = errors.Join(viewError, err) + + continue } - } - manifestSignatures[sygMeta.SignatureType] = signatureSlice + protoRepoMeta, err := getProtoRepoMetaFromAttribute(repoMetaAttribute) + if err != nil { + viewError = errors.Join(viewError, err) - repoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures + continue + } - return dwr.SetRepoMeta(repoMeta.Name, repoMeta) -} + if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { + continue + } -func (dwr *DynamoDB) DeleteSignature(repo string, signedManifestDigest godigest.Digest, - sigMeta mTypes.SignatureMetadata, -) error { - repoMeta, err := dwr.GetRepoMeta(repo) - if err != nil { - return err - } + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) + repoMeta := mConvert.GetRepoMeta(protoRepoMeta) - sigType := sigMeta.SignatureType + for tag, descriptor := range repoMeta.Tags { + if !filterRepoTag(repoMeta.Name, tag) { + continue + } - var ( - manifestSignatures mTypes.ManifestSignatures - found bool - ) - - if manifestSignatures, found = repoMeta.Signatures[signedManifestDigest.String()]; !found { - return zerr.ErrManifestMetaNotFound - } - - signatureSlice := manifestSignatures[sigType] - - newSignatureSlice := make([]mTypes.SignatureInfo, 0, len(signatureSlice)-1) - - for _, sigDigest := range signatureSlice { - if sigDigest.SignatureManifestDigest != sigMeta.SignatureDigest { - newSignatureSlice = append(newSignatureSlice, sigDigest) - } - } - - manifestSignatures[sigType] = newSignatureSlice - - repoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures - - err = dwr.SetRepoMeta(repoMeta.Name, repoMeta) - - return err -} - -func (dwr *DynamoDB) GetMultipleRepoMeta(ctx context.Context, - filter func(repoMeta mTypes.RepoMetadata) bool, -) ([]mTypes.RepoMetadata, error) { - var ( - foundRepos = []mTypes.RepoMetadata{} - repoMetaAttributeIterator AttributesIterator - ) - - repoMetaAttributeIterator = NewBaseDynamoAttributesIterator( - dwr.Client, dwr.RepoMetaTablename, "RepoMetadata", 0, dwr.Log, - ) - - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) - - for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { - if err != nil { - return []mTypes.RepoMetadata{}, err - } - - var repoMeta mTypes.RepoMetadata - - err := attributevalue.Unmarshal(repoMetaAttribute, &repoMeta) - if err != nil { - return []mTypes.RepoMetadata{}, err - } - - if ok, err := reqCtx.RepoIsUserAvailable(ctx, repoMeta.Name); !ok || err != nil { - continue - } - - if filter(repoMeta) { - foundRepos = append(foundRepos, repoMeta) - } - } - - return foundRepos, err -} - -func (dwr *DynamoDB) SearchRepos(ctx context.Context, searchText string, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - var ( - repos = []mTypes.RepoMetadata{} - manifestMetadataMap = make(map[string]mTypes.ManifestMetadata) - indexDataMap = make(map[string]mTypes.IndexData) - repoMetaAttributeIterator AttributesIterator - - userBookmarks = getUserBookmarks(ctx, dwr) - userStars = getUserStars(ctx, dwr) - ) - - repoMetaAttributeIterator = NewBaseDynamoAttributesIterator( - dwr.Client, dwr.RepoMetaTablename, "RepoMetadata", 0, dwr.Log, - ) - - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) - - for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err - } - - var repoMeta mTypes.RepoMetadata - - err := attributevalue.Unmarshal(repoMetaAttribute, &repoMeta) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err - } - - if ok, err := reqCtx.RepoIsUserAvailable(ctx, repoMeta.Name); !ok || err != nil { - continue - } - - rank := common.RankRepoName(searchText, repoMeta.Name) - if rank == -1 { - continue - } - - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) - repoMeta.Rank = rank - - for _, descriptor := range repoMeta.Tags { switch descriptor.MediaType { case ispec.MediaTypeImageManifest: manifestDigest := descriptor.Digest - manifestMeta, err := dwr.fetchManifestMetaWithCheck(repoMeta.Name, manifestDigest, //nolint:contextcheck - manifestMetadataMap) + imageManifestData, err := dwr.GetProtoImageData(ctx, godigest.Digest(manifestDigest)) if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err + viewError = errors.Join(viewError, err) + + continue } - manifestMetadataMap[descriptor.Digest] = manifestMeta + imageData := mConvert.GetImageData(imageManifestData) + + if filterFunc(repoMeta, imageData) { + images = append(images, mConvert.GetImageMetaFromProto(tag, protoRepoMeta, imageManifestData)) + } case ispec.MediaTypeImageIndex: - indexData, err := dwr.fetchIndexDataWithCheck(descriptor.Digest, indexDataMap) //nolint:contextcheck + indexDigest := descriptor.Digest + + imageIndexData, err := dwr.GetProtoImageData(ctx, godigest.Digest(indexDigest)) if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err + viewError = errors.Join(viewError, err) + + continue } - var indexContent ispec.Index + matchedManifests := []*proto_go.ManifestData{} - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("metadb: error while unmarshaling index content for digest %s %w", descriptor.Digest, err) - } + for _, manifest := range imageIndexData.Index.Index.Manifests { + manifestDigest := manifest.Digest - for _, manifest := range indexContent.Manifests { - manifestMeta, err := dwr.fetchManifestMetaWithCheck(repoMeta.Name, manifest.Digest.String(), //nolint: contextcheck - manifestMetadataMap) + imageManifestData, err := dwr.GetProtoImageData(ctx, godigest.Digest(manifestDigest)) if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err + viewError = errors.Join(viewError, err) + + continue } - manifestMetadataMap[manifest.Digest.String()] = manifestMeta + imageData := mConvert.GetImageData(imageManifestData) + + if filterFunc(repoMeta, imageData) { + matchedManifests = append(matchedManifests, imageManifestData.Manifests[0]) + } } - indexDataMap[descriptor.Digest] = indexData + if len(matchedManifests) > 0 { + imageIndexData.Manifests = matchedManifests + + images = append(images, mConvert.GetImageMetaFromProto(tag, protoRepoMeta, imageIndexData)) + } default: dwr.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") continue } } - - repos = append(repos, repoMeta) } - return repos, manifestMetadataMap, indexDataMap, nil + viewError = errors.Join(viewError, err) + + return images, viewError } -func getUserStars(ctx context.Context, dwr *DynamoDB) []string { - starredRepos, err := dwr.GetStarredRepos(ctx) +func getProtoRepoMetaFromAttribute(repoMetaAttribute types.AttributeValue) (*proto_go.RepoMeta, error) { + blob := []byte{} + + err := attributevalue.Unmarshal(repoMetaAttribute, &blob) if err != nil { - return []string{} + return nil, err } - return starredRepos -} + protoRepoMeta := &proto_go.RepoMeta{} -func getUserBookmarks(ctx context.Context, dwr *DynamoDB) []string { - bookmarkedRepos, err := dwr.GetBookmarkedRepos(ctx) + err = proto.Unmarshal(blob, protoRepoMeta) if err != nil { - return []string{} + return nil, err } - return bookmarkedRepos + return protoRepoMeta, nil } -func (dwr *DynamoDB) fetchManifestMetaWithCheck(repoName string, manifestDigest string, - manifestMetadataMap map[string]mTypes.ManifestMetadata, -) (mTypes.ManifestMetadata, error) { - var ( - manifestMeta mTypes.ManifestMetadata - err error - ) - - manifestMeta, manifestDownloaded := manifestMetadataMap[manifestDigest] +func getProtoImageDataFromAttribute(imageDataAttribute types.AttributeValue) (*proto_go.ImageData, error) { + blob := []byte{} - if !manifestDownloaded { - manifestMeta, err = dwr.GetManifestMeta(repoName, godigest.Digest(manifestDigest)) //nolint:contextcheck - if err != nil { - return mTypes.ManifestMetadata{}, err - } + err := attributevalue.Unmarshal(imageDataAttribute, &blob) + if err != nil { + return nil, err } - return manifestMeta, nil -} - -func (dwr *DynamoDB) fetchIndexDataWithCheck(indexDigest string, indexDataMap map[string]mTypes.IndexData, -) (mTypes.IndexData, error) { - var ( - indexData mTypes.IndexData - err error - ) - - indexData, indexExists := indexDataMap[indexDigest] + protoImageData := &proto_go.ImageData{} - if !indexExists { - indexData, err = dwr.GetIndexData(godigest.Digest(indexDigest)) //nolint:contextcheck - if err != nil { - return mTypes.IndexData{}, - fmt.Errorf("metadb: error while unmarshaling index data for digest %s \n%w", indexDigest, err) - } + err = proto.Unmarshal(blob, protoImageData) + if err != nil { + return nil, err } - return indexData, err + return protoImageData, nil } -func (dwr *DynamoDB) FilterTags(ctx context.Context, filterFunc mTypes.FilterFunc, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error, -) { - var ( - foundRepos = make([]mTypes.RepoMetadata, 0) - manifestMetadataMap = make(map[string]mTypes.ManifestMetadata) - indexDataMap = make(map[string]mTypes.IndexData) - repoMetaAttributeIterator AttributesIterator - userBookmarks = getUserBookmarks(ctx, dwr) - userStars = getUserStars(ctx, dwr) - aggregateError error - ) - - repoMetaAttributeIterator = NewBaseDynamoAttributesIterator( - dwr.Client, dwr.RepoMetaTablename, "RepoMetadata", 0, dwr.Log, - ) - - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) +func (dwr *DynamoDB) ResetRepoReferences(repo string) error { + protoRepoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) if err != nil { - return foundRepos, manifestMetadataMap, indexDataMap, err + return err } - for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { - if err != nil { - aggregateError = errors.Join(aggregateError, err) - - continue - } - - var repoMeta mTypes.RepoMetadata - - err := attributevalue.Unmarshal(repoMetaAttribute, &repoMeta) - if err != nil { - aggregateError = errors.Join(aggregateError, err) - - continue - } - - if ok, err := reqCtx.RepoIsUserAvailable(ctx, repoMeta.Name); !ok || err != nil { - continue - } + return dwr.setProtoRepoMeta(repo, &proto_go.RepoMeta{ + Name: repo, + Statistics: protoRepoMeta.Statistics, + Stars: protoRepoMeta.Stars, + Tags: map[string]*proto_go.TagDescriptor{"": {}}, + Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, + Signatures: map[string]*proto_go.ManifestSignatures{"": {Map: map[string]*proto_go.SignaturesInfo{"": {}}}}, + }) +} - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) +func (dwr *DynamoDB) GetRepoMeta(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + protoRepoMeta, err := dwr.getProtoRepoMeta(ctx, repo) + if err != nil { + return mTypes.RepoMeta{}, err + } - matchedTags := make(map[string]mTypes.Descriptor) + delete(protoRepoMeta.Tags, "") - for tag, descriptor := range repoMeta.Tags { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest + userBookmarks := getUserBookmarks(ctx, dwr) + userStars := getUserStars(ctx, dwr) - manifestMeta, err := dwr.fetchManifestMetaWithCheck(repoMeta.Name, manifestDigest, //nolint:contextcheck - manifestMetadataMap) - if err != nil { - err = fmt.Errorf("metadb: error while unmashaling manifest metadata for digest %s \n%w", manifestDigest, err) - aggregateError = errors.Join(aggregateError, err) + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repo) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, repo) - continue - } + return mConvert.GetRepoMeta(protoRepoMeta), nil +} - if filterFunc(repoMeta, manifestMeta) { - matchedTags[tag] = descriptor - manifestMetadataMap[manifestDigest] = manifestMeta - } - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest +func (dwr *DynamoDB) GetImageMeta(ctx context.Context, repo string, tag string) (mTypes.ImageMeta, error) { + protoRepoMeta, err := dwr.getProtoRepoMeta(ctx, repo) + if err != nil { + return mTypes.ImageMeta{}, err + } - indexData, err := dwr.fetchIndexDataWithCheck(indexDigest, indexDataMap) //nolint:contextcheck - if err != nil { - err = fmt.Errorf("metadb: error while getting index data for digest %s %w", indexDigest, err) - aggregateError = errors.Join(aggregateError, err) + delete(protoRepoMeta.Tags, "") - continue - } + bookmarks, stars := dwr.getUserBookmarksAndStars(ctx) - var indexContent ispec.Index + protoRepoMeta.IsBookmarked = zcommon.Contains(bookmarks, repo) + protoRepoMeta.IsStarred = zcommon.Contains(stars, repo) - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - err = fmt.Errorf("metadb: error while unmashaling index content for digest %s %w", indexDigest, err) - aggregateError = errors.Join(aggregateError, err) + descriptor, ok := protoRepoMeta.Tags[tag] + if !ok { + return mTypes.ImageMeta{}, zerr.ErrImageDataNotFound + } - continue - } + protoImageData, err := dwr.GetProtoImageData(ctx, godigest.Digest(descriptor.Digest)) + if err != nil { + return mTypes.ImageMeta{}, err + } - matchedManifests := []ispec.Descriptor{} + if protoImageData.MediaType == ispec.MediaTypeImageIndex { + manifestDataList := make([]*proto_go.ManifestData, 0, len(protoImageData.Index.Index.Manifests)) - for _, manifest := range indexContent.Manifests { - manifestDigest := manifest.Digest.String() + for _, manifest := range protoImageData.Index.Index.Manifests { + imageManifestData, err := dwr.GetProtoImageData(ctx, godigest.Digest(manifest.Digest)) + if err != nil { + return mTypes.ImageMeta{}, err + } - manifestMeta, err := dwr.fetchManifestMetaWithCheck(repoMeta.Name, manifestDigest, //nolint:contextcheck - manifestMetadataMap) - if err != nil { - err = fmt.Errorf("%w metadb: error while getting manifest data for digest %s", err, manifestDigest) - aggregateError = errors.Join(aggregateError, err) + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) + } - continue - } + protoImageData.Manifests = manifestDataList + } - if filterFunc(repoMeta, manifestMeta) { - matchedManifests = append(matchedManifests, manifest) - manifestMetadataMap[manifestDigest] = manifestMeta - } - } + return mConvert.GetImageMetaFromProto(tag, protoRepoMeta, protoImageData), nil +} - if len(matchedManifests) > 0 { - indexContent.Manifests = matchedManifests +func (dwr *DynamoDB) getUserBookmarksAndStars(ctx context.Context) ([]string, []string) { + userData, err := dwr.GetUserData(ctx) + if err != nil { + return []string{}, []string{} + } - indexBlob, err := json.Marshal(indexContent) - if err != nil { - aggregateError = errors.Join(aggregateError, err) + return userData.BookmarkedRepos, userData.StarredRepos +} - continue - } +func (dwr *DynamoDB) GetImageData(digest godigest.Digest) (mTypes.ImageData, error) { + protoImageData, err := dwr.GetProtoImageData(context.Background(), digest) + if err != nil { + return mTypes.ImageData{}, err + } - indexData.IndexBlob = indexBlob + if protoImageData.MediaType == ispec.MediaTypeImageIndex { + manifestDataList := make([]*proto_go.ManifestData, 0, len(protoImageData.Index.Index.Manifests)) - indexDataMap[indexDigest] = indexData - matchedTags[tag] = descriptor - } - default: - dwr.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + for _, manifest := range protoImageData.Index.Index.Manifests { + manifestDigest := godigest.Digest(manifest.Digest) - continue + imageManifestData, err := dwr.GetProtoImageData(context.Background(), manifestDigest) + if err != nil { + return mTypes.ImageData{}, err } - } - if len(matchedTags) == 0 { - continue + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) } - repoMeta.Tags = matchedTags - - foundRepos = append(foundRepos, repoMeta) + protoImageData.Manifests = manifestDataList } - return foundRepos, manifestMetadataMap, indexDataMap, aggregateError + return mConvert.GetImageData(protoImageData), nil } -func (dwr *DynamoDB) FilterRepos(ctx context.Context, filter mTypes.FilterRepoFunc, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { +func (dwr *DynamoDB) GetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMeta) bool, +) ([]mTypes.RepoMeta, error) { var ( - foundRepos = []mTypes.RepoMetadata{} + foundRepos = []mTypes.RepoMeta{} repoMetaAttributeIterator AttributesIterator - userBookmarks = getUserBookmarks(ctx, dwr) - userStars = getUserStars(ctx, dwr) ) repoMetaAttributeIterator = NewBaseDynamoAttributesIterator( @@ -1237,2162 +796,1236 @@ func (dwr *DynamoDB) FilterRepos(ctx context.Context, filter mTypes.FilterRepoFu for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err + return []mTypes.RepoMeta{}, err } - var repoMeta mTypes.RepoMetadata + repoMetaBlob := []byte{} - err := attributevalue.Unmarshal(repoMetaAttribute, &repoMeta) + err := attributevalue.Unmarshal(repoMetaAttribute, &repoMetaBlob) if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err + return []mTypes.RepoMeta{}, err + } + + protoRepoMeta := &proto_go.RepoMeta{} + + err = proto.Unmarshal(repoMetaBlob, protoRepoMeta) + if err != nil { + return []mTypes.RepoMeta{}, err } - if ok, err := reqCtx.RepoIsUserAvailable(ctx, repoMeta.Name); !ok || err != nil { + delete(protoRepoMeta.Tags, "") + + if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { continue } - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) + repoMeta := mConvert.GetRepoMeta(protoRepoMeta) if filter(repoMeta) { foundRepos = append(foundRepos, repoMeta) } } - foundManifestMetadataMap, foundIndexDataMap, err := common.FetchDataForRepos(dwr, foundRepos) - - return foundRepos, foundManifestMetadataMap, foundIndexDataMap, err + return foundRepos, err } -func (dwr *DynamoDB) SearchTags(ctx context.Context, searchText string, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error, -) { - var ( - foundRepos = make([]mTypes.RepoMetadata, 0, 1) - manifestMetadataMap = make(map[string]mTypes.ManifestMetadata) - indexDataMap = make(map[string]mTypes.IndexData) - repoMetaAttributeIterator AttributesIterator - userBookmarks = getUserBookmarks(ctx, dwr) - userStars = getUserStars(ctx, dwr) - ) +func (dwr *DynamoDB) FilterRepos(ctx context.Context, acceptName mTypes.FilterRepoNameFunc, + filterFunc mTypes.FilterFullRepoFunc, +) ([]mTypes.RepoMeta, error) { + repos := []mTypes.RepoMeta{} + userBookmarks := getUserBookmarks(ctx, dwr) + userStars := getUserStars(ctx, dwr) - repoMetaAttributeIterator = NewBaseDynamoAttributesIterator( + repoMetaAttributeIterator := NewBaseDynamoAttributesIterator( dwr.Client, dwr.RepoMetaTablename, "RepoMetadata", 0, dwr.Log, ) - searchedRepo, searchedTag, err := common.GetRepoTag(searchText) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("metadb: error while parsing search text, invalid format %w", err) - } - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err - } - var repoMeta mTypes.RepoMetadata - - err = attributevalue.Unmarshal(repoMetaAttribute, &repoMeta) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err - } - - if ok, err := reqCtx.RepoIsUserAvailable(ctx, repoMeta.Name); !ok || err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err - } - - if repoMeta.Name != searchedRepo { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err - } - - repoMeta.IsBookmarked = zcommon.Contains(userBookmarks, repoMeta.Name) - repoMeta.IsStarred = zcommon.Contains(userStars, repoMeta.Name) + for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { + if err != nil { + return []mTypes.RepoMeta{}, + err + } - matchedTags := make(map[string]mTypes.Descriptor) + protoRepoMeta, err := getProtoRepoMetaFromAttribute(repoMetaAttribute) + if err != nil { + return nil, err + } - for tag, descriptor := range repoMeta.Tags { - if !strings.HasPrefix(tag, searchedTag) { + if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { continue } - matchedTags[tag] = descriptor - - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest - - manifestMeta, err := dwr.fetchManifestMetaWithCheck(repoMeta.Name, manifestDigest, //nolint:contextcheck - manifestMetadataMap) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("metadb: error while unmashaling manifest metadata for digest %s %w", descriptor.Digest, err) - } - - manifestMetadataMap[descriptor.Digest] = manifestMeta - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest + if !acceptName(protoRepoMeta.Name) { + continue + } - indexData, err := dwr.fetchIndexDataWithCheck(indexDigest, indexDataMap) //nolint:contextcheck - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("%w", err) - } + protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) + protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) - var indexContent ispec.Index + fullRepoMeta := mConvert.GetRepoMeta(protoRepoMeta) - err = json.Unmarshal(indexData.IndexBlob, &indexContent) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("metadb: error while unmashaling index content for digest %s %w", indexDigest, err) - } + if filterFunc(fullRepoMeta) { + repos = append(repos, fullRepoMeta) + } + } - for _, manifest := range indexContent.Manifests { - manifestDigest := manifest.Digest.String() + return repos, err +} - manifestMeta, err := dwr.fetchManifestMetaWithCheck(repoMeta.Name, manifestDigest, //nolint:contextcheck - manifestMetadataMap) - if err != nil { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - fmt.Errorf("%w", err) - } +func (dwr *DynamoDB) IncrementRepoStars(repo string) error { + repoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) + if err != nil { + return err + } - manifestMetadataMap[manifestDigest] = manifestMeta - } + repoMeta.Stars++ - indexDataMap[indexDigest] = indexData - default: - dwr.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + return dwr.setProtoRepoMeta(repo, repoMeta) +} - continue - } +func (dwr *DynamoDB) DecrementRepoStars(repo string) error { + repoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) + if err != nil { + return err } - if len(matchedTags) == 0 { - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, map[string]mTypes.IndexData{}, - err + if repoMeta.Stars > 0 { + repoMeta.Stars-- } - repoMeta.Tags = matchedTags - - foundRepos = append(foundRepos, repoMeta) - - return foundRepos, manifestMetadataMap, indexDataMap, err + return dwr.setProtoRepoMeta(repo, repoMeta) } -func (dwr *DynamoDB) PatchDB() error { - DBVersion, err := dwr.getDBVersion() +func (dwr *DynamoDB) GetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + protoRepoMeta, err := dwr.getProtoRepoMeta(ctx, repo) if err != nil { - return fmt.Errorf("patching dynamo failed, error retrieving database version %w", err) + return mTypes.RepoMeta{}, err } - if version.GetVersionIndex(DBVersion) == -1 { - return fmt.Errorf("DB has broken format, no version found %w", err) - } + bookmarks, stars := dwr.getUserBookmarksAndStars(ctx) - for patchIndex, patch := range dwr.Patches { - if patchIndex < version.GetVersionIndex(DBVersion) { - continue - } + protoRepoMeta.IsBookmarked = zcommon.Contains(bookmarks, repo) + protoRepoMeta.IsStarred = zcommon.Contains(stars, repo) - tableNames := map[string]string{ - "RepoMetaTablename": dwr.RepoMetaTablename, - "ManifestDataTablename": dwr.ManifestDataTablename, - "VersionTablename": dwr.VersionTablename, - } + return mConvert.GetRepoMeta(protoRepoMeta), nil +} - err := patch(dwr.Client, tableNames) - if err != nil { - return err - } - } +func (dwr *DynamoDB) SetRepoMeta(repo string, repoMeta mTypes.RepoMeta) error { + protoRepoMeta := mConvert.GetProtoRepoMeta(repoMeta) - return nil + return dwr.setProtoRepoMeta(repo, protoRepoMeta) } -func (dwr *DynamoDB) SetRepoMeta(repo string, repoMeta mTypes.RepoMetadata) error { - repoMeta.Name = repo - - repoAttributeValue, err := attributevalue.Marshal(repoMeta) +func (dwr *DynamoDB) GetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string, +) ([]mTypes.ReferrerInfo, error) { + repoMeta, err := dwr.GetRepoMeta(context.Background(), repo) if err != nil { - return err + return []mTypes.ReferrerInfo{}, err } - _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#RM": "RepoMetadata", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoMetadata": repoAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ - Value: repo, - }, - }, - TableName: aws.String(dwr.RepoMetaTablename), - UpdateExpression: aws.String("SET #RM = :RepoMetadata"), - }) + referrersInfo := repoMeta.Referrers[referredDigest.String()] - return err -} + filteredResults := make([]mTypes.ReferrerInfo, 0, len(referrersInfo)) -func (dwr *DynamoDB) createRepoMetaTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.RepoMetaTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("RepoName"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("RepoName"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) - if err != nil { - if strings.Contains(err.Error(), "Table already exists") { - return nil + for _, referrerInfo := range referrersInfo { + if !common.MatchesArtifactTypes(referrerInfo.ArtifactType, artifactTypes) { + continue } - return err + filteredResults = append(filteredResults, referrerInfo) } - return dwr.waitTableToBeCreated(dwr.RepoMetaTablename) + return filteredResults, nil } -func (dwr *DynamoDB) createProtoRepoMetaTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.ProtoRepoMetaTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("RepoName"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("RepoName"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) - - if err != nil && !strings.Contains(err.Error(), "Table already exists") { +func (dwr *DynamoDB) IncrementImageDownloads(repo string, reference string) error { + repoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) + if err != nil { return err } - return dwr.waitTableToBeCreated(dwr.ProtoRepoMetaTablename) -} - -func (dwr *DynamoDB) createProtoRepoBlobsTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.ProtoRepoBlobsTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("RepoName"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("RepoName"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) + descriptorDigest := reference - if err != nil && !strings.Contains(err.Error(), "Table already exists") { - return err - } + if !common.ReferenceIsDigest(reference) { + // search digest for tag + descriptor, found := repoMeta.Tags[reference] - return dwr.waitTableToBeCreated(dwr.ProtoRepoMetaTablename) -} + if !found { + return zerr.ErrManifestMetaNotFound + } -func (dwr *DynamoDB) deleteRepoMetaTable() error { - _, err := dwr.Client.DeleteTable(context.Background(), &dynamodb.DeleteTableInput{ - TableName: aws.String(dwr.RepoMetaTablename), - }) + descriptorDigest = descriptor.Digest + } - if temp := new(types.ResourceNotFoundException); errors.As(err, &temp) { - return nil + manifestStatistics, ok := repoMeta.Statistics[descriptorDigest] + if !ok { + return zerr.ErrManifestMetaNotFound } - return dwr.waitTableToBeDeleted(dwr.RepoMetaTablename) + manifestStatistics.DownloadCount++ + repoMeta.Statistics[descriptorDigest] = manifestStatistics + + return dwr.setProtoRepoMeta(repo, repoMeta) } -func (dwr *DynamoDB) deleteProtoRepoMetaTable() error { - _, err := dwr.Client.DeleteTable(context.Background(), &dynamodb.DeleteTableInput{ - TableName: aws.String(dwr.ProtoRepoMetaTablename), - }) +func (dwr *DynamoDB) UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { + imgTrustStore := dwr.ImageTrustStore() - if temp := new(types.ResourceNotFoundException); errors.As(err, &temp) { + if imgTrustStore == nil { return nil } - return dwr.waitTableToBeDeleted(dwr.ProtoRepoMetaTablename) -} - -func (dwr *DynamoDB) ResetRepoMetaTable() error { - err := dwr.deleteRepoMetaTable() + protoImageData, err := dwr.GetProtoImageData(context.Background(), manifestDigest) if err != nil { return err } - return dwr.createRepoMetaTable() -} - -func (dwr *DynamoDB) ResetRepoRepoMetaTable() error { - err := dwr.deleteProtoRepoMetaTable() + // update signatures with details about validity and author + protoRepoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) if err != nil { return err } - return dwr.createRepoMetaTable() -} + manifestSignatures := proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} -func (dwr *DynamoDB) createProtoImageDataTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.ProtoImageDataTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("Digest"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("Digest"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) + for sigType, sigs := range protoRepoMeta.Signatures[manifestDigest.String()].Map { + signaturesInfo := []*proto_go.SignatureInfo{} - if err != nil && !strings.Contains(err.Error(), "Table already exists") { - return err - } + for _, sigInfo := range sigs.List { + layersInfo := []*proto_go.LayersInfo{} - return dwr.waitTableToBeCreated(dwr.ProtoImageDataTablename) -} + for _, layerInfo := range sigInfo.LayersInfo { + author, date, isTrusted, _ := imgTrustStore.VerifySignature(sigType, layerInfo.LayerContent, + layerInfo.SignatureKey, manifestDigest, mConvert.GetImageData(protoImageData), repo) -func (dwr *DynamoDB) waitTableToBeCreated(tableName string) error { - const maxWaitTime = 20 * time.Second + if isTrusted { + layerInfo.Signer = author + } - waiter := dynamodb.NewTableExistsWaiter(dwr.Client) + if !date.IsZero() { + layerInfo.Signer = author + layerInfo.Date = timestamppb.New(date) + } - return waiter.Wait(context.Background(), &dynamodb.DescribeTableInput{ - TableName: &tableName, - }, maxWaitTime) -} + layersInfo = append(layersInfo, layerInfo) + } -func (dwr *DynamoDB) waitTableToBeDeleted(tableName string) error { - const maxWaitTime = 20 * time.Second + signaturesInfo = append(signaturesInfo, &proto_go.SignatureInfo{ + SignatureManifestDigest: sigInfo.SignatureManifestDigest, + LayersInfo: layersInfo, + }) + } - waiter := dynamodb.NewTableNotExistsWaiter(dwr.Client) + manifestSignatures.Map[sigType] = &proto_go.SignaturesInfo{List: signaturesInfo} + } - return waiter.Wait(context.Background(), &dynamodb.DescribeTableInput{ - TableName: &tableName, - }, maxWaitTime) + protoRepoMeta.Signatures[manifestDigest.String()] = &manifestSignatures + + return dwr.setProtoRepoMeta(protoRepoMeta.Name, protoRepoMeta) } -func (dwr *DynamoDB) createManifestDataTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.ManifestDataTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("Digest"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("Digest"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) - if err != nil { - if strings.Contains(err.Error(), "Table already exists") { - return nil - } - - return err - } - - return dwr.waitTableToBeCreated(dwr.ManifestDataTablename) -} - -func (dwr *DynamoDB) createIndexDataTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.IndexDataTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("IndexDigest"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("IndexDigest"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) - if err != nil { - if strings.Contains(err.Error(), "Table already exists") { - return nil - } - - return err - } - - return dwr.waitTableToBeCreated(dwr.IndexDataTablename) -} - -func (dwr *DynamoDB) createVersionTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.VersionTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("VersionKey"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("VersionKey"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) - if err != nil { - if strings.Contains(err.Error(), "Table already exists") { - return nil - } - - return err - } - - err = dwr.waitTableToBeCreated(dwr.VersionTablename) +func (dwr *DynamoDB) AddManifestSignature(repo string, signedManifestDigest godigest.Digest, + sygMeta mTypes.SignatureMetadata, +) error { + protoRepoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) if err != nil { - return err - } - - if err == nil { - mdAttributeValue, err := attributevalue.Marshal(version.CurrentVersion) - if err != nil { - return err - } - - _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#V": "Version", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":Version": mdAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "VersionKey": &types.AttributeValueMemberS{ - Value: version.DBVersionKey, + if errors.Is(err, zerr.ErrRepoMetaNotFound) { + protoRepoMeta = &proto_go.RepoMeta{ + Name: repo, + Tags: map[string]*proto_go.TagDescriptor{"": {}}, + Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, + Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, + Signatures: map[string]*proto_go.ManifestSignatures{ + signedManifestDigest.String(): { + Map: map[string]*proto_go.SignaturesInfo{ + sygMeta.SignatureType: { + List: []*proto_go.SignatureInfo{ + { + SignatureManifestDigest: sygMeta.SignatureDigest, + LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), + }, + }, + }, + }, + }, }, - }, - TableName: aws.String(dwr.VersionTablename), - UpdateExpression: aws.String("SET #V = :Version"), - }) + } - if err != nil { - return err + return dwr.setProtoRepoMeta(repo, protoRepoMeta) } - } - - return nil -} - -func (dwr *DynamoDB) getDBVersion() (string, error) { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.VersionTablename), - Key: map[string]types.AttributeValue{ - "VersionKey": &types.AttributeValueMemberS{Value: version.DBVersionKey}, - }, - }) - if err != nil { - return "", err - } - if resp.Item == nil { - return "", nil - } - - var version string - - err = attributevalue.Unmarshal(resp.Item["Version"], &version) - if err != nil { - return "", err - } - - return version, nil -} - -func (dwr *DynamoDB) deleteManifestDataTable() error { - _, err := dwr.Client.DeleteTable(context.Background(), &dynamodb.DeleteTableInput{ - TableName: aws.String(dwr.ManifestDataTablename), - }) - - if temp := new(types.ResourceNotFoundException); errors.As(err, &temp) { - return nil - } - - return dwr.waitTableToBeDeleted(dwr.ManifestDataTablename) -} - -func (dwr *DynamoDB) ResetManifestDataTable() error { - err := dwr.deleteManifestDataTable() - if err != nil { return err } - return dwr.createManifestDataTable() -} - -func (dwr *DynamoDB) ToggleBookmarkRepo(ctx context.Context, repo string) ( - mTypes.ToggleState, error, -) { - res := mTypes.NotChanged - - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return mTypes.NotChanged, err - } - - if userAc.IsAnonymous() || !userAc.Can(constants.ReadPermission, repo) { - return mTypes.NotChanged, zerr.ErrUserDataNotAllowed - } - - userData, err := dwr.GetUserData(ctx) - if err != nil { - if errors.Is(err, zerr.ErrUserDataNotFound) { - return mTypes.NotChanged, nil - } - - return res, err - } - - if !zcommon.Contains(userData.BookmarkedRepos, repo) { - userData.BookmarkedRepos = append(userData.BookmarkedRepos, repo) - res = mTypes.Added - } else { - userData.BookmarkedRepos = zcommon.RemoveFrom(userData.BookmarkedRepos, repo) - res = mTypes.Removed - } - - if res != mTypes.NotChanged { - err = dwr.SetUserData(ctx, userData) - } - - if err != nil { - res = mTypes.NotChanged - - return res, err - } - - return res, nil -} - -func (dwr *DynamoDB) GetBookmarkedRepos(ctx context.Context) ([]string, error) { - userMeta, err := dwr.GetUserData(ctx) - - if errors.Is(err, zerr.ErrUserDataNotFound) || errors.Is(err, zerr.ErrUserDataNotAllowed) { - return []string{}, nil - } - - return userMeta.BookmarkedRepos, err -} - -func (dwr *DynamoDB) ToggleStarRepo(ctx context.Context, repo string) ( - mTypes.ToggleState, error, -) { - res := mTypes.NotChanged - - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return mTypes.NotChanged, err - } - - if userAc.IsAnonymous() || !userAc.Can(constants.ReadPermission, repo) { - return mTypes.NotChanged, zerr.ErrUserDataNotAllowed - } - - userid := userAc.GetUsername() - - userData, err := dwr.GetUserData(ctx) - if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { - return res, err - } + var ( + manifestSignatures *proto_go.ManifestSignatures + found bool + ) - if !zcommon.Contains(userData.StarredRepos, repo) { - userData.StarredRepos = append(userData.StarredRepos, repo) - res = mTypes.Added - } else { - userData.StarredRepos = zcommon.RemoveFrom(userData.StarredRepos, repo) - res = mTypes.Removed + if manifestSignatures, found = protoRepoMeta.Signatures[signedManifestDigest.String()]; !found { + manifestSignatures = &proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} } - if res != mTypes.NotChanged { - repoMeta, err := dwr.GetRepoMeta(repo) //nolint:contextcheck - if err != nil { - return mTypes.NotChanged, err - } - - switch res { - case mTypes.Added: - repoMeta.Stars++ - case mTypes.Removed: - repoMeta.Stars-- - } - - repoAttributeValue, err := attributevalue.Marshal(repoMeta) - if err != nil { - return mTypes.NotChanged, err - } - - userAttributeValue, err := attributevalue.Marshal(userData) - if err != nil { - return mTypes.NotChanged, err - } - - _, err = dwr.Client.TransactWriteItems(ctx, &dynamodb.TransactWriteItemsInput{ - TransactItems: []types.TransactWriteItem{ - { - // Update User Profile - Update: &types.Update{ - ExpressionAttributeNames: map[string]string{ - "#UP": "UserData", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":UserData": userAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "Identity": &types.AttributeValueMemberS{ - Value: userid, - }, - }, - TableName: aws.String(dwr.UserDataTablename), - UpdateExpression: aws.String("SET #UP = :UserData"), - }, - }, - { - // Update Repo Meta with updated repo stars - Update: &types.Update{ - ExpressionAttributeNames: map[string]string{ - "#RM": "RepoMetadata", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoMetadata": repoAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ - Value: repo, - }, - }, - TableName: aws.String(dwr.RepoMetaTablename), - UpdateExpression: aws.String("SET #RM = :RepoMetadata"), - }, - }, - }, - }) - if err != nil { - return mTypes.NotChanged, err - } - } - - return res, nil -} - -func (dwr *DynamoDB) GetStarredRepos(ctx context.Context) ([]string, error) { - userMeta, err := dwr.GetUserData(ctx) - - if errors.Is(err, zerr.ErrUserDataNotFound) || errors.Is(err, zerr.ErrUserDataNotAllowed) { - return []string{}, nil - } - - return userMeta.StarredRepos, err -} - -func (dwr *DynamoDB) createUserDataTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.UserDataTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("Identity"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("Identity"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) - if err != nil { - if strings.Contains(err.Error(), "Table already exists") { - return nil - } - - return err - } - - return dwr.waitTableToBeCreated(dwr.UserDataTablename) -} - -func (dwr DynamoDB) createAPIKeyTable() error { - _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ - TableName: aws.String(dwr.APIKeyTablename), - AttributeDefinitions: []types.AttributeDefinition{ - { - AttributeName: aws.String("HashedKey"), - AttributeType: types.ScalarAttributeTypeS, - }, - }, - KeySchema: []types.KeySchemaElement{ - { - AttributeName: aws.String("HashedKey"), - KeyType: types.KeyTypeHash, - }, - }, - BillingMode: types.BillingModePayPerRequest, - }) - if err != nil { - if strings.Contains(err.Error(), "Table already exists") { - return nil - } - - return err - } - - return dwr.waitTableToBeCreated(dwr.APIKeyTablename) -} - -func (dwr DynamoDB) SetUserGroups(ctx context.Context, groups []string) error { - userData, err := dwr.GetUserData(ctx) - if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { - return err - } - - userData.Groups = append(userData.Groups, groups...) - - return dwr.SetUserData(ctx, userData) -} - -func (dwr DynamoDB) GetUserGroups(ctx context.Context) ([]string, error) { - userData, err := dwr.GetUserData(ctx) - - return userData.Groups, err -} - -func (dwr *DynamoDB) IsAPIKeyExpired(ctx context.Context, hashedKey string) (bool, error) { - userData, err := dwr.GetUserData(ctx) - if err != nil { - return false, err - } - - var isExpired bool - - apiKeyDetails := userData.APIKeys[hashedKey] - if apiKeyDetails.IsExpired { - isExpired = true - - return isExpired, nil - } - - // if expiresAt is not nil value - if !apiKeyDetails.ExpirationDate.Equal(time.Time{}) && time.Now().After(apiKeyDetails.ExpirationDate) { - isExpired = true - apiKeyDetails.IsExpired = true - } - - userData.APIKeys[hashedKey] = apiKeyDetails - - err = dwr.SetUserData(ctx, userData) - - return isExpired, err -} - -func (dwr DynamoDB) UpdateUserAPIKeyLastUsed(ctx context.Context, hashedKey string) error { - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return err - } - - if userAc.IsAnonymous() { - return zerr.ErrUserDataNotAllowed - } - - userData, err := dwr.GetUserData(ctx) - if err != nil { - return err - } - - apiKeyDetails := userData.APIKeys[hashedKey] - apiKeyDetails.LastUsed = time.Now() - - userData.APIKeys[hashedKey] = apiKeyDetails - - err = dwr.SetUserData(ctx, userData) - - return err -} - -func (dwr DynamoDB) GetUserAPIKeys(ctx context.Context) ([]mTypes.APIKeyDetails, error) { - apiKeys := make([]mTypes.APIKeyDetails, 0) - - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return nil, err - } - - if userAc.IsAnonymous() { - return nil, zerr.ErrUserDataNotAllowed - } - - userid := userAc.GetUsername() - - userData, err := dwr.GetUserData(ctx) - if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { - return nil, fmt.Errorf("metaDB: error while getting userData for identity %s %w", userid, err) - } - - for hashedKey, apiKeyDetails := range userData.APIKeys { - // if expiresAt is not nil value - if !apiKeyDetails.ExpirationDate.Equal(time.Time{}) && time.Now().After(apiKeyDetails.ExpirationDate) { - apiKeyDetails.IsExpired = true - } - - userData.APIKeys[hashedKey] = apiKeyDetails - - err = dwr.SetUserData(ctx, userData) - if err != nil { - return nil, err - } - - apiKeys = append(apiKeys, apiKeyDetails) - } - - return apiKeys, nil -} - -func (dwr DynamoDB) AddUserAPIKey(ctx context.Context, hashedKey string, apiKeyDetails *mTypes.APIKeyDetails) error { - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return err - } - - if userAc.IsAnonymous() { - return zerr.ErrUserDataNotAllowed - } - - userid := userAc.GetUsername() - - userData, err := dwr.GetUserData(ctx) - if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { - return fmt.Errorf("metaDB: error while getting userData for identity %s %w", userid, err) - } - - if userData.APIKeys == nil { - userData.APIKeys = make(map[string]mTypes.APIKeyDetails) - } - - userData.APIKeys[hashedKey] = *apiKeyDetails - - userAttributeValue, err := attributevalue.Marshal(userData) - if err != nil { - return err - } - - _, err = dwr.Client.TransactWriteItems(ctx, &dynamodb.TransactWriteItemsInput{ - TransactItems: []types.TransactWriteItem{ - { - // Update UserData - Update: &types.Update{ - ExpressionAttributeNames: map[string]string{ - "#UP": "UserData", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":UserData": userAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "Identity": &types.AttributeValueMemberS{ - Value: userid, - }, - }, - TableName: aws.String(dwr.UserDataTablename), - UpdateExpression: aws.String("SET #UP = :UserData"), - }, - }, - { - // Update APIKeyInfo - Update: &types.Update{ - ExpressionAttributeNames: map[string]string{ - "#EM": "Identity", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":Identity": &types.AttributeValueMemberS{Value: userid}, - }, - Key: map[string]types.AttributeValue{ - "HashedKey": &types.AttributeValueMemberS{ - Value: hashedKey, - }, - }, - TableName: aws.String(dwr.APIKeyTablename), - UpdateExpression: aws.String("SET #EM = :Identity"), - }, - }, - }, - }) - - return err -} - -func (dwr DynamoDB) DeleteUserAPIKey(ctx context.Context, keyID string) error { - userData, err := dwr.GetUserData(ctx) - if err != nil { - return fmt.Errorf("metaDB: error while getting userData %w", err) - } - - for hash, apiKeyDetails := range userData.APIKeys { - if apiKeyDetails.UUID == keyID { - delete(userData.APIKeys, hash) - - _, err = dwr.Client.DeleteItem(ctx, &dynamodb.DeleteItemInput{ - TableName: aws.String(dwr.APIKeyTablename), - Key: map[string]types.AttributeValue{ - "HashedKey": &types.AttributeValueMemberS{Value: hash}, - }, - }) - if err != nil { - return fmt.Errorf("metaDB: error while deleting userAPIKey entry for hash %s %w", hash, err) - } - - err := dwr.SetUserData(ctx, userData) - - return err - } - } - - return nil -} - -func (dwr DynamoDB) GetUserAPIKeyInfo(hashedKey string) (string, error) { - var userid string - - resp, err := dwr.Client.GetItem(context.Background(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.APIKeyTablename), - Key: map[string]types.AttributeValue{ - "HashedKey": &types.AttributeValueMemberS{Value: hashedKey}, - }, - }) - if err != nil { - return "", err - } - - if resp.Item == nil { - return "", zerr.ErrUserAPIKeyNotFound - } - - err = attributevalue.Unmarshal(resp.Item["Identity"], &userid) - if err != nil { - return "", err - } - - return userid, nil -} - -func (dwr DynamoDB) GetUserData(ctx context.Context) (mTypes.UserData, error) { - var userData mTypes.UserData - - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return userData, err - } - - if userAc.IsAnonymous() { - return userData, zerr.ErrUserDataNotAllowed - } - - userid := userAc.GetUsername() - - resp, err := dwr.Client.GetItem(ctx, &dynamodb.GetItemInput{ - TableName: aws.String(dwr.UserDataTablename), - Key: map[string]types.AttributeValue{ - "Identity": &types.AttributeValueMemberS{Value: userid}, - }, - }) - if err != nil { - return mTypes.UserData{}, err - } - - if resp.Item == nil { - return mTypes.UserData{}, zerr.ErrUserDataNotFound - } - - err = attributevalue.Unmarshal(resp.Item["UserData"], &userData) - if err != nil { - return mTypes.UserData{}, err - } - - return userData, nil -} - -func (dwr DynamoDB) SetUserData(ctx context.Context, userData mTypes.UserData) error { - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return err - } - - if userAc.IsAnonymous() { - return zerr.ErrUserDataNotAllowed - } - - userid := userAc.GetUsername() - - userAttributeValue, err := attributevalue.Marshal(userData) - if err != nil { - return err - } - - _, err = dwr.Client.UpdateItem(ctx, &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#UP": "UserData", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":UserData": userAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "Identity": &types.AttributeValueMemberS{ - Value: userid, - }, - }, - TableName: aws.String(dwr.UserDataTablename), - UpdateExpression: aws.String("SET #UP = :UserData"), - }) - - return err -} - -func (dwr DynamoDB) DeleteUserData(ctx context.Context) error { - userAc, err := reqCtx.UserAcFromContext(ctx) - if err != nil { - return err - } - - if userAc.IsAnonymous() { - return zerr.ErrUserDataNotAllowed - } - - userid := userAc.GetUsername() - - _, err = dwr.Client.DeleteItem(ctx, &dynamodb.DeleteItemInput{ - TableName: aws.String(dwr.UserDataTablename), - Key: map[string]types.AttributeValue{ - "Identity": &types.AttributeValueMemberS{Value: userid}, - }, - }) - - return err -} - -func (dwr *DynamoDB) SetProtoImageData(digest godigest.Digest, protoImageData *proto_go.ImageData) error { - bytes, err := proto.Marshal(protoImageData) - if err != nil { - return err - } - - mdAttributeValue, err := attributevalue.Marshal(bytes) - if err != nil { - return err - } - - resp, err := dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#ID": "ImageData", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":ImageData": mdAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "Digest": &types.AttributeValueMemberS{ - Value: protoImageData.Digest, - }, - }, - TableName: aws.String(dwr.ProtoImageDataTablename), - UpdateExpression: aws.String("SET #ID = :ImageData"), - }) - _ = resp - - return err -} - -func (dwr *DynamoDB) ProtoSetImageData(digest godigest.Digest, imageData mTypes.ImageData2) error { - return dwr.SetProtoImageData(imageData.Digest, mConvert.GetProtoImageData(imageData)) -} - -func (dwr *DynamoDB) GetProtoImageData(digest godigest.Digest) (*proto_go.ImageData, error) { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.ProtoImageDataTablename), - Key: map[string]types.AttributeValue{ - "Digest": &types.AttributeValueMemberS{Value: digest.String()}, - }, - }) - if err != nil { - return nil, err - } - - blob := []byte{} - - err = attributevalue.Unmarshal(resp.Item["ImageData"], &blob) - if err != nil { - return nil, err - } - - imageData := &proto_go.ImageData{} - - err = proto.Unmarshal(blob, imageData) - if err != nil { - return nil, err - } - - return imageData, nil -} - -func (dwr *DynamoDB) setProtoRepoMeta(repo string, repoMeta *proto_go.ProtoRepoMeta) error { - repoMeta.Name = repo - - repoMetaBlob, err := proto.Marshal(repoMeta) - if err != nil { - return err - } - - repoAttributeValue, err := attributevalue.Marshal(repoMetaBlob) - if err != nil { - return err - } - - _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#RM": "RepoMetadata", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoMetadata": repoAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ - Value: repo, - }, - }, - TableName: aws.String(dwr.ProtoRepoMetaTablename), - UpdateExpression: aws.String("SET #RM = :RepoMetadata"), - }) - - return err -} - -func (dwr *DynamoDB) getProtoRepoMeta(repo string) (*proto_go.ProtoRepoMeta, error) { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.ProtoRepoMetaTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return nil, err - } - - if resp.Item == nil { - return nil, zerr.ErrRepoMetaNotFound - } - - blob := []byte{} - - err = attributevalue.Unmarshal(resp.Item["RepoMetadata"], &blob) - if err != nil { - return nil, err - } - - repoMeta := &proto_go.ProtoRepoMeta{} - - err = proto.Unmarshal(blob, repoMeta) - if err != nil { - return nil, err - } - - return repoMeta, nil -} - -func (dwr *DynamoDB) ProtoSetRepoReference(repo string, reference string, imageData mTypes.ImageData2) error { - if err := common.ValidateRepoReferenceInput(repo, reference, imageData.Digest); err != nil { - return err - } - - // 1. Add image data to db if needed - protoImageData := mConvert.GetProtoImageData(imageData) - - err := dwr.SetProtoImageData(imageData.Digest, protoImageData) - if err != nil { - return err - } - - repoMeta, err := dwr.getProtoRepoMeta(repo) - if err != nil { - if !errors.Is(err, zerr.ErrRepoMetaNotFound) { - return err - } - - repoMeta = &proto_go.ProtoRepoMeta{ - Name: repo, - Tags: map[string]*proto_go.TagDescriptor{"": {}}, - Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, - Signatures: map[string]*proto_go.ManifestSignatures{"": {Map: map[string]*proto_go.SignaturesInfo{"": {}}}}, - Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, - } - } - - // 2. Referrers - if protoImageData.Subject != nil { - refList := repoMeta.Referrers[protoImageData.Subject.Digest].List - newRefList := append(refList, &proto_go.ReferrerInfo{ - Digest: protoImageData.Digest, - MediaType: protoImageData.MediaType, - ArtifactType: protoImageData.ArtifacType, - Size: protoImageData.Size, - Annotations: protoImageData.Annotations, - }) - repoMeta.Referrers[protoImageData.Subject.Digest].List = newRefList - } - - // 3. Update tag - if !common.ReferenceIsDigest(reference) { - repoMeta.Tags[reference] = &proto_go.TagDescriptor{ - Digest: imageData.Digest.String(), - MediaType: imageData.MediaType, - } - } - - if _, ok := repoMeta.Statistics[imageData.Digest.String()]; !ok { - repoMeta.Statistics[imageData.Digest.String()] = &proto_go.DescriptorStatistics{DownloadCount: 0} - } - - if _, ok := repoMeta.Signatures[imageData.Digest.String()]; !ok { - repoMeta.Signatures[imageData.Digest.String()] = &proto_go.ManifestSignatures{ - Map: map[string]*proto_go.SignaturesInfo{"": {}}, - } - } - - if _, ok := repoMeta.Referrers[imageData.Digest.String()]; !ok { - repoMeta.Referrers[imageData.Digest.String()] = &proto_go.ReferrersInfo{ - List: []*proto_go.ReferrerInfo{}, - } - } - - // 4. Blobs - repoBlobs, err := dwr.getRepoBlobsInfo(repo) - if err != nil { - return err - } - - repoMeta, repoBlobs, err = common.GetUpdatedRepoMeta(repoMeta, repoBlobs, reference, imageData) - if err != nil { - return err - } - - err = dwr.setRepoBlobsInfo(repo, repoBlobs) - if err != nil { - return err - } - - return dwr.setProtoRepoMeta(repo, repoMeta) -} - -func (dwr *DynamoDB) getRepoBlobsInfo(repo string) (*proto_go.RepoBlobs, error) { - resp, err := dwr.Client.GetItem(context.Background(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.ProtoRepoBlobsTablename), - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, - }, - }) - if err != nil { - return nil, err - } - - if resp.Item == nil { - return &proto_go.RepoBlobs{Name: repo, Blobs: map[string]*proto_go.BlobInfo{"": {}}}, nil - } - - repoBlobsBytes := []byte{} - - err = attributevalue.Unmarshal(resp.Item["RepoBlobsInfo"], &repoBlobsBytes) - if err != nil { - return nil, err - } - - repoBlobs := &proto_go.RepoBlobs{} - if repoBlobsBytes == nil { - repoBlobs.Blobs = map[string]*proto_go.BlobInfo{} - } else { - err := proto.Unmarshal(repoBlobsBytes, repoBlobs) - if err != nil { - return nil, err + signatureSlice := &proto_go.SignaturesInfo{List: []*proto_go.SignatureInfo{}} + if sigSlice, found := manifestSignatures.Map[sygMeta.SignatureType]; found { + signatureSlice = sigSlice + } + + if !common.ProtoSignatureAlreadyExists(signatureSlice.List, sygMeta) { + switch sygMeta.SignatureType { + case zcommon.NotationSignature: + signatureSlice.List = append(signatureSlice.List, &proto_go.SignatureInfo{ + SignatureManifestDigest: sygMeta.SignatureDigest, + LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), + }) + case zcommon.CosignSignature: + signatureSlice.List = []*proto_go.SignatureInfo{{ + SignatureManifestDigest: sygMeta.SignatureDigest, + LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), + }} } } - return repoBlobs, nil + manifestSignatures.Map[sygMeta.SignatureType] = signatureSlice + protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures + + return dwr.setProtoRepoMeta(protoRepoMeta.Name, protoRepoMeta) } -func (dwr *DynamoDB) setRepoBlobsInfo(repo string, repoBlobs *proto_go.RepoBlobs) error { - bytes, err := proto.Marshal(repoBlobs) +func (dwr *DynamoDB) DeleteSignature(repo string, signedManifestDigest godigest.Digest, + sigMeta mTypes.SignatureMetadata, +) error { + protoRepoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) if err != nil { return err } - mdAttributeValue, err := attributevalue.Marshal(bytes) - if err != nil { - return err + sigType := sigMeta.SignatureType + + var ( + manifestSignatures *proto_go.ManifestSignatures + found bool + ) + + if manifestSignatures, found = protoRepoMeta.Signatures[signedManifestDigest.String()]; !found { + return zerr.ErrManifestMetaNotFound } - _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#RBI": "RepoBlobsInfo", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoBlobsInfo": mdAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ - Value: repo, - }, - }, - TableName: aws.String(dwr.ProtoRepoBlobsTablename), - UpdateExpression: aws.String("SET #RBI = :RepoBlobsInfo"), - }) + signatureSlice := manifestSignatures.Map[sigType] - return err -} + newSignatureSlice := make([]*proto_go.SignatureInfo, 0, len(signatureSlice.List)-1) -func (dwr *DynamoDB) ProtoSearchRepos(ctx context.Context, searchText string) ([]mTypes.FullRepoMetadata, error) { - repos := []mTypes.FullRepoMetadata{} + for _, sigDigest := range signatureSlice.List { + if sigDigest.SignatureManifestDigest != sigMeta.SignatureDigest { + newSignatureSlice = append(newSignatureSlice, sigDigest) + } + } - userBookmarks := getUserBookmarks(ctx, dwr) - userStars := getUserStars(ctx, dwr) + manifestSignatures.Map[sigMeta.SignatureType] = &proto_go.SignaturesInfo{List: newSignatureSlice} - repoMetaAttributeIterator := NewBaseDynamoAttributesIterator( - dwr.Client, dwr.ProtoRepoMetaTablename, "RepoMetadata", 0, dwr.Log, - ) + protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) + return dwr.setProtoRepoMeta(protoRepoMeta.Name, protoRepoMeta) +} - for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { - if err != nil { - return []mTypes.FullRepoMetadata{}, err - } +func (dwr *DynamoDB) FilterImageData(ctx context.Context, digests []string, +) (map[string]mTypes.ImageData, error) { + imageDataAttributes, err := dwr.fetchImageDataAttributesByDigest(ctx, digests) + if err != nil { + return nil, err + } - repoMetaBlob := []byte{} + results := map[string]mTypes.ImageData{} - err := attributevalue.Unmarshal(repoMetaAttribute, &repoMetaBlob) + for _, attributes := range imageDataAttributes { + protoImageData, err := getProtoImageDataFromAttribute(attributes["ImageData"]) if err != nil { - return []mTypes.FullRepoMetadata{}, err + return nil, err } - protoRepoMeta := &proto_go.ProtoRepoMeta{} + if protoImageData.MediaType == ispec.MediaTypeImageIndex { + manifestDataList := make([]*proto_go.ManifestData, 0, len(protoImageData.Index.Index.Manifests)) - err = proto.Unmarshal(repoMetaBlob, protoRepoMeta) - if err != nil { - return []mTypes.FullRepoMetadata{}, err - } + indexDigests := make([]string, 0, len(protoImageData.Index.Index.Manifests)) + for i := range protoImageData.Index.Index.Manifests { + indexDigests = append(indexDigests, protoImageData.Index.Index.Manifests[i].Digest) + } - if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { - continue - } + manifestsAttributes, err := dwr.fetchImageDataAttributesByDigest(ctx, indexDigests) + if err != nil { + return nil, err + } - rank := common.RankRepoName(searchText, protoRepoMeta.Name) - if rank == -1 { - continue - } + for _, manifestAttribute := range manifestsAttributes { + imageManifestData, err := getProtoImageDataFromAttribute(manifestAttribute["ImageData"]) + if err != nil { + return nil, err + } - protoRepoMeta.Rank = int32(rank) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) + manifestDataList = append(manifestDataList, imageManifestData.Manifests[0]) + } - repos = append(repos, mConvert.GetFullRepoMeta(protoRepoMeta)) + protoImageData.Manifests = manifestDataList + } + results[mConvert.GetImageDigestStr(protoImageData)] = mConvert.GetImageData(protoImageData) } - return repos, nil + return results, nil } -func (dwr *DynamoDB) ProtoSearchTags(ctx context.Context, searchText string) ([]mTypes.FullImageData, error) { - images := []mTypes.FullImageData{} - userBookmarks := getUserBookmarks(ctx, dwr) - userStars := getUserStars(ctx, dwr) - - searchedRepo, searchedTag, err := common.GetRepoTag(searchText) +func (dwr *DynamoDB) RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest, +) error { + protoRepoMeta, err := dwr.getProtoRepoMeta(context.Background(), repo) if err != nil { - return []mTypes.FullImageData{}, - fmt.Errorf("metadb: error while parsing search text, invalid format %w", err) - } + if errors.Is(err, zerr.ErrRepoMetaNotFound) { + return nil + } - if ok, err := reqCtx.RepoIsUserAvailable(ctx, searchedRepo); !ok || err != nil { - return []mTypes.FullImageData{}, err + return err } - protoRepoMeta, err := dwr.getProtoRepoMeta(searchedRepo) + protoImageData, err := dwr.GetProtoImageData(context.TODO(), manifestDigest) if err != nil { - return nil, err - } + if errors.Is(err, zerr.ErrImageDataNotFound) { + return nil + } - delete(protoRepoMeta.Tags, "") + return err + } - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, searchedRepo) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, searchedRepo) + // Remove Referrers + if subject := mConvert.GetImageSubject(protoImageData); subject != nil { + referredDigest := subject.Digest.String() + refInfo := &proto_go.ReferrersInfo{} - for tag, descriptor := range protoRepoMeta.Tags { - if !strings.HasPrefix(tag, searchedTag) { - continue + if protoRepoMeta.Referrers[referredDigest] != nil { + refInfo = protoRepoMeta.Referrers[referredDigest] } - protoImageData := &proto_go.ImageData{} + referrers := refInfo.List - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest + for i := range referrers { + if referrers[i].Digest == manifestDigest.String() { + referrers[i].Count -= 1 - imageManifestData, err := dwr.GetProtoImageData(godigest.Digest(manifestDigest)) - if err != nil { - return []mTypes.FullImageData{}, - fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", manifestDigest, err) + if referrers[i].Count == 0 || common.ReferenceIsDigest(reference) { + referrers = append(referrers[:i], referrers[i+1:]...) + } + + break } + } - imageManifestData.Repo = ref(searchedRepo) - imageManifestData.Tag = &tag + refInfo.List = referrers - protoImageData = imageManifestData - case ispec.MediaTypeImageIndex: - indexDigest := godigest.Digest(descriptor.Digest) + protoRepoMeta.Referrers[referredDigest] = refInfo + } - imageIndexData, err := dwr.GetProtoImageData(indexDigest) - if err != nil { - return []mTypes.FullImageData{}, - fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", indexDigest, err) + if !common.ReferenceIsDigest(reference) { + delete(protoRepoMeta.Tags, reference) + } else { + // find all tags pointing to this digest + tags := []string{} + for tag, desc := range protoRepoMeta.Tags { + if desc.Digest == reference { + tags = append(tags, tag) } + } - for i, manifest := range imageIndexData.Manifests { - manifestDigest := godigest.Digest(manifest.Digest) + // remove all tags + for _, tag := range tags { + delete(protoRepoMeta.Tags, tag) + } + } - imageManifestData, err := dwr.GetProtoImageData(manifestDigest) - if err != nil { - return []mTypes.FullImageData{}, err - } + /* try to find at least one tag pointing to manifestDigest + if not found then we can also remove everything related to this digest */ + var foundTag bool - imageIndexData.Manifests[i] = imageManifestData.Manifests[0] - } + for _, desc := range protoRepoMeta.Tags { + if desc.Digest == manifestDigest.String() { + foundTag = true + } + } - imageIndexData.Repo = ref(searchedRepo) - imageIndexData.Tag = &tag + if !foundTag { + delete(protoRepoMeta.Statistics, manifestDigest.String()) + delete(protoRepoMeta.Signatures, manifestDigest.String()) + delete(protoRepoMeta.Referrers, manifestDigest.String()) + } - protoImageData = imageIndexData - default: - dwr.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + repoBlobsInfo, err := dwr.getRepoBlobsInfo(repo) + if err != nil { + return err + } - continue - } + protoRepoMeta, repoBlobsInfo, err = common.RemoveImageFromRepoMeta(protoRepoMeta, repoBlobsInfo, reference) + if err != nil { + return err + } - images = append(images, mConvert.GetFullImageData(tag, protoRepoMeta, protoImageData)) + err = dwr.setRepoBlobsInfo(repo, repoBlobsInfo) + if err != nil { + return err } + err = dwr.setProtoRepoMeta(repo, protoRepoMeta) - return images, err + return err } -func ref[T any](input T) *T { - ref := input +func getUserStars(ctx context.Context, dwr *DynamoDB) []string { + starredRepos, err := dwr.GetStarredRepos(ctx) + if err != nil { + return []string{} + } - return &ref + return starredRepos } -func (dwr *DynamoDB) ProtoFilterTags(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterProtoFunc, -) ([]mTypes.FullImageData, error) { - images := []mTypes.FullImageData{} - userBookmarks := getUserBookmarks(ctx, dwr) - userStars := getUserStars(ctx, dwr) - - repoMetaAttributeIterator := NewBaseDynamoAttributesIterator( - dwr.Client, dwr.ProtoRepoMetaTablename, "RepoMetadata", 0, dwr.Log, - ) +func getUserBookmarks(ctx context.Context, dwr *DynamoDB) []string { + bookmarkedRepos, err := dwr.GetBookmarkedRepos(ctx) + if err != nil { + return []string{} + } - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) + return bookmarkedRepos +} - for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { - if err != nil { - return []mTypes.FullImageData{}, - err - } +func (dwr *DynamoDB) ToggleBookmarkRepo(ctx context.Context, repo string) ( + mTypes.ToggleState, error, +) { + res := mTypes.NotChanged - protoRepoMeta, err := getProtoRepoMetaFromAttribute(repoMetaAttribute) - if err != nil { - return nil, err - } + userAc, err := reqCtx.UserAcFromContext(ctx) + if err != nil { + return mTypes.NotChanged, err + } - if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { - continue - } + if userAc.IsAnonymous() || !userAc.Can(constants.ReadPermission, repo) { + return mTypes.NotChanged, zerr.ErrUserDataNotAllowed + } - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) - repoMeta := mConvert.GetRepoMeta(protoRepoMeta) + userData, err := dwr.GetUserData(ctx) + if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { + return res, err + } - for tag, descriptor := range repoMeta.Tags { - if !filterRepoTag(repoMeta.Name, tag) { - continue - } + if !zcommon.Contains(userData.BookmarkedRepos, repo) { + userData.BookmarkedRepos = append(userData.BookmarkedRepos, repo) + res = mTypes.Added + } else { + userData.BookmarkedRepos = zcommon.RemoveFrom(userData.BookmarkedRepos, repo) + res = mTypes.Removed + } - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestDigest := descriptor.Digest + if res != mTypes.NotChanged { + err = dwr.SetUserData(ctx, userData) + } - imageManifestData, err := dwr.GetProtoImageData(godigest.Digest(manifestDigest)) - if err != nil { - return []mTypes.FullImageData{}, - fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", manifestDigest, err) - } + if err != nil { + res = mTypes.NotChanged - imageManifestData.Repo = ref(repoMeta.Name) - imageManifestData.Tag = &tag + return res, err + } - imageData := mConvert.GetImageData(imageManifestData) + return res, nil +} - if filterFunc(repoMeta, imageData) { - images = append(images, mConvert.GetFullImageData(tag, protoRepoMeta, imageManifestData)) - } - case ispec.MediaTypeImageIndex: - indexDigest := descriptor.Digest +func (dwr *DynamoDB) GetBookmarkedRepos(ctx context.Context) ([]string, error) { + userMeta, err := dwr.GetUserData(ctx) - imageIndexData, err := dwr.GetProtoImageData(godigest.Digest(indexDigest)) - if err != nil { - return []mTypes.FullImageData{}, - fmt.Errorf("metadb: error fetching manifest meta for manifest with digest %s %w", indexDigest, err) - } + if errors.Is(err, zerr.ErrUserDataNotFound) || errors.Is(err, zerr.ErrUserDataNotAllowed) { + return []string{}, nil + } - matchedManifests := []*proto_go.ManifestData{} + return userMeta.BookmarkedRepos, err +} - for _, manifest := range imageIndexData.Manifests { - manifestDigest := manifest.Digest +func (dwr *DynamoDB) ToggleStarRepo(ctx context.Context, repo string) ( + mTypes.ToggleState, error, +) { + res := mTypes.NotChanged - imageManifestData, err := dwr.GetProtoImageData(godigest.Digest(manifestDigest)) - if err != nil { - return []mTypes.FullImageData{}, - fmt.Errorf("metadb: error while getting manifest data for digest %s %w", manifestDigest, err) - } + userAc, err := reqCtx.UserAcFromContext(ctx) + if err != nil { + return mTypes.NotChanged, err + } - imageData := mConvert.GetImageData(imageManifestData) + if userAc.IsAnonymous() || !userAc.Can(constants.ReadPermission, repo) { + return mTypes.NotChanged, zerr.ErrUserDataNotAllowed + } - if filterFunc(repoMeta, imageData) { - matchedManifests = append(matchedManifests, imageManifestData.Manifests[0]) - } - } + userid := userAc.GetUsername() - if len(matchedManifests) > 0 { - imageIndexData.Manifests = matchedManifests + userData, err := dwr.GetUserData(ctx) + if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { + return res, err + } - images = append(images, mConvert.GetFullImageData(tag, protoRepoMeta, imageIndexData)) - } - default: - dwr.Log.Error().Str("mediaType", descriptor.MediaType).Msg("Unsupported media type") + if !zcommon.Contains(userData.StarredRepos, repo) { + userData.StarredRepos = append(userData.StarredRepos, repo) + res = mTypes.Added + } else { + userData.StarredRepos = zcommon.RemoveFrom(userData.StarredRepos, repo) + res = mTypes.Removed + } - continue - } + if res != mTypes.NotChanged { + repoMeta, err := dwr.getProtoRepoMeta(ctx, repo) //nolint:contextcheck + if err != nil { + return mTypes.NotChanged, err } - } - return images, err -} + switch res { + case mTypes.Added: + repoMeta.Stars++ + case mTypes.Removed: + repoMeta.Stars-- + } -func getProtoRepoMetaFromAttribute(repoMetaAttribute types.AttributeValue) (*proto_go.ProtoRepoMeta, error) { - blob := []byte{} + repoMetaBlob, err := proto.Marshal(repoMeta) + if err != nil { + return mTypes.NotChanged, err + } - err := attributevalue.Unmarshal(repoMetaAttribute, &blob) - if err != nil { - return nil, err - } + repoAttributeValue, err := attributevalue.Marshal(repoMetaBlob) + if err != nil { + return mTypes.NotChanged, err + } - protoRepoMeta := &proto_go.ProtoRepoMeta{} + userAttributeValue, err := attributevalue.Marshal(userData) + if err != nil { + return mTypes.NotChanged, err + } - err = proto.Unmarshal(blob, protoRepoMeta) - if err != nil { - return nil, err + _, err = dwr.Client.TransactWriteItems(ctx, &dynamodb.TransactWriteItemsInput{ + TransactItems: []types.TransactWriteItem{ + { + // Update User Profile + Update: &types.Update{ + ExpressionAttributeNames: map[string]string{ + "#UP": "UserData", + }, + ExpressionAttributeValues: map[string]types.AttributeValue{ + ":UserData": userAttributeValue, + }, + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{ + Value: userid, + }, + }, + TableName: aws.String(dwr.UserDataTablename), + UpdateExpression: aws.String("SET #UP = :UserData"), + }, + }, + { + // Update Repo Meta with updated repo stars + Update: &types.Update{ + ExpressionAttributeNames: map[string]string{ + "#RM": "RepoMetadata", + }, + ExpressionAttributeValues: map[string]types.AttributeValue{ + ":RepoMetadata": repoAttributeValue, + }, + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{ + Value: repo, + }, + }, + TableName: aws.String(dwr.RepoMetaTablename), + UpdateExpression: aws.String("SET #RM = :RepoMetadata"), + }, + }, + }, + }) + if err != nil { + return mTypes.NotChanged, err + } } - return protoRepoMeta, nil + return res, nil } -func getProtoImageDataFromAttribute(imageDataAttribute types.AttributeValue) (*proto_go.ImageData, error) { - blob := []byte{} +func (dwr *DynamoDB) GetStarredRepos(ctx context.Context) ([]string, error) { + userMeta, err := dwr.GetUserData(ctx) - err := attributevalue.Unmarshal(imageDataAttribute, &blob) - if err != nil { - return nil, err + if errors.Is(err, zerr.ErrUserDataNotFound) || errors.Is(err, zerr.ErrUserDataNotAllowed) { + return []string{}, nil } - protoImageData := &proto_go.ImageData{} + return userMeta.StarredRepos, err +} - err = proto.Unmarshal(blob, protoImageData) - if err != nil { - return nil, err +func (dwr DynamoDB) SetUserGroups(ctx context.Context, groups []string) error { + userData, err := dwr.GetUserData(ctx) + if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { + return err } - return protoImageData, nil -} + userData.Groups = append(userData.Groups, groups...) -func (dwr *DynamoDB) ProtoGetRepoMeta(repo string) (mTypes.RepoMetadata2, error) { - protoRepoMeta, err := dwr.getProtoRepoMeta(repo) - if err != nil { - return mTypes.RepoMetadata2{}, err - } + return dwr.SetUserData(ctx, userData) +} - delete(protoRepoMeta.Tags, "") +func (dwr DynamoDB) GetUserGroups(ctx context.Context) ([]string, error) { + userData, err := dwr.GetUserData(ctx) - return mConvert.GetRepoMeta(protoRepoMeta), nil + return userData.Groups, err } -func (dwr *DynamoDB) ProtoGetFullRepoMeta(ctx context.Context, repo string) (mTypes.FullRepoMetadata, error) { - protoRepoMeta, err := dwr.getProtoRepoMeta(repo) +func (dwr *DynamoDB) IsAPIKeyExpired(ctx context.Context, hashedKey string) (bool, error) { + userData, err := dwr.GetUserData(ctx) if err != nil { - return mTypes.FullRepoMetadata{}, err + return false, err } - delete(protoRepoMeta.Tags, "") + var isExpired bool - return mConvert.GetFullRepoMeta(protoRepoMeta), nil -} + apiKeyDetails := userData.APIKeys[hashedKey] + if apiKeyDetails.IsExpired { + isExpired = true -func (dwr *DynamoDB) ProtoGetImageData(digest godigest.Digest) (mTypes.ImageData2, error) { - imageData, err := dwr.GetProtoImageData(digest) - if err != nil { - return mTypes.ImageData2{}, err + return isExpired, nil } - return mConvert.GetImageData(imageData), nil -} + // if expiresAt is not nil value + if !apiKeyDetails.ExpirationDate.Equal(time.Time{}) && time.Now().After(apiKeyDetails.ExpirationDate) { + isExpired = true + apiKeyDetails.IsExpired = true + } -func (dwr *DynamoDB) ProtoGetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata2) bool, -) ([]mTypes.RepoMetadata2, error) { - var ( - foundRepos = []mTypes.RepoMetadata2{} - repoMetaAttributeIterator AttributesIterator - ) + userData.APIKeys[hashedKey] = apiKeyDetails - repoMetaAttributeIterator = NewBaseDynamoAttributesIterator( - dwr.Client, dwr.ProtoRepoMetaTablename, "RepoMetadata", 0, dwr.Log, - ) + err = dwr.SetUserData(ctx, userData) - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) + return isExpired, err +} - for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { - if err != nil { - return []mTypes.RepoMetadata2{}, err - } +func (dwr DynamoDB) UpdateUserAPIKeyLastUsed(ctx context.Context, hashedKey string) error { + userAc, err := reqCtx.UserAcFromContext(ctx) + if err != nil { + return err + } - repoMetaBlob := []byte{} + if userAc.IsAnonymous() { + return zerr.ErrUserDataNotAllowed + } - err := attributevalue.Unmarshal(repoMetaAttribute, &repoMetaBlob) - if err != nil { - return []mTypes.RepoMetadata2{}, err - } + userData, err := dwr.GetUserData(ctx) + if err != nil { + return err + } - protoRepoMeta := &proto_go.ProtoRepoMeta{} + apiKeyDetails := userData.APIKeys[hashedKey] + apiKeyDetails.LastUsed = time.Now() - err = proto.Unmarshal(repoMetaBlob, protoRepoMeta) - if err != nil { - return []mTypes.RepoMetadata2{}, err - } + userData.APIKeys[hashedKey] = apiKeyDetails - if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { - continue - } + err = dwr.SetUserData(ctx, userData) - repoMeta := mConvert.GetRepoMeta(protoRepoMeta) + return err +} - if filter(repoMeta) { - foundRepos = append(foundRepos, repoMeta) - } - } +func (dwr DynamoDB) GetUserAPIKeys(ctx context.Context) ([]mTypes.APIKeyDetails, error) { + apiKeys := make([]mTypes.APIKeyDetails, 0) - return foundRepos, err -} + userAc, err := reqCtx.UserAcFromContext(ctx) + if err != nil { + return nil, err + } -func (dwr *DynamoDB) ProtoFilterRepos(ctx context.Context, rankName mTypes.FilterRepoNameFunc, filterFunc mTypes.FilterFullRepoFunc, -) ([]mTypes.FullRepoMetadata, error) { - repos := []mTypes.FullRepoMetadata{} - userBookmarks := getUserBookmarks(ctx, dwr) - userStars := getUserStars(ctx, dwr) + if userAc.IsAnonymous() { + return nil, zerr.ErrUserDataNotAllowed + } - repoMetaAttributeIterator := NewBaseDynamoAttributesIterator( - dwr.Client, dwr.ProtoRepoMetaTablename, "RepoMetadata", 0, dwr.Log, - ) + userid := userAc.GetUsername() - repoMetaAttribute, err := repoMetaAttributeIterator.First(ctx) + userData, err := dwr.GetUserData(ctx) + if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { + return nil, fmt.Errorf("metaDB: error while getting userData for identity %s %w", userid, err) + } - for ; repoMetaAttribute != nil; repoMetaAttribute, err = repoMetaAttributeIterator.Next(ctx) { - if err != nil { - return []mTypes.FullRepoMetadata{}, - err + for hashedKey, apiKeyDetails := range userData.APIKeys { + // if expiresAt is not nil value + if !apiKeyDetails.ExpirationDate.Equal(time.Time{}) && time.Now().After(apiKeyDetails.ExpirationDate) { + apiKeyDetails.IsExpired = true } - protoRepoMeta, err := getProtoRepoMetaFromAttribute(repoMetaAttribute) + userData.APIKeys[hashedKey] = apiKeyDetails + + err = dwr.SetUserData(ctx, userData) if err != nil { return nil, err } - if ok, err := reqCtx.RepoIsUserAvailable(ctx, protoRepoMeta.Name); !ok || err != nil { - continue - } + apiKeys = append(apiKeys, apiKeyDetails) + } - rank := rankName(string(protoRepoMeta.Name)) - if rank < 0 { - continue - } + return apiKeys, nil +} - protoRepoMeta.Rank = int32(rank) - protoRepoMeta.IsBookmarked = zcommon.Contains(userBookmarks, protoRepoMeta.Name) - protoRepoMeta.IsStarred = zcommon.Contains(userStars, protoRepoMeta.Name) +func (dwr DynamoDB) AddUserAPIKey(ctx context.Context, hashedKey string, apiKeyDetails *mTypes.APIKeyDetails) error { + userAc, err := reqCtx.UserAcFromContext(ctx) + if err != nil { + return err + } - fullRepoMeta := mConvert.GetFullRepoMeta(protoRepoMeta) + if userAc.IsAnonymous() { + return zerr.ErrUserDataNotAllowed + } - if filterFunc(fullRepoMeta) { - repos = append(repos, fullRepoMeta) - } + userid := userAc.GetUsername() + + userData, err := dwr.GetUserData(ctx) + if err != nil && !errors.Is(err, zerr.ErrUserDataNotFound) { + return fmt.Errorf("metaDB: error while getting userData for identity %s %w", userid, err) } - return repos, err -} + if userData.APIKeys == nil { + userData.APIKeys = make(map[string]mTypes.APIKeyDetails) + } + + userData.APIKeys[hashedKey] = *apiKeyDetails -func (dwr *DynamoDB) ProtoIncrementRepoStars(repo string) error { - repoMeta, err := dwr.getProtoRepoMeta(repo) + userAttributeValue, err := attributevalue.Marshal(userData) if err != nil { return err } - repoMeta.Stars++ + _, err = dwr.Client.TransactWriteItems(ctx, &dynamodb.TransactWriteItemsInput{ + TransactItems: []types.TransactWriteItem{ + { + // Update UserData + Update: &types.Update{ + ExpressionAttributeNames: map[string]string{ + "#UP": "UserData", + }, + ExpressionAttributeValues: map[string]types.AttributeValue{ + ":UserData": userAttributeValue, + }, + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{ + Value: userid, + }, + }, + TableName: aws.String(dwr.UserDataTablename), + UpdateExpression: aws.String("SET #UP = :UserData"), + }, + }, + { + // Update APIKeyInfo + Update: &types.Update{ + ExpressionAttributeNames: map[string]string{ + "#EM": "Identity", + }, + ExpressionAttributeValues: map[string]types.AttributeValue{ + ":Identity": &types.AttributeValueMemberS{Value: userid}, + }, + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{ + Value: hashedKey, + }, + }, + TableName: aws.String(dwr.APIKeyTablename), + UpdateExpression: aws.String("SET #EM = :Identity"), + }, + }, + }, + }) - return dwr.setProtoRepoMeta(repo, repoMeta) + return err } -func (dwr *DynamoDB) ProtoDecrementRepoStars(repo string) error { - repoMeta, err := dwr.getProtoRepoMeta(repo) +func (dwr DynamoDB) DeleteUserAPIKey(ctx context.Context, keyID string) error { + userData, err := dwr.GetUserData(ctx) if err != nil { - return err + return fmt.Errorf("metaDB: error while getting userData %w", err) } - repoMeta.Stars-- + for hash, apiKeyDetails := range userData.APIKeys { + if apiKeyDetails.UUID == keyID { + delete(userData.APIKeys, hash) - return dwr.setProtoRepoMeta(repo, repoMeta) -} + _, err = dwr.Client.DeleteItem(ctx, &dynamodb.DeleteItemInput{ + TableName: aws.String(dwr.APIKeyTablename), + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{Value: hash}, + }, + }) + if err != nil { + return fmt.Errorf("metaDB: error while deleting userAPIKey entry for hash %s %w", hash, err) + } -func (dwr *DynamoDB) ProtoGetRepoStars(repo string) (int, error) { - repoMeta, err := dwr.getProtoRepoMeta(repo) - if err != nil { - return 0, err + err := dwr.SetUserData(ctx, userData) + + return err + } } - return int(repoMeta.Stars), nil + return nil } -func (dwr *DynamoDB) ProtoDeleteRepoTag(repo string, tag string) error { - resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ - TableName: aws.String(dwr.ProtoRepoMetaTablename), +func (dwr DynamoDB) GetUserAPIKeyInfo(hashedKey string) (string, error) { + var userid string + + resp, err := dwr.Client.GetItem(context.Background(), &dynamodb.GetItemInput{ + TableName: aws.String(dwr.APIKeyTablename), Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, + "Key": &types.AttributeValueMemberS{Value: hashedKey}, }, }) if err != nil { - return err + return "", err } if resp.Item == nil { - return nil + return "", zerr.ErrUserAPIKeyNotFound } - repoMetaBlob := []byte{} - - err = attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMetaBlob) + err = attributevalue.Unmarshal(resp.Item["Identity"], &userid) if err != nil { - return err + return "", err } - repoMeta := &proto_go.ProtoRepoMeta{} - - err = proto.Unmarshal(repoMetaBlob, repoMeta) - if err != nil { - return err - } + return userid, nil +} - delete(repoMeta.Tags, tag) +func (dwr DynamoDB) GetUserData(ctx context.Context) (mTypes.UserData, error) { + var userData mTypes.UserData - repoMetaBlob, err = proto.Marshal(repoMeta) + userAc, err := reqCtx.UserAcFromContext(ctx) if err != nil { - return err + return userData, err } - repoAttributeValue, err := attributevalue.Marshal(repoMetaBlob) - if err != nil { - return err + if userAc.IsAnonymous() { + return userData, zerr.ErrUserDataNotAllowed } - _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#RM": "RepoMetadata", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoMetadata": repoAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ - Value: repo, - }, - }, - TableName: aws.String(dwr.ProtoRepoMetaTablename), - UpdateExpression: aws.String("SET #RM = :RepoMetadata"), - }) - - return err -} + userid := userAc.GetUsername() -func (dwr *DynamoDB) ProtoGetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMetadata2, error) { resp, err := dwr.Client.GetItem(ctx, &dynamodb.GetItemInput{ - TableName: aws.String(dwr.ProtoRepoMetaTablename), + TableName: aws.String(dwr.UserDataTablename), Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{Value: repo}, + "Key": &types.AttributeValueMemberS{Value: userid}, }, }) if err != nil { - return mTypes.RepoMetadata2{}, err + return mTypes.UserData{}, err } if resp.Item == nil { - return mTypes.RepoMetadata2{}, zerr.ErrRepoMetaNotFound - } - - var repoMeta mTypes.RepoMetadata2 - - err = attributevalue.Unmarshal(resp.Item["RepoMetadata"], &repoMeta) - if err != nil { - return mTypes.RepoMetadata2{}, err + return mTypes.UserData{}, zerr.ErrUserDataNotFound } - userData, err := dwr.GetUserData(ctx) + err = attributevalue.Unmarshal(resp.Item["UserData"], &userData) if err != nil { - return mTypes.RepoMetadata2{}, err + return mTypes.UserData{}, err } - repoMeta.IsBookmarked = zcommon.Contains(userData.BookmarkedRepos, repo) - repoMeta.IsStarred = zcommon.Contains(userData.StarredRepos, repo) - - return repoMeta, nil -} - -func (dwr *DynamoDB) ProtoSetRepoMeta(repo string, repoMeta mTypes.RepoMetadata2) error { - protoRepoMeta := mConvert.GetProtoRepoMeta(repoMeta) - - return dwr.setProtoRepoMeta(repo, protoRepoMeta) + return userData, nil } -func (dwr *DynamoDB) ProtoGetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string) ([]mTypes.ReferrerInfo, error) { - repoMeta, err := dwr.GetRepoMeta(repo) +func (dwr DynamoDB) SetUserData(ctx context.Context, userData mTypes.UserData) error { + userAc, err := reqCtx.UserAcFromContext(ctx) if err != nil { - return []mTypes.ReferrerInfo{}, err + return err } - referrersInfo := repoMeta.Referrers[referredDigest.String()] - - filteredResults := make([]mTypes.ReferrerInfo, 0, len(referrersInfo)) - - for _, referrerInfo := range referrersInfo { - if !common.MatchesArtifactTypes(referrerInfo.ArtifactType, artifactTypes) { - continue - } - - filteredResults = append(filteredResults, referrerInfo) + if userAc.IsAnonymous() { + return zerr.ErrUserDataNotAllowed } - return filteredResults, nil -} + userid := userAc.GetUsername() -func (dwr *DynamoDB) ProtoIncrementImageDownloads(repo string, reference string) error { - repoMeta, err := dwr.getProtoRepoMeta(repo) + userAttributeValue, err := attributevalue.Marshal(userData) if err != nil { return err } - descriptorDigest := reference + _, err = dwr.Client.UpdateItem(ctx, &dynamodb.UpdateItemInput{ + ExpressionAttributeNames: map[string]string{ + "#UP": "UserData", + }, + ExpressionAttributeValues: map[string]types.AttributeValue{ + ":UserData": userAttributeValue, + }, + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{ + Value: userid, + }, + }, + TableName: aws.String(dwr.UserDataTablename), + UpdateExpression: aws.String("SET #UP = :UserData"), + }) - if !common.ReferenceIsDigest(reference) { - // search digest for tag - descriptor, found := repoMeta.Tags[reference] + return err +} - if !found { - return zerr.ErrManifestMetaNotFound - } +func (dwr DynamoDB) DeleteUserData(ctx context.Context) error { + userAc, err := reqCtx.UserAcFromContext(ctx) + if err != nil { + return err + } - descriptorDigest = descriptor.Digest + if userAc.IsAnonymous() { + return zerr.ErrUserDataNotAllowed } - manifestStatistics := repoMeta.Statistics[descriptorDigest] - manifestStatistics.DownloadCount++ - repoMeta.Statistics[descriptorDigest] = manifestStatistics + userid := userAc.GetUsername() - return dwr.setProtoRepoMeta(repo, repoMeta) -} + _, err = dwr.Client.DeleteItem(ctx, &dynamodb.DeleteItemInput{ + TableName: aws.String(dwr.UserDataTablename), + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{Value: userid}, + }, + }) -func (dwr *DynamoDB) ProtoUpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { - imgTrustStore := dwr.ImageTrustStore() + return err +} - if imgTrustStore == nil { - return nil +func (dwr *DynamoDB) fetchImageDataAttributesByDigest(ctx context.Context, digests []string, +) ([]map[string]types.AttributeValue, error) { + resp, err := dwr.Client.BatchGetItem(ctx, &dynamodb.BatchGetItemInput{ + RequestItems: map[string]types.KeysAndAttributes{ + dwr.ImageDataTablename: { + Keys: getBatchImageKeys(digests), + }, + }, + }) + if err != nil { + return nil, err } - protoImageData, err := dwr.GetProtoImageData(manifestDigest) - if err != nil { - return err + if len(resp.Responses[dwr.ImageDataTablename]) != len(digests) { + return nil, zerr.ErrImageDataNotFound } - // update signatures with details about validity and author - protoRepoMeta, err := dwr.getProtoRepoMeta(repo) - if err != nil { - return err + return resp.Responses[dwr.ImageDataTablename], nil +} + +func getBatchImageKeys(digests []string) []map[string]types.AttributeValue { + result := []map[string]types.AttributeValue{} + + for _, digest := range digests { + result = append(result, map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{ + Value: digest, + }, + }) } - manifestSignatures := proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} - for sigType, sigs := range protoRepoMeta.Signatures[manifestDigest.String()].Map { - signaturesInfo := []*proto_go.SignatureInfo{} - - for _, sigInfo := range sigs.List { - layersInfo := []*proto_go.LayersInfo{} - - for _, layerInfo := range sigInfo.LayersInfo { - author, date, isTrusted, _ := imgTrustStore.ProtoVerifySignature(sigType, layerInfo.LayerContent, - layerInfo.SignatureKey, manifestDigest, mConvert.GetImageData(protoImageData), repo) + return result +} - if isTrusted { - layerInfo.Signer = author - } +func (dwr *DynamoDB) PatchDB() error { + DBVersion, err := dwr.getDBVersion() + if err != nil { + return fmt.Errorf("patching dynamo failed, error retrieving database version %w", err) + } - if !date.IsZero() { - layerInfo.Signer = author - layerInfo.Date = timestamppb.New(date) - } + if version.GetVersionIndex(DBVersion) == -1 { + return fmt.Errorf("DB has broken format, no version found %w", err) + } - layersInfo = append(layersInfo, layerInfo) - } + for patchIndex, patch := range dwr.Patches { + if patchIndex < version.GetVersionIndex(DBVersion) { + continue + } - signaturesInfo = append(signaturesInfo, &proto_go.SignatureInfo{ - SignatureManifestDigest: sigInfo.SignatureManifestDigest, - LayersInfo: layersInfo, - }) + tableNames := map[string]string{ + "RepoMetaTablename": dwr.RepoMetaTablename, + "VersionTablename": dwr.VersionTablename, } - manifestSignatures.Map[sigType] = &proto_go.SignaturesInfo{List: signaturesInfo} + err := patch(dwr.Client, tableNames) + if err != nil { + return err + } } - protoRepoMeta.Signatures[manifestDigest.String()] = &manifestSignatures - - return dwr.setProtoRepoMeta(protoRepoMeta.Name, protoRepoMeta) + return nil } -func (dwr *DynamoDB) ProtoAddManifestSignature(repo string, signedManifestDigest godigest.Digest, - sygMeta mTypes.SignatureMetadata, -) error { - protoRepoMeta, err := dwr.getProtoRepoMeta(repo) +func (dwr *DynamoDB) ResetDB() error { + err := dwr.ResetTable(dwr.APIKeyTablename) if err != nil { - if errors.Is(err, zerr.ErrRepoMetaNotFound) { - protoRepoMeta = &proto_go.ProtoRepoMeta{ - Name: repo, - Tags: map[string]*proto_go.TagDescriptor{}, - Statistics: map[string]*proto_go.DescriptorStatistics{"": {}}, - Referrers: map[string]*proto_go.ReferrersInfo{"": {}}, - Signatures: map[string]*proto_go.ManifestSignatures{ - signedManifestDigest.String(): { - Map: map[string]*proto_go.SignaturesInfo{ - sygMeta.SignatureType: { - List: []*proto_go.SignatureInfo{ - { - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), - }, - }, - }, - }, - }, - }, - } - - return dwr.setProtoRepoMeta(repo, protoRepoMeta) - } - return err } - var ( - manifestSignatures *proto_go.ManifestSignatures - found bool - ) - - if manifestSignatures, found = protoRepoMeta.Signatures[signedManifestDigest.String()]; !found { - manifestSignatures = &proto_go.ManifestSignatures{Map: map[string]*proto_go.SignaturesInfo{"": {}}} + err = dwr.ResetTable(dwr.ImageDataTablename) + if err != nil { + return err } - signatureSlice := &proto_go.SignaturesInfo{List: []*proto_go.SignatureInfo{}} - if sigSlice, found := manifestSignatures.Map[sygMeta.SignatureType]; found { - signatureSlice = sigSlice + err = dwr.ResetTable(dwr.RepoBlobsTablename) + if err != nil { + return err } - if !common.ProtoSignatureAlreadyExists(signatureSlice.List, sygMeta) { - switch sygMeta.SignatureType { - case zcommon.NotationSignature: - signatureSlice.List = append(signatureSlice.List, &proto_go.SignatureInfo{ - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), - }) - case zcommon.CosignSignature: - signatureSlice.List = []*proto_go.SignatureInfo{{ - SignatureManifestDigest: sygMeta.SignatureDigest, - LayersInfo: mConvert.GetProtoLayersInfo(sygMeta.LayersInfo), - }} - } + err = dwr.ResetTable(dwr.RepoMetaTablename) + if err != nil { + return err } - manifestSignatures.Map[sygMeta.SignatureType] = signatureSlice - protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures + err = dwr.ResetTable(dwr.UserDataTablename) + if err != nil { + return err + } - return dwr.setProtoRepoMeta(protoRepoMeta.Name, protoRepoMeta) + return nil } -func (dwr *DynamoDB) ProtoDeleteSignature(repo string, signedManifestDigest godigest.Digest, sigMeta mTypes.SignatureMetadata, -) error { - protoRepoMeta, err := dwr.getProtoRepoMeta(repo) +func (dwr *DynamoDB) ResetTable(tableName string) error { + err := dwr.deleteTable(tableName) if err != nil { return err } - sigType := sigMeta.SignatureType + return dwr.createTable(tableName) +} - var ( - manifestSignatures *proto_go.ManifestSignatures - found bool - ) +func (dwr *DynamoDB) createTable(tableName string) error { + _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ + TableName: aws.String(tableName), + AttributeDefinitions: []types.AttributeDefinition{ + { + AttributeName: aws.String("Key"), + AttributeType: types.ScalarAttributeTypeS, + }, + }, + KeySchema: []types.KeySchemaElement{ + { + AttributeName: aws.String("Key"), + KeyType: types.KeyTypeHash, + }, + }, + BillingMode: types.BillingModePayPerRequest, + }) - if manifestSignatures, found = protoRepoMeta.Signatures[signedManifestDigest.String()]; !found { - return zerr.ErrManifestMetaNotFound + if err != nil && !strings.Contains(err.Error(), "Table already exists") { + return err } - signatureSlice := manifestSignatures.Map[sigType] + return dwr.waitTableToBeCreated(tableName) +} - newSignatureSlice := make([]*proto_go.SignatureInfo, 0, len(signatureSlice.List)-1) +func (dwr *DynamoDB) deleteTable(tableName string) error { + _, err := dwr.Client.DeleteTable(context.Background(), &dynamodb.DeleteTableInput{ + TableName: aws.String(tableName), + }) - for _, sigDigest := range signatureSlice.List { - if sigDigest.SignatureManifestDigest != sigMeta.SignatureDigest { - newSignatureSlice = append(newSignatureSlice, sigDigest) - } + if temp := new(types.ResourceNotFoundException); errors.As(err, &temp) { + return nil } - manifestSignatures.Map[sigMeta.SignatureType] = &proto_go.SignaturesInfo{List: newSignatureSlice} + return dwr.waitTableToBeDeleted(tableName) +} - protoRepoMeta.Signatures[signedManifestDigest.String()] = manifestSignatures +func (dwr *DynamoDB) waitTableToBeCreated(tableName string) error { + const maxWaitTime = 20 * time.Second - return dwr.setProtoRepoMeta(protoRepoMeta.Name, protoRepoMeta) + waiter := dynamodb.NewTableExistsWaiter(dwr.Client) + + return waiter.Wait(context.Background(), &dynamodb.DescribeTableInput{ + TableName: &tableName, + }, maxWaitTime) } -func (dwr *DynamoDB) ProtoFilterImageData(ctx context.Context, digests []string, -) (map[string]mTypes.ImageData2, error) { - imageDataAttributes, err := dwr.fetchImageDataAttributesByDigest(ctx, digests) +func (dwr *DynamoDB) waitTableToBeDeleted(tableName string) error { + const maxWaitTime = 20 * time.Second + + waiter := dynamodb.NewTableNotExistsWaiter(dwr.Client) + + return waiter.Wait(context.Background(), &dynamodb.DescribeTableInput{ + TableName: &tableName, + }, maxWaitTime) +} + +func (dwr *DynamoDB) createVersionTable() error { + _, err := dwr.Client.CreateTable(context.Background(), &dynamodb.CreateTableInput{ + TableName: aws.String(dwr.VersionTablename), + AttributeDefinitions: []types.AttributeDefinition{ + { + AttributeName: aws.String("Key"), + AttributeType: types.ScalarAttributeTypeS, + }, + }, + KeySchema: []types.KeySchemaElement{ + { + AttributeName: aws.String("Key"), + KeyType: types.KeyTypeHash, + }, + }, + BillingMode: types.BillingModePayPerRequest, + }) if err != nil { - return nil, err + if strings.Contains(err.Error(), "Table already exists") { + return nil + } + + return err } - results := map[string]mTypes.ImageData2{} + err = dwr.waitTableToBeCreated(dwr.VersionTablename) + if err != nil { + return err + } - for _, attributes := range imageDataAttributes { - protoImageData, err := getProtoImageDataFromAttribute(attributes["ImageData"]) + if err == nil { + mdAttributeValue, err := attributevalue.Marshal(version.CurrentVersion) if err != nil { - return nil, err + return err } - if protoImageData.MediaType == ispec.MediaTypeImageIndex { - indexDigests := []string{} - for i := range protoImageData.Manifests { - indexDigests = append(indexDigests, protoImageData.Manifests[i].Digest) - } - - manifestsAttributes, err := dwr.fetchImageDataAttributesByDigest(ctx, indexDigests) - if err != nil { - return nil, err - } - - for i, manifestAttribute := range manifestsAttributes { - imageManifestData, err := getProtoImageDataFromAttribute(manifestAttribute["ImageData"]) - if err != nil { - return nil, err - } + _, err = dwr.Client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ + ExpressionAttributeNames: map[string]string{ + "#V": "Version", + }, + ExpressionAttributeValues: map[string]types.AttributeValue{ + ":Version": mdAttributeValue, + }, + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{ + Value: version.DBVersionKey, + }, + }, + TableName: aws.String(dwr.VersionTablename), + UpdateExpression: aws.String("SET #V = :Version"), + }) - protoImageData.Manifests[i] = imageManifestData.Manifests[0] - } + if err != nil { + return err } - - results[protoImageData.Digest] = mConvert.GetImageData(protoImageData) } - return results, nil + return nil } -func (dwr *DynamoDB) fetchImageDataAttributesByDigest(ctx context.Context, digests []string, -) ([]map[string]types.AttributeValue, error) { - resp, err := dwr.Client.BatchGetItem(ctx, &dynamodb.BatchGetItemInput{ - RequestItems: map[string]types.KeysAndAttributes{ - dwr.ProtoImageDataTablename: { - Keys: getBatchImageKeys(digests), - }, +func (dwr *DynamoDB) getDBVersion() (string, error) { + resp, err := dwr.Client.GetItem(context.TODO(), &dynamodb.GetItemInput{ + TableName: aws.String(dwr.VersionTablename), + Key: map[string]types.AttributeValue{ + "Key": &types.AttributeValueMemberS{Value: version.DBVersionKey}, }, }) if err != nil { - return nil, err + return "", err } - return resp.Responses[dwr.ProtoImageDataTablename], nil -} + if resp.Item == nil { + return "", nil + } -func getBatchImageKeys(digests []string) []map[string]types.AttributeValue { - result := []map[string]types.AttributeValue{} + var version string - for _, digest := range digests { - result = append(result, map[string]types.AttributeValue{ - "Digest": &types.AttributeValueMemberS{ - Value: digest, - }, - }) + err = attributevalue.Unmarshal(resp.Item["Version"], &version) + if err != nil { + return "", err } - return result + return version, nil } diff --git a/pkg/meta/dynamodb/dynamodb_internal_test.go b/pkg/meta/dynamodb/dynamodb_internal_test.go index cad85a330b..42157aab9a 100644 --- a/pkg/meta/dynamodb/dynamodb_internal_test.go +++ b/pkg/meta/dynamodb/dynamodb_internal_test.go @@ -29,8 +29,6 @@ func TestWrapperErrors(t *testing.T) { } repoMetaTablename := "RepoMetadataTable" + uuid.String() - manifestDataTablename := "ManifestDataTable" + uuid.String() - indexDataTablename := "IndexDataTable" + uuid.String() userDataTablename := "UserDataTable" + uuid.String() apiKeyTablename := "ApiKeyTable" + uuid.String() @@ -54,31 +52,23 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldBeNil) dynamoWrapper := DynamoDB{ - Client: dynamodb.NewFromConfig(cfg), - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - VersionTablename: versionTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - Patches: version.GetDynamoDBPatches(), - Log: log.Logger{Logger: zerolog.New(os.Stdout)}, + Client: dynamodb.NewFromConfig(cfg), + RepoMetaTablename: repoMetaTablename, + VersionTablename: versionTablename, + UserDataTablename: userDataTablename, + APIKeyTablename: apiKeyTablename, + Patches: version.GetDynamoDBPatches(), + Log: log.Logger{Logger: zerolog.New(os.Stdout)}, } // The table creation should fail as the endpoint is not configured correctly - err = dynamoWrapper.createRepoMetaTable() - So(err, ShouldNotBeNil) - - err = dynamoWrapper.createManifestDataTable() - So(err, ShouldNotBeNil) - - err = dynamoWrapper.createIndexDataTable() + err = dynamoWrapper.createTable(dynamoWrapper.RepoMetaTablename) So(err, ShouldNotBeNil) err = dynamoWrapper.createVersionTable() So(err, ShouldNotBeNil) - err = dynamoWrapper.createAPIKeyTable() + err = dynamoWrapper.createTable(dynamoWrapper.APIKeyTablename) So(err, ShouldNotBeNil) }) @@ -98,21 +88,16 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldBeNil) dynamoWrapper := DynamoDB{ - Client: dynamodb.NewFromConfig(cfg), - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - VersionTablename: versionTablename, - IndexDataTablename: indexDataTablename, - UserDataTablename: userDataTablename, - Patches: version.GetDynamoDBPatches(), - Log: log.Logger{Logger: zerolog.New(os.Stdout)}, + Client: dynamodb.NewFromConfig(cfg), + RepoMetaTablename: repoMetaTablename, + VersionTablename: versionTablename, + UserDataTablename: userDataTablename, + Patches: version.GetDynamoDBPatches(), + Log: log.Logger{Logger: zerolog.New(os.Stdout)}, } // The tables were not created so delete calls fail, but dynamoWrapper should not error - err = dynamoWrapper.deleteRepoMetaTable() - So(err, ShouldBeNil) - - err = dynamoWrapper.deleteManifestDataTable() + err = dynamoWrapper.deleteTable(dynamoWrapper.RepoMetaTablename) So(err, ShouldBeNil) }) } diff --git a/pkg/meta/dynamodb/dynamodb_test.go b/pkg/meta/dynamodb/dynamodb_test.go index fe0bb32430..c4bf71bf82 100644 --- a/pkg/meta/dynamodb/dynamodb_test.go +++ b/pkg/meta/dynamodb/dynamodb_test.go @@ -2,9 +2,7 @@ package dynamodb_test import ( "context" - "fmt" "os" - "strings" "testing" "github.com/aws/aws-sdk-go-v2/aws" @@ -13,19 +11,16 @@ import ( "github.com/aws/aws-sdk-go-v2/service/dynamodb" "github.com/aws/aws-sdk-go-v2/service/dynamodb/types" guuid "github.com/gofrs/uuid" - "github.com/opencontainers/go-digest" - ispec "github.com/opencontainers/image-spec/specs-go/v1" "github.com/rs/zerolog" . "github.com/smartystreets/goconvey/convey" "zotregistry.io/zot/pkg/extensions/imagetrust" "zotregistry.io/zot/pkg/log" - "zotregistry.io/zot/pkg/meta" mdynamodb "zotregistry.io/zot/pkg/meta/dynamodb" mTypes "zotregistry.io/zot/pkg/meta/types" reqCtx "zotregistry.io/zot/pkg/requestcontext" - tskip "zotregistry.io/zot/pkg/test/skip" . "zotregistry.io/zot/pkg/test/image-utils" + tskip "zotregistry.io/zot/pkg/test/skip" ) const badTablename = "bad tablename" @@ -42,9 +37,9 @@ func TestIterator(t *testing.T) { } repoMetaTablename := "RepoMetadataTable" + uuid.String() - manifestDataTablename := "ManifestDataTable" + uuid.String() versionTablename := "Version" + uuid.String() - indexDataTablename := "IndexDataTable" + uuid.String() + imageDataTablename := "ImageData" + uuid.String() + repoBlobsTablename := "RepoBlobs" + uuid.String() userDataTablename := "UserDataTable" + uuid.String() apiKeyTablename := "ApiKeyTable" + uuid.String() @@ -52,14 +47,14 @@ func TestIterator(t *testing.T) { Convey("TestIterator", t, func() { params := mdynamodb.DBDriverParameters{ - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - VersionTablename: versionTablename, - APIKeyTablename: apiKeyTablename, - UserDataTablename: userDataTablename, + Endpoint: endpoint, + Region: region, + RepoMetaTablename: repoMetaTablename, + ImageDataTablename: imageDataTablename, + RepoBlobsInfoTablename: repoBlobsTablename, + VersionTablename: versionTablename, + APIKeyTablename: apiKeyTablename, + UserDataTablename: userDataTablename, } client, err := mdynamodb.GetDynamoClient(params) So(err, ShouldBeNil) @@ -67,21 +62,21 @@ func TestIterator(t *testing.T) { dynamoWrapper, err := mdynamodb.New(client, params, log) So(err, ShouldBeNil) - So(dynamoWrapper.ResetManifestDataTable(), ShouldBeNil) - So(dynamoWrapper.ResetRepoMetaTable(), ShouldBeNil) + So(dynamoWrapper.ResetTable(dynamoWrapper.ImageDataTablename), ShouldBeNil) + So(dynamoWrapper.ResetTable(dynamoWrapper.RepoMetaTablename), ShouldBeNil) - err = dynamoWrapper.SetRepoReference("repo1", "tag1", "manifestType", "manifestDigest1") + err = dynamoWrapper.SetRepoReference("repo1", "tag1", CreateRandomImage().AsImageData()) So(err, ShouldBeNil) - err = dynamoWrapper.SetRepoReference("repo2", "tag2", "manifestType", "manifestDigest2") + err = dynamoWrapper.SetRepoReference("repo2", "tag2", CreateRandomImage().AsImageData()) So(err, ShouldBeNil) - err = dynamoWrapper.SetRepoReference("repo3", "tag3", "manifestType", "manifestDigest3") + err = dynamoWrapper.SetRepoReference("repo3", "tag3", CreateRandomImage().AsImageData()) So(err, ShouldBeNil) repoMetaAttributeIterator := mdynamodb.NewBaseDynamoAttributesIterator( dynamoWrapper.Client, - repoMetaTablename, + dynamoWrapper.RepoMetaTablename, "RepoMetadata", 1, log, @@ -145,25 +140,25 @@ func TestWrapperErrors(t *testing.T) { } repoMetaTablename := "RepoMetadataTable" + uuid.String() - manifestDataTablename := "ManifestDataTable" + uuid.String() versionTablename := "Version" + uuid.String() - indexDataTablename := "IndexDataTable" + uuid.String() userDataTablename := "UserDataTable" + uuid.String() apiKeyTablename := "ApiKeyTable" + uuid.String() wrongTableName := "WRONG Tables" + imageDataTablename := "ImageData" + uuid.String() + repoBlobsTablename := "RepoBlobs" + uuid.String() log := log.NewLogger("debug", "") Convey("Errors", t, func() { params := mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - VersionTablename: versionTablename, + Endpoint: endpoint, + Region: region, + RepoMetaTablename: repoMetaTablename, + ImageDataTablename: imageDataTablename, + RepoBlobsInfoTablename: repoBlobsTablename, + UserDataTablename: userDataTablename, + APIKeyTablename: apiKeyTablename, + VersionTablename: versionTablename, } client, err := mdynamodb.GetDynamoClient(params) //nolint:contextcheck So(err, ShouldBeNil) @@ -176,8 +171,9 @@ func TestWrapperErrors(t *testing.T) { dynamoWrapper.SetImageTrustStore(imgTrustStore) - So(dynamoWrapper.ResetManifestDataTable(), ShouldBeNil) //nolint:contextcheck - So(dynamoWrapper.ResetRepoMetaTable(), ShouldBeNil) //nolint:contextcheck + So(dynamoWrapper.ResetTable(dynamoWrapper.RepoMetaTablename), ShouldBeNil) //nolint:contextcheck + So(dynamoWrapper.ResetTable(dynamoWrapper.ImageDataTablename), ShouldBeNil) //nolint:contextcheck + So(dynamoWrapper.ResetTable(dynamoWrapper.UserDataTablename), ShouldBeNil) //nolint:contextcheck userAc := reqCtx.NewUserAccessControl() userAc.SetUsername("test") @@ -242,7 +238,7 @@ func TestWrapperErrors(t *testing.T) { status, err := dynamoWrapper.ToggleBookmarkRepo(ctx, "repo") So(err, ShouldBeNil) - So(status, ShouldEqual, mTypes.NotChanged) + So(status, ShouldEqual, mTypes.Added) }) Convey("ToggleBookmarkRepo GetUserMeta client error", func() { @@ -478,608 +474,6 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldNotBeNil) }) - Convey("SetManifestData", func() { - dynamoWrapper.ManifestDataTablename = wrongTableName - - err := dynamoWrapper.SetManifestData("dig", mTypes.ManifestData{}) - So(err, ShouldNotBeNil) - }) - - Convey("GetManifestData", func() { - dynamoWrapper.ManifestDataTablename = wrongTableName - - _, err := dynamoWrapper.GetManifestData("dig") - So(err, ShouldNotBeNil) - }) - - Convey("GetManifestData unmarshal error", func() { - err := setBadManifestData(dynamoWrapper.Client, manifestDataTablename, "dig") - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetManifestData("dig") - So(err, ShouldNotBeNil) - }) - - Convey("GetIndexData", func() { - dynamoWrapper.IndexDataTablename = wrongTableName - - _, err := dynamoWrapper.GetIndexData("dig") - So(err, ShouldNotBeNil) - }) - - Convey("GetIndexData unmarshal error", func() { - err := setBadIndexData(dynamoWrapper.Client, indexDataTablename, "dig") - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetManifestData("dig") - So(err, ShouldNotBeNil) - }) - - Convey("SetManifestMeta GetRepoMeta error", func() { - err := setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo1") - So(err, ShouldBeNil) - - err = dynamoWrapper.SetManifestMeta("repo1", "dig", mTypes.ManifestMetadata{}) - So(err, ShouldNotBeNil) - }) - - Convey("GetManifestMeta GetManifestData not found error", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag", "dig", "") - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetManifestMeta("repo", "dig") - So(err, ShouldNotBeNil) - }) - - Convey("GetManifestMeta GetRepoMeta Not Found error", func() { - err := dynamoWrapper.SetManifestData("dig", mTypes.ManifestData{}) - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetManifestMeta("repoNotFound", "dig") - So(err, ShouldNotBeNil) - }) - - Convey("GetManifestMeta GetRepoMeta error", func() { - err := dynamoWrapper.SetManifestData("dig", mTypes.ManifestData{}) - So(err, ShouldBeNil) - - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetManifestMeta("repo", "dig") - So(err, ShouldNotBeNil) - }) - - Convey("SetRepoReference client error", func() { - dynamoWrapper.RepoMetaTablename = badTablename - digest := digest.FromString("str") - err := dynamoWrapper.SetRepoReference("repo", digest.String(), digest, ispec.MediaTypeImageManifest) - So(err, ShouldNotBeNil) - }) - - Convey("SetReferrer client error", func() { - dynamoWrapper.RepoMetaTablename = badTablename - err := dynamoWrapper.SetReferrer("repo", "", mTypes.ReferrerInfo{}) - So(err, ShouldNotBeNil) - }) - - Convey("SetReferrer bad repoMeta", func() { - err := setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - err = dynamoWrapper.SetReferrer("repo", "", mTypes.ReferrerInfo{}) - So(err, ShouldNotBeNil) - }) - - Convey("GetReferrers client error", func() { - dynamoWrapper.RepoMetaTablename = badTablename - _, err := dynamoWrapper.GetReferrers("repo", "") - So(err, ShouldNotBeNil) - }) - - Convey("GetReferrers bad repoMeta", func() { - err := setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetReferrers("repo", "") - So(err, ShouldNotBeNil) - }) - - Convey("DeleteReferrer client error", func() { - dynamoWrapper.RepoMetaTablename = badTablename - err := dynamoWrapper.DeleteReferrer("repo", "", "") - So(err, ShouldNotBeNil) - }) - - Convey("DeleteReferrer bad repoMeta", func() { - err := setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - err = dynamoWrapper.DeleteReferrer("repo", "", "") - So(err, ShouldNotBeNil) - }) - - Convey("GetReferrersInfo GetReferrers errors", func() { - dynamoWrapper.RepoMetaTablename = badTablename - _, err := dynamoWrapper.GetReferrersInfo("repo", "", nil) - So(err, ShouldNotBeNil) - }) - - Convey("GetReferrersInfo getData fails", func() { - dynamoWrapper.ManifestDataTablename = badTablename - err = dynamoWrapper.SetReferrer("repo", "rf", mTypes.ReferrerInfo{ - Digest: "dig1", - MediaType: ispec.MediaTypeImageManifest, - }) - So(err, ShouldBeNil) - - err = dynamoWrapper.SetReferrer("repo", "rf", mTypes.ReferrerInfo{ - Digest: "dig2", - MediaType: ispec.MediaTypeImageManifest, - }) - So(err, ShouldBeNil) - - _, err := dynamoWrapper.GetReferrersInfo("repo", "rf", nil) - So(err, ShouldBeNil) - }) - - Convey("GetReferrersInfo bad descriptor blob", func() { - err = dynamoWrapper.SetManifestData("dig3", mTypes.ManifestData{ - ManifestBlob: []byte("bad json"), - }) - So(err, ShouldBeNil) - - err = dynamoWrapper.SetReferrer("repo", "rf", mTypes.ReferrerInfo{ - Digest: "dig2", - MediaType: ispec.MediaTypeImageManifest, - }) - So(err, ShouldBeNil) - - err = dynamoWrapper.SetReferrer("repo", "rf", mTypes.ReferrerInfo{ - Digest: "dig3", - MediaType: ispec.MediaTypeImageManifest, - }) - So(err, ShouldBeNil) - - _, err := dynamoWrapper.GetReferrersInfo("repo", "rf", nil) - So(err, ShouldBeNil) - }) - - Convey("IncrementRepoStars GetRepoMeta error", func() { - err = dynamoWrapper.IncrementRepoStars("repo") - So(err, ShouldNotBeNil) - }) - - Convey("DecrementRepoStars GetRepoMeta error", func() { - err = dynamoWrapper.DecrementRepoStars("repo") - So(err, ShouldNotBeNil) - }) - - Convey("DeleteRepoTag Client.GetItem error", func() { - strSlice := make([]string, 10000) - repoName := strings.Join(strSlice, ".") - - err = dynamoWrapper.DeleteRepoTag(repoName, "tag") - So(err, ShouldNotBeNil) - }) - - Convey("DeleteRepoTag unmarshal error", func() { - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - err = dynamoWrapper.DeleteRepoTag("repo", "tag") - So(err, ShouldNotBeNil) - }) - - Convey("GetRepoMeta Client.GetItem error", func() { - strSlice := make([]string, 10000) - repoName := strings.Join(strSlice, ".") - - _, err = dynamoWrapper.GetRepoMeta(repoName) - So(err, ShouldNotBeNil) - }) - - Convey("GetRepoMeta unmarshal error", func() { - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetRepoMeta("repo") - So(err, ShouldNotBeNil) - }) - - Convey("IncrementImageDownloads GetRepoMeta error", func() { - err = dynamoWrapper.IncrementImageDownloads("repoNotFound", "") - So(err, ShouldNotBeNil) - }) - - Convey("IncrementImageDownloads tag not found error", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag", "dig", "") - So(err, ShouldBeNil) - - err = dynamoWrapper.IncrementImageDownloads("repo", "notFoundTag") - So(err, ShouldNotBeNil) - }) - - Convey("UpdateSignaturesValidity GetManifestData error", func() { - err := setBadManifestData(dynamoWrapper.Client, manifestDataTablename, "dig") - So(err, ShouldBeNil) - - err = dynamoWrapper.UpdateSignaturesValidity("repo", "dig") - So(err, ShouldNotBeNil) - - err = dynamoWrapper.UpdateSignaturesValidity("repo", digest.FromString("dig")) - So(err, ShouldBeNil) - }) - - Convey("UpdateSignaturesValidity GetRepoMeta error", func() { - err := dynamoWrapper.SetManifestData("dig", mTypes.ManifestData{}) - So(err, ShouldBeNil) - - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - err = dynamoWrapper.UpdateSignaturesValidity("repo", "dig") - So(err, ShouldNotBeNil) - }) - - Convey("AddManifestSignature GetRepoMeta error", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag", "dig", "") - So(err, ShouldBeNil) - - err = dynamoWrapper.AddManifestSignature("repoNotFound", "tag", mTypes.SignatureMetadata{}) - So(err, ShouldNotBeNil) - }) - - Convey("AddManifestSignature ManifestSignatures signedManifestDigest not found error", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag", "dig", "") - So(err, ShouldBeNil) - - err = dynamoWrapper.AddManifestSignature("repo", "tagNotFound", mTypes.SignatureMetadata{}) - So(err, ShouldNotBeNil) - }) - - Convey("AddManifestSignature SignatureType metadb.NotationType", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag", "dig", "") - So(err, ShouldBeNil) - - err = dynamoWrapper.AddManifestSignature("repo", "tagNotFound", mTypes.SignatureMetadata{ - SignatureType: "notation", - }) - So(err, ShouldBeNil) - }) - - Convey("DeleteSignature GetRepoMeta error", func() { - err = dynamoWrapper.DeleteSignature("repoNotFound", "tagNotFound", mTypes.SignatureMetadata{}) - So(err, ShouldNotBeNil) - }) - - Convey("DeleteSignature sigDigest.SignatureManifestDigest != sigMeta.SignatureDigest true", func() { - err := setRepoMeta(dynamoWrapper.Client, repoMetaTablename, mTypes.RepoMetadata{ - Name: "repo", - Signatures: map[string]mTypes.ManifestSignatures{ - "tag1": { - "cosign": []mTypes.SignatureInfo{ - {SignatureManifestDigest: "dig1"}, - {SignatureManifestDigest: "dig2"}, - }, - }, - }, - }) - So(err, ShouldBeNil) - - err = dynamoWrapper.DeleteSignature("repo", "tag1", mTypes.SignatureMetadata{ - SignatureDigest: "dig2", - SignatureType: "cosign", - }) - So(err, ShouldBeNil) - }) - - Convey("GetMultipleRepoMeta unmarshal error", func() { - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") //nolint:contextcheck - So(err, ShouldBeNil) - - _, err = dynamoWrapper.GetMultipleRepoMeta(ctx, func(repoMeta mTypes.RepoMetadata) bool { return true }) - - So(err, ShouldNotBeNil) - }) - - Convey("SearchRepos repoMeta unmarshal error", func() { - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchRepos(ctx, "") - - So(err, ShouldNotBeNil) - }) - - Convey("SearchRepos bad tablename", func() { - dynamoWrapper.RepoMetaTablename = badTablename - - _, _, _, err = dynamoWrapper.SearchRepos(ctx, "") - - So(err, ShouldNotBeNil) - }) - - Convey("GetMultipleRepoMeta bad tablename", func() { - dynamoWrapper.RepoMetaTablename = badTablename - - _, err = dynamoWrapper.GetMultipleRepoMeta(ctx, func(repoMeta mTypes.RepoMetadata) bool { return true }) - - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags bad tablename", func() { - dynamoWrapper.RepoMetaTablename = badTablename - - _, _, _, err = dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return true - }) - - So(err, ShouldNotBeNil) - }) - - Convey("FilterRepos bad tablename", func() { - dynamoWrapper.RepoMetaTablename = badTablename - - _, _, _, err = dynamoWrapper.FilterRepos(ctx, func(repoMeta mTypes.RepoMetadata) bool { return true }) - - So(err, ShouldNotBeNil) - }) - - Convey("SearchTags bad tablename", func() { - dynamoWrapper.RepoMetaTablename = badTablename - - _, _, _, err = dynamoWrapper.SearchTags(ctx, "repo:tag") - - So(err, ShouldNotBeNil) - }) - - Convey("SearchRepos GetManifestMeta error", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag1", "notFoundDigest", //nolint:contextcheck - ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchRepos(ctx, "") - - So(err, ShouldNotBeNil) - }) - - Convey("Unsuported type", func() { - digest := digest.FromString("digest") - - err := dynamoWrapper.SetRepoReference("repo", "tag1", digest, "invalid type") //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchRepos(ctx, "") - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchTags(ctx, "repo:") - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) - So(err, ShouldBeNil) - }) - - Convey("SearchRepos bad index data", func() { - indexDigest := digest.FromString("indexDigest") - - err := dynamoWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = setBadIndexData(dynamoWrapper.Client, indexDataTablename, indexDigest.String()) //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchRepos(ctx, "") - So(err, ShouldNotBeNil) - }) - - Convey("SearchRepos bad indexBlob in IndexData", func() { - indexDigest := digest.FromString("indexDigest") - - err := dynamoWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = dynamoWrapper.SetIndexData(indexDigest, mTypes.IndexData{ //nolint:contextcheck - IndexBlob: []byte("bad json"), - }) - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchRepos(ctx, "") - So(err, ShouldNotBeNil) - }) - - Convey("SearchTags repoMeta unmarshal error", func() { - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchTags(ctx, "repo:") - - So(err, ShouldNotBeNil) - }) - - Convey("SearchTags GetManifestMeta error", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag1", "manifestNotFound", //nolint:contextcheck - ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchTags(ctx, "repo:") - - So(err, ShouldNotBeNil) - }) - - Convey("SearchTags bad index data", func() { - indexDigest := digest.FromString("indexDigest") - - err := dynamoWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = setBadIndexData(dynamoWrapper.Client, indexDataTablename, indexDigest.String()) //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchTags(ctx, "repo:") - So(err, ShouldNotBeNil) - }) - - Convey("SearchTags bad indexBlob in IndexData", func() { - indexDigest := digest.FromString("indexDigest") - - err := dynamoWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = dynamoWrapper.SetIndexData(indexDigest, mTypes.IndexData{ //nolint:contextcheck - IndexBlob: []byte("bad json"), - }) - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.SearchTags(ctx, "repo:") - So(err, ShouldNotBeNil) - }) - - Convey("SearchRepos attr", func() { - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err := dynamoWrapper.SearchRepos(ctx, "repo") - So(err, ShouldNotBeNil) - }) - - Convey("FilterRepos attributevalue.Unmarshal(repoMetaAttribute) errors", func() { - dynamoWrapper.RepoMetaTablename = "bad-table-FilterRepos" - - _, _, _, err := dynamoWrapper.FilterRepos(ctx, func(repoMeta mTypes.RepoMetadata) bool { - return true - }) - So(err, ShouldNotBeNil) - }) - - Convey("SearchRepos bad RepoMeta table name", func() { - dynamoWrapper.RepoMetaTablename = "SearchRepos-bad-table" - - _, _, _, err := dynamoWrapper.SearchRepos(ctx, "repo") - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags repoMeta unmarshal error", func() { - err = setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return true - }) - - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags bad RepoMeta table name", func() { - dynamoWrapper.RepoMetaTablename = "bad-table" - - _, _, _, err := dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return true - }) - - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags manifestMeta not found", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag1", "manifestNotFound", //nolint:contextcheck - ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return true - }) - - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags manifestMeta unmarshal error", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag1", "dig", ispec.MediaTypeImageManifest) //nolint:contextcheck - So(err, ShouldBeNil) - - err = setBadManifestData(dynamoWrapper.Client, manifestDataTablename, "dig") //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.FilterTags( - ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return true - }) - - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags bad IndexData", func() { - indexDigest := digest.FromString("indexDigest") - - err := dynamoWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = setBadIndexData(dynamoWrapper.Client, indexDataTablename, indexDigest.String()) //nolint:contextcheck - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags bad indexBlob in IndexData", func() { - indexDigest := digest.FromString("indexDigest") - - err := dynamoWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - err = dynamoWrapper.SetIndexData(indexDigest, mTypes.IndexData{ //nolint:contextcheck - IndexBlob: []byte("bad json"), - }) - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) - So(err, ShouldNotBeNil) - }) - - Convey("FilterTags didn't match any index manifest", func() { - var ( - indexDigest = digest.FromString("indexDigest") - manifestDigestFromIndex1 = digest.FromString("manifestDigestFromIndex1") - manifestDigestFromIndex2 = digest.FromString("manifestDigestFromIndex2") - ) - - err := dynamoWrapper.SetRepoReference("repo", "tag1", indexDigest, ispec.MediaTypeImageIndex) //nolint:contextcheck - So(err, ShouldBeNil) - - indexBlob, err := GetIndexBlobWithManifests([]digest.Digest{ - manifestDigestFromIndex1, manifestDigestFromIndex2, - }) - So(err, ShouldBeNil) - - err = dynamoWrapper.SetIndexData(indexDigest, mTypes.IndexData{ //nolint:contextcheck - IndexBlob: indexBlob, - }) - So(err, ShouldBeNil) - - err = dynamoWrapper.SetManifestData(manifestDigestFromIndex1, mTypes.ManifestData{ //nolint:contextcheck - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) - - err = dynamoWrapper.SetManifestData(manifestDigestFromIndex2, mTypes.ManifestData{ //nolint:contextcheck - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) - - _, _, _, err = dynamoWrapper.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return false }) - So(err, ShouldBeNil) - }) - Convey("PatchDB dwr.getDBVersion errors", func() { dynamoWrapper.VersionTablename = badTablename @@ -1104,7 +498,7 @@ func TestWrapperErrors(t *testing.T) { Convey("ResetRepoMetaTable client errors", func() { dynamoWrapper.RepoMetaTablename = badTablename - err := dynamoWrapper.ResetRepoMetaTable() + err := dynamoWrapper.ResetTable(dynamoWrapper.RepoMetaTablename) So(err, ShouldNotBeNil) }) @@ -1114,61 +508,18 @@ func TestWrapperErrors(t *testing.T) { err := dynamoWrapper.PatchDB() So(err, ShouldNotBeNil) }) - - Convey("GetUserRepoMeta client.GetItem error", func() { - dynamoWrapper.RepoMetaTablename = badTablename - - _, err = dynamoWrapper.GetUserRepoMeta(ctx, "repo") - So(err, ShouldNotBeNil) - }) - - Convey("GetUserRepoMeta repoMeta not found", func() { - _, err = dynamoWrapper.GetUserRepoMeta(ctx, "unknown-repo-meta") - So(err, ShouldNotBeNil) - }) - - Convey("GetUserRepoMeta userMeta not found", func() { - err := dynamoWrapper.SetRepoReference("repo", "tag", digest.FromString("1"), ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - dynamoWrapper.UserDataTablename = badTablename - - userAc := reqCtx.NewUserAccessControl() - userAc.SetUsername("username") - userAc.SetGlobPatterns("read", map[string]bool{ - "repo": true, - }) - ctx := userAc.DeriveContext(context.Background()) - - _, err = dynamoWrapper.GetUserRepoMeta(ctx, "repo") - So(err, ShouldNotBeNil) - }) - - Convey("GetUserRepoMeta unmarshal error", func() { - err := setBadRepoMeta(dynamoWrapper.Client, repoMetaTablename, "repo") - So(err, ShouldBeNil) - - userAc := reqCtx.NewUserAccessControl() - userAc.SetUsername("username") - userAc.SetGlobPatterns("read", map[string]bool{ - "repo": true, - }) - ctx := userAc.DeriveContext(context.Background()) - - _, err = dynamoWrapper.GetUserRepoMeta(ctx, "repo") - So(err, ShouldNotBeNil) - }) }) Convey("NewDynamoDBWrapper errors", t, func() { params := mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: "", - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - VersionTablename: versionTablename, + Endpoint: endpoint, + Region: region, + RepoMetaTablename: "", + ImageDataTablename: imageDataTablename, + RepoBlobsInfoTablename: repoBlobsTablename, + UserDataTablename: userDataTablename, + APIKeyTablename: apiKeyTablename, + VersionTablename: versionTablename, } client, err := mdynamodb.GetDynamoClient(params) So(err, ShouldBeNil) @@ -1177,30 +528,14 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldNotBeNil) params = mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: "", - IndexDataTablename: indexDataTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - VersionTablename: versionTablename, - } - client, err = mdynamodb.GetDynamoClient(params) - So(err, ShouldBeNil) - - _, err = mdynamodb.New(client, params, log) - So(err, ShouldNotBeNil) - - params = mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: "", - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - VersionTablename: versionTablename, + Endpoint: endpoint, + Region: region, + RepoMetaTablename: repoMetaTablename, + ImageDataTablename: imageDataTablename, + RepoBlobsInfoTablename: repoBlobsTablename, + UserDataTablename: userDataTablename, + APIKeyTablename: apiKeyTablename, + VersionTablename: "", } client, err = mdynamodb.GetDynamoClient(params) So(err, ShouldBeNil) @@ -1209,46 +544,14 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldNotBeNil) params = mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - VersionTablename: "", - } - client, err = mdynamodb.GetDynamoClient(params) - So(err, ShouldBeNil) - - _, err = mdynamodb.New(client, params, log) - So(err, ShouldNotBeNil) - - params = mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - VersionTablename: versionTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - } - client, err = mdynamodb.GetDynamoClient(params) - So(err, ShouldBeNil) - - _, err = mdynamodb.New(client, params, log) - So(err, ShouldBeNil) - - params = mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - VersionTablename: versionTablename, - UserDataTablename: "", - APIKeyTablename: apiKeyTablename, + Endpoint: endpoint, + Region: region, + RepoMetaTablename: repoMetaTablename, + ImageDataTablename: imageDataTablename, + RepoBlobsInfoTablename: repoBlobsTablename, + VersionTablename: versionTablename, + UserDataTablename: "", + APIKeyTablename: apiKeyTablename, } client, err = mdynamodb.GetDynamoClient(params) So(err, ShouldBeNil) @@ -1257,14 +560,14 @@ func TestWrapperErrors(t *testing.T) { So(err, ShouldNotBeNil) params = mdynamodb.DBDriverParameters{ //nolint:contextcheck - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - VersionTablename: versionTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: "", + Endpoint: endpoint, + Region: region, + RepoMetaTablename: repoMetaTablename, + ImageDataTablename: imageDataTablename, + RepoBlobsInfoTablename: repoBlobsTablename, + VersionTablename: versionTablename, + UserDataTablename: userDataTablename, + APIKeyTablename: "", } client, err = mdynamodb.GetDynamoClient(params) So(err, ShouldBeNil) @@ -1274,81 +577,6 @@ func TestWrapperErrors(t *testing.T) { }) } -func setBadManifestData(client *dynamodb.Client, manifestDataTableName, digest string) error { - mdAttributeValue, err := attributevalue.Marshal("string") - if err != nil { - return err - } - - _, err = client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#MD": "ManifestData", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":ManifestData": mdAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "Digest": &types.AttributeValueMemberS{ - Value: digest, - }, - }, - TableName: aws.String(manifestDataTableName), - UpdateExpression: aws.String("SET #MD = :ManifestData"), - }) - - return err -} - -func setBadRepoMeta(client *dynamodb.Client, repoMetadataTableName, repoName string) error { - repoAttributeValue, err := attributevalue.Marshal("string") - if err != nil { - return err - } - - _, err = client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#RM": "RepoMetadata", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoMetadata": repoAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ - Value: repoName, - }, - }, - TableName: aws.String(repoMetadataTableName), - UpdateExpression: aws.String("SET #RM = :RepoMetadata"), - }) - - return err -} - -func setBadIndexData(client *dynamodb.Client, indexDataTableName, digest string) error { - mdAttributeValue, err := attributevalue.Marshal("string") - if err != nil { - return err - } - - _, err = client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#ID": "IndexData", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":IndexData": mdAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "IndexDigest": &types.AttributeValueMemberS{ - Value: digest, - }, - }, - TableName: aws.String(indexDataTableName), - UpdateExpression: aws.String("SET #ID = :IndexData"), - }) - - return err -} - func setBadUserData(client *dynamodb.Client, userDataTablename, userID string) error { userAttributeValue, err := attributevalue.Marshal("string") if err != nil { @@ -1363,7 +591,7 @@ func setBadUserData(client *dynamodb.Client, userDataTablename, userID string) e ":UserData": userAttributeValue, }, Key: map[string]types.AttributeValue{ - "Identity": &types.AttributeValueMemberS{ + "Key": &types.AttributeValueMemberS{ Value: userID, }, }, @@ -1388,7 +616,7 @@ func setVersion(client *dynamodb.Client, versionTablename string, version string ":Version": mdAttributeValue, }, Key: map[string]types.AttributeValue{ - "VersionKey": &types.AttributeValueMemberS{ + "Key": &types.AttributeValueMemberS{ Value: "DBVersion", }, }, @@ -1398,111 +626,3 @@ func setVersion(client *dynamodb.Client, versionTablename string, version string return err } - -func setRepoMeta(client *dynamodb.Client, repoMetadataTableName string, repoMeta mTypes.RepoMetadata) error { - repoAttributeValue, err := attributevalue.Marshal(repoMeta) - if err != nil { - return err - } - - _, err = client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{ - ExpressionAttributeNames: map[string]string{ - "#RM": "RepoMetadata", - }, - ExpressionAttributeValues: map[string]types.AttributeValue{ - ":RepoMetadata": repoAttributeValue, - }, - Key: map[string]types.AttributeValue{ - "RepoName": &types.AttributeValueMemberS{ - Value: repoMeta.Name, - }, - }, - TableName: aws.String(repoMetadataTableName), - UpdateExpression: aws.String("SET #RM = :RepoMetadata"), - }) - - return err -} - -func TestProto(t *testing.T) { - Convey("Proto", t, func() { - dynamoDBDriverParams := mdynamodb.DBDriverParameters{ - Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), - RepoMetaTablename: "RepoMetadataTable", - ManifestDataTablename: "ManifestDataTable", - IndexDataTablename: "IndexDataTable", - UserDataTablename: "UserDataTable", - APIKeyTablename: "ApiKeyTable", - VersionTablename: "Version", - Region: "us-east-2", - } - - client, err := mdynamodb.GetDynamoClient(dynamoDBDriverParams) - So(err, ShouldBeNil) - - log := log.NewLogger("debug", "") - - metaDB, err := meta.Create("dynamodb", client, dynamoDBDriverParams, log) - So(metaDB, ShouldNotBeNil) - So(err, ShouldBeNil) - - metaDB.ProtoSetRepoMeta("repo", mTypes.RepoMetadata2{ - Name: "repo", - Tags: map[string]mTypes.Descriptor{"tag": {Digest: "d", MediaType: "m"}}, - Statistics: map[string]mTypes.DescriptorStatistics{"d": {DownloadCount: 2}}, - Signatures: map[string]mTypes.ManifestSignatures{"d": {}}, - Referrers: map[string][]mTypes.ReferrerInfo{"d": {{Digest: "d"}}}, - Stars: 20, - }) - - repoMeta, _ := metaDB.ProtoGetRepoMeta("repo") - fmt.Println(repoMeta) - - img1 := CreateRandomImage() - imageData1 := mTypes.ImageData2{ - MediaType: ispec.MediaTypeImageManifest, - Digest: img1.Digest(), - Size: img1.ManifestDescriptor.Size, - Index: nil, - Manifests: []mTypes.ManifestData2{ - { - Size: img1.ManifestDescriptor.Size, - Digest: img1.Digest(), - Manifest: img1.Manifest, - ConfigContent: img1.Config, - }, - }, - } - - img2 := CreateRandomImage() - imageData2 := mTypes.ImageData2{ - MediaType: ispec.MediaTypeImageManifest, - Digest: img2.Digest(), - Size: img2.ManifestDescriptor.Size, - Index: nil, - Manifests: []mTypes.ManifestData2{ - { - Size: img2.ManifestDescriptor.Size, - Digest: img2.Digest(), - Manifest: img2.Manifest, - ConfigContent: img2.Config, - }, - }, - } - - err = metaDB.ProtoSetImageData(imageData1.Digest, imageData1) - So(err, ShouldBeNil) - - err = metaDB.ProtoSetImageData(imageData2.Digest, imageData2) - So(err, ShouldBeNil) - - imgData, err := metaDB.ProtoGetImageData(imageData1.Digest) - So(err, ShouldBeNil) - _ = imgData - - x, _ := metaDB.ProtoFilterImageData(context.Background(), []string{ - imageData1.Digest.String(), imageData2.Digest.String(), - }) - _ = x - }) -} diff --git a/pkg/meta/dynamodb/parameters.go b/pkg/meta/dynamodb/parameters.go index cd73f2be14..573df3df5d 100644 --- a/pkg/meta/dynamodb/parameters.go +++ b/pkg/meta/dynamodb/parameters.go @@ -9,7 +9,7 @@ import ( ) type DBDriverParameters struct { - Endpoint, Region, RepoMetaTablename, ManifestDataTablename, IndexDataTablename, + Endpoint, Region, RepoMetaTablename, RepoBlobsInfoTablename, ImageDataTablename, UserDataTablename, APIKeyTablename, VersionTablename string } diff --git a/pkg/meta/hooks.go b/pkg/meta/hooks.go index 21b3bb9736..74d6d9380c 100644 --- a/pkg/meta/hooks.go +++ b/pkg/meta/hooks.go @@ -2,10 +2,10 @@ package meta import ( godigest "github.com/opencontainers/go-digest" + v1 "github.com/opencontainers/image-spec/specs-go/v1" zcommon "zotregistry.io/zot/pkg/common" "zotregistry.io/zot/pkg/log" - "zotregistry.io/zot/pkg/meta/common" mTypes "zotregistry.io/zot/pkg/meta/types" "zotregistry.io/zot/pkg/storage" ) @@ -22,78 +22,9 @@ func OnUpdateManifest(repo, reference, mediaType string, digest godigest.Digest, imgStore := storeController.GetImageStore(repo) - // check if image is a signature - isSignature, signatureType, signedManifestDigest, err := storage.CheckIsImageSignature(repo, body, reference) + err := SetImageMetaFromInput(repo, reference, mediaType, digest, body, + imgStore, metaDB, log) if err != nil { - log.Error().Err(err).Msg("can't check if image is a signature or not") - - if err := imgStore.DeleteImageManifest(repo, reference, false); err != nil { - log.Error().Err(err).Str("manifest", reference).Str("repository", repo).Msg("couldn't remove image manifest in repo") - - return err - } - - return err - } - - metadataSuccessfullySet := true - - if isSignature { - layersInfo, errGetLayers := GetSignatureLayersInfo(repo, reference, digest.String(), signatureType, body, - imgStore, log) - if errGetLayers != nil { - metadataSuccessfullySet = false - err = errGetLayers - } else { - err = metaDB.AddManifestSignature(repo, signedManifestDigest, mTypes.SignatureMetadata{ - SignatureType: signatureType, - SignatureDigest: digest.String(), - LayersInfo: layersInfo, - }) - if err != nil { - log.Error().Err(err).Msg("metadb: error while putting repo meta") - metadataSuccessfullySet = false - } else { - err = metaDB.UpdateSignaturesValidity(repo, signedManifestDigest) - if err != nil { - log.Error().Err(err).Str("repository", repo).Str("reference", reference).Str("digest", - signedManifestDigest.String()).Msg("metadb: failed verify signatures validity for signed image") - metadataSuccessfullySet = false - } - } - // Proto - err = metaDB.ProtoAddManifestSignature(repo, signedManifestDigest, mTypes.SignatureMetadata{ - SignatureType: signatureType, - SignatureDigest: digest.String(), - LayersInfo: layersInfo, - }) - if err != nil { - log.Error().Err(err).Msg("metadb: error while putting repo meta") - metadataSuccessfullySet = false - } else { - err = metaDB.ProtoUpdateSignaturesValidity(repo, signedManifestDigest) - if err != nil { - log.Error().Err(err).Str("repository", repo).Str("reference", reference).Str("digest", - signedManifestDigest.String()).Msg("metadb: failed verify signatures validity for signed image") - metadataSuccessfullySet = false - } - } - } - } else { - err = SetImageMetaFromInput(repo, reference, mediaType, digest, body, - imgStore, metaDB, log) - if err != nil { - metadataSuccessfullySet = false - } - - err = ProtoSetImageMetaFromInput(repo, reference, mediaType, digest, body, - imgStore, metaDB, log) - if err != nil { - metadataSuccessfullySet = false - } - } - - if !metadataSuccessfullySet { log.Info().Str("tag", reference).Str("repository", repo).Msg("uploading image meta was unsuccessful for tag in repo") if err := imgStore.DeleteImageManifest(repo, reference, false); err != nil { @@ -140,15 +71,6 @@ func OnDeleteManifest(repo, reference, mediaType string, digest godigest.Digest, log.Error().Err(err).Msg("metadb: can't check if image is a signature or not") manageRepoMetaSuccessfully = false } - - err = metaDB.ProtoDeleteSignature(repo, signedManifestDigest, mTypes.SignatureMetadata{ - SignatureDigest: digest.String(), - SignatureType: signatureType, - }) - if err != nil { - log.Error().Err(err).Msg("metadb: can't check if image is a signature or not") - manageRepoMetaSuccessfully = false - } } else { err = metaDB.RemoveRepoReference(repo, reference, digest) if err != nil { @@ -162,28 +84,6 @@ func OnDeleteManifest(repo, reference, mediaType string, digest godigest.Digest, manageRepoMetaSuccessfully = false } - - err = metaDB.ProtoDeleteRepoTag(repo, reference) - if err != nil { - log.Info().Msg("metadb: restoring image store") - - // restore image store - _, _, err := imgStore.PutImageManifest(repo, reference, mediaType, manifestBlob) - if err != nil { - log.Error().Err(err).Msg("metadb: error while restoring image store, database is not consistent") - } - - manageRepoMetaSuccessfully = false - } - - if referredDigest, hasSubject := common.GetReferredSubject(manifestBlob); hasSubject { - err := metaDB.DeleteReferrer(repo, referredDigest, digest) - if err != nil { - log.Error().Err(err).Msg("metadb: error while deleting referrer") - - return err - } - } } if !manageRepoMetaSuccessfully { @@ -197,7 +97,7 @@ func OnDeleteManifest(repo, reference, mediaType string, digest godigest.Digest, } // OnDeleteManifest is called when a manifest is downloaded. It increments the download couter on that manifest. -func OnGetManifest(name, reference string, body []byte, +func OnGetManifest(name, reference, mediaType string, body []byte, storeController storage.StoreController, metaDB mTypes.MetaDB, log log.Logger, ) error { // check if image is a signature @@ -208,14 +108,20 @@ func OnGetManifest(name, reference string, body []byte, return err } - if !isSignature && !zcommon.IsReferrersTag(reference) { - err := metaDB.IncrementImageDownloads(name, reference) - if err != nil { - log.Error().Err(err).Str("repository", name).Str("reference", reference). - Msg("unexpected error for image") + if isSignature || zcommon.IsReferrersTag(reference) { + return nil + } - return err - } + if !(mediaType == v1.MediaTypeImageManifest || mediaType == v1.MediaTypeImageIndex) { + return nil + } + + err = metaDB.IncrementImageDownloads(name, reference) + if err != nil { + log.Error().Err(err).Str("repository", name).Str("reference", reference). + Msg("unexpected error for image") + + return err } return nil diff --git a/pkg/meta/hooks_test.go b/pkg/meta/hooks_test.go index 5f697905c6..bfea45e40e 100644 --- a/pkg/meta/hooks_test.go +++ b/pkg/meta/hooks_test.go @@ -1,24 +1,19 @@ package meta_test import ( - "encoding/json" + "context" "errors" "testing" - notreg "github.com/notaryproject/notation-go/registry" - 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" "zotregistry.io/zot/pkg/meta" "zotregistry.io/zot/pkg/meta/boltdb" - mTypes "zotregistry.io/zot/pkg/meta/types" "zotregistry.io/zot/pkg/storage" "zotregistry.io/zot/pkg/storage/local" - "zotregistry.io/zot/pkg/test/deprecated" . "zotregistry.io/zot/pkg/test/image-utils" "zotregistry.io/zot/pkg/test/mocks" ) @@ -42,259 +37,38 @@ func TestOnUpdateManifest(t *testing.T) { metaDB, err := boltdb.New(boltDriver, log) So(err, ShouldBeNil) - config, layers, manifest, err := deprecated.GetRandomImageComponents(100) //nolint:staticcheck - So(err, ShouldBeNil) - - err = WriteImageToFileSystem( - Image{ - Config: config, Manifest: manifest, Layers: layers, - }, "repo", "tag1", storeController) - So(err, ShouldBeNil) + image := CreateDefaultImage() - manifestBlob, err := json.Marshal(manifest) + err = WriteImageToFileSystem(CreateDefaultImage(), "repo", "tag1", storeController) So(err, ShouldBeNil) - digest := godigest.FromBytes(manifestBlob) - - err = meta.OnUpdateManifest("repo", "tag1", "", digest, manifestBlob, storeController, metaDB, log) + err = meta.OnUpdateManifest("repo", "tag1", ispec.MediaTypeImageManifest, image.Digest(), + image.ManifestDescriptor.Data, storeController, metaDB, log) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta("repo") + repoMeta, err := metaDB.GetRepoMeta(context.Background(), "repo") So(err, ShouldBeNil) So(repoMeta.Tags, ShouldContainKey, "tag1") }) - - Convey("metadataSuccessfullySet is false", t, func() { - rootDir := t.TempDir() - storeController := storage.StoreController{} - log := log.NewLogger("debug", "") - metrics := monitoring.NewMetricsServer(false, log) - storeController.DefaultStore = local.NewImageStore(rootDir, true, true, log, metrics, nil, nil) - - metaDB := mocks.MetaDBMock{ - SetManifestDataFn: func(manifestDigest godigest.Digest, mm mTypes.ManifestData) error { - return ErrTestError - }, - } - - err := meta.OnUpdateManifest("repo", "tag1", ispec.MediaTypeImageManifest, "digest", - []byte("{}"), storeController, metaDB, log) - So(err, ShouldNotBeNil) - }) } func TestUpdateErrors(t *testing.T) { Convey("Update operations", t, func() { - Convey("On UpdateManifest", func() { - imageStore := mocks.MockedImageStore{} - storeController := storage.StoreController{DefaultStore: &imageStore} - metaDB := mocks.MetaDBMock{} - 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, - storeController, metaDB, log) - So(err, ShouldNotBeNil) - }) - - Convey("IsReferrersTag true", func() { - err := meta.OnUpdateManifest("repo", "sha256-123", "digest", "media", []byte("bad"), - storeController, metaDB, log) - So(err, ShouldBeNil) - }) - - 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, - storeController, metaDB, log) - 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 - } - - metaDB.UpdateSignaturesValidityFn = func(repo string, manifestDigest godigest.Digest) error { - return ErrTestError - } - - err = meta.OnUpdateManifest("repo", "tag1", "", "digest", notationManifestBlob, - storeController, metaDB, log) - So(err, ShouldNotBeNil) - }) - }) - - Convey("On DeleteManifest", func() { - imageStore := mocks.MockedImageStore{} - storeController := storage.StoreController{DefaultStore: &imageStore} - metaDB := mocks.MetaDBMock{} - 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 - } - - err := meta.OnDeleteManifest("repo", "tag1", "digest", "media", badManifestBlob, - storeController, metaDB, log) - So(err, ShouldNotBeNil) - }) - - Convey("IsReferrersTag true", func() { - err := meta.OnDeleteManifest("repo", "sha256-123", "digest", "media", []byte("bad"), - storeController, metaDB, log) - So(err, ShouldBeNil) - }) - - Convey("DeleteReferrers errors", func() { - metaDB.DeleteReferrerFn = func(repo string, referredDigest, referrerDigest godigest.Digest) error { - return ErrTestError - } - - err := meta.OnDeleteManifest("repo", "tag1", "digest", "media", - []byte(`{"subject": {"digest": "dig"}}`), - storeController, metaDB, log) - So(err, ShouldNotBeNil) - }) - }) - - Convey("On GetManifest", func() { - imageStore := mocks.MockedImageStore{} - storeController := storage.StoreController{DefaultStore: &imageStore} - metaDB := mocks.MetaDBMock{} - 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 - } - - err := meta.OnGetManifest("repo", "tag1", badManifestBlob, - storeController, metaDB, log) - So(err, ShouldNotBeNil) - }) - }) - - Convey("SetImageMetaFromInput", func() { - imageStore := mocks.MockedImageStore{} - metaDB := mocks.MetaDBMock{} - log := log.NewLogger("debug", "") - - err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageManifest, "digest", - []byte("BadManifestBlob"), imageStore, metaDB, log) - 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 - } + imageStore := mocks.MockedImageStore{} + storeController := storage.StoreController{DefaultStore: &imageStore} + metaDB := mocks.MetaDBMock{} + log := log.NewLogger("debug", "") - err = meta.SetImageMetaFromInput("repo", string(godigest.FromString("reference")), "", "digest", - manifestBlob, imageStore, metaDB, log) + Convey("IsReferrersTag true update", func() { + err := meta.OnUpdateManifest("repo", "sha256-123", "digest", "media", []byte("bad"), + storeController, metaDB, log) So(err, ShouldBeNil) }) - - Convey("SetImageMetaFromInput SetData errors", func() { - imageStore := mocks.MockedImageStore{} - log := log.NewLogger("debug", "") - - metaDB := mocks.MetaDBMock{ - SetManifestDataFn: func(manifestDigest godigest.Digest, mm mTypes.ManifestData) error { - return ErrTestError - }, - } - err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageManifest, "digest", - []byte("{}"), imageStore, metaDB, log) - So(err, ShouldNotBeNil) - }) - - Convey("SetImageMetaFromInput SetIndexData errors", func() { - imageStore := mocks.MockedImageStore{} - log := log.NewLogger("debug", "") - - metaDB := mocks.MetaDBMock{ - SetIndexDataFn: func(digest godigest.Digest, indexData mTypes.IndexData) error { - return ErrTestError - }, - } - err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageIndex, "digest", - []byte("{}"), imageStore, metaDB, log) - So(err, ShouldNotBeNil) - }) - - Convey("SetImageMetaFromInput SetReferrer errors", func() { - imageStore := mocks.MockedImageStore{ - GetBlobContentFn: func(repo string, digest godigest.Digest) ([]byte, error) { - return []byte("{}"), nil - }, - } - log := log.NewLogger("debug", "") - - metaDB := mocks.MetaDBMock{ - SetReferrerFn: func(repo string, referredDigest godigest.Digest, referrer mTypes.ReferrerInfo) error { - return ErrTestError - }, - } - - err := meta.SetImageMetaFromInput("repo", "ref", ispec.MediaTypeImageManifest, "digest", - []byte(`{"subject": {"digest": "subjDigest"}}`), imageStore, metaDB, log) - So(err, ShouldNotBeNil) + Convey("IsReferrersTag true delete", func() { + err := meta.OnDeleteManifest("repo", "sha256-123", "digest", "media", []byte("bad"), + storeController, metaDB, log) + So(err, ShouldBeNil) }) }) } diff --git a/pkg/meta/meta.go b/pkg/meta/meta.go index 7d47c78e36..ab1619352c 100644 --- a/pkg/meta/meta.go +++ b/pkg/meta/meta.go @@ -81,10 +81,10 @@ func getDynamoParams(cacheDriverConfig map[string]interface{}, log log.Logger) m repoMetaTablename, ok := toStringIfOk(cacheDriverConfig, "repometatablename", log) allParametersOk = allParametersOk && ok - manifestDataTablename, ok := toStringIfOk(cacheDriverConfig, "manifestdatatablename", log) + repoBlobsInfoTablename, ok := toStringIfOk(cacheDriverConfig, "repoblobsinfotablename", log) allParametersOk = allParametersOk && ok - indexDataTablename, ok := toStringIfOk(cacheDriverConfig, "indexdatatablename", log) + imageDataTablename, ok := toStringIfOk(cacheDriverConfig, "imagedatatablename", log) allParametersOk = allParametersOk && ok apiKeyTablename, ok := toStringIfOk(cacheDriverConfig, "apikeytablename", log) @@ -101,14 +101,14 @@ func getDynamoParams(cacheDriverConfig map[string]interface{}, log log.Logger) m } return mdynamodb.DBDriverParameters{ - Endpoint: endpoint, - Region: region, - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - VersionTablename: versionTablename, + Endpoint: endpoint, + Region: region, + RepoMetaTablename: repoMetaTablename, + RepoBlobsInfoTablename: repoBlobsInfoTablename, + ImageDataTablename: imageDataTablename, + UserDataTablename: userDataTablename, + APIKeyTablename: apiKeyTablename, + VersionTablename: versionTablename, } } diff --git a/pkg/meta/meta_test.go b/pkg/meta/meta_test.go index 7e9996df49..67ed97e4e9 100644 --- a/pkg/meta/meta_test.go +++ b/pkg/meta/meta_test.go @@ -5,9 +5,7 @@ package meta_test import ( "context" - "encoding/json" "fmt" - "math/rand" "os" "path" "testing" @@ -19,11 +17,12 @@ import ( "github.com/notaryproject/notation-go" "github.com/notaryproject/notation-go/signer" godigest "github.com/opencontainers/go-digest" - "github.com/opencontainers/image-spec/specs-go" ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" + zcommon "zotregistry.io/zot/pkg/common" "zotregistry.io/zot/pkg/extensions/imagetrust" + "zotregistry.io/zot/pkg/extensions/search/convert" "zotregistry.io/zot/pkg/log" "zotregistry.io/zot/pkg/meta" "zotregistry.io/zot/pkg/meta/boltdb" @@ -31,10 +30,9 @@ import ( mdynamodb "zotregistry.io/zot/pkg/meta/dynamodb" mTypes "zotregistry.io/zot/pkg/meta/types" reqCtx "zotregistry.io/zot/pkg/requestcontext" - test "zotregistry.io/zot/pkg/test/common" - "zotregistry.io/zot/pkg/test/deprecated" + tCommon "zotregistry.io/zot/pkg/test/common" . "zotregistry.io/zot/pkg/test/image-utils" - signature "zotregistry.io/zot/pkg/test/signature" + "zotregistry.io/zot/pkg/test/signature" tskip "zotregistry.io/zot/pkg/test/skip" ) @@ -45,6 +43,8 @@ const ( ARM = "arm64" ) +func getManifestDigest(md mTypes.ManifestData) string { return md.Digest.String() } + func TestBoltDB(t *testing.T) { Convey("BoltDB creation", t, func() { boltDBParams := boltdb.DBParameters{RootDir: t.TempDir()} @@ -106,26 +106,24 @@ func TestDynamoDBWrapper(t *testing.T) { } repoMetaTablename := "RepoMetadataTable" + uuid.String() - manifestDataTablename := "ManifestDataTable" + uuid.String() versionTablename := "Version" + uuid.String() - indexDataTablename := "IndexDataTable" + uuid.String() userDataTablename := "UserDataTable" + uuid.String() apiKeyTablename := "ApiKeyTable" + uuid.String() + imageDataTablename := "ImageData" + uuid.String() + repoBlobsTablename := "RepoBlobs" + uuid.String() Convey("DynamoDB Wrapper", t, func() { dynamoDBDriverParams := mdynamodb.DBDriverParameters{ - Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), - RepoMetaTablename: repoMetaTablename, - ManifestDataTablename: manifestDataTablename, - IndexDataTablename: indexDataTablename, - VersionTablename: versionTablename, - UserDataTablename: userDataTablename, - APIKeyTablename: apiKeyTablename, - Region: "us-east-2", + Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), + RepoMetaTablename: repoMetaTablename, + RepoBlobsInfoTablename: repoBlobsTablename, + ImageDataTablename: imageDataTablename, + VersionTablename: versionTablename, + UserDataTablename: userDataTablename, + APIKeyTablename: apiKeyTablename, + Region: "us-east-2", } - t.Logf("using dynamo driver options: %v", dynamoDBDriverParams) - dynamoClient, err := mdynamodb.GetDynamoClient(dynamoDBDriverParams) So(err, ShouldBeNil) @@ -141,14 +139,22 @@ func TestDynamoDBWrapper(t *testing.T) { dynamoDriver.SetImageTrustStore(imgTrustStore) resetDynamoDBTables := func() error { - err := dynamoDriver.ResetRepoMetaTable() + err := dynamoDriver.ResetTable(dynamoDriver.RepoMetaTablename) if err != nil { return err } - // Note: Tests are very slow if we reset the UserData table every new convey. We'll reset it as needed + err = dynamoDriver.ResetTable(dynamoDriver.ImageDataTablename) + if err != nil { + return err + } + + err = dynamoDriver.ResetTable(dynamoDriver.RepoBlobsTablename) + if err != nil { + return err + } - err = dynamoDriver.ResetManifestDataTable() + // Note: Tests are very slow if we reset the UserData table every new convey. We'll reset it as needed return err } @@ -158,6 +164,8 @@ func TestDynamoDBWrapper(t *testing.T) { } func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func() error) { //nolint: thelper + ctx := context.Background() + Convey("Test MetaDB Interface implementation", func() { for _, prepFunc := range preparationFuncs { err := prepFunc() @@ -453,242 +461,219 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func }) }) - Convey("Test SetManifestData and GetManifestData", func() { - configBlob, manifestBlob, err := generateTestImage() - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob) + Convey("Test Setting an image by tag and retrieving data", func() { + imgData := CreateDefaultImage().AsImageData() - err = metaDB.SetManifestData(manifestDigest, mTypes.ManifestData{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - }) + err := metaDB.SetImageData(imgData.Digest, imgData) So(err, ShouldBeNil) - mm, err := metaDB.GetManifestData(manifestDigest) + retrievedImgData, err := metaDB.GetImageData(imgData.Digest) So(err, ShouldBeNil) - So(mm.ManifestBlob, ShouldResemble, manifestBlob) - So(mm.ConfigBlob, ShouldResemble, configBlob) - }) + So(imgData, ShouldResemble, retrievedImgData) - Convey("Test GetManifestMeta fails", func() { - _, err := metaDB.GetManifestMeta("repo", "bad digest") - So(err, ShouldNotBeNil) - }) - - Convey("Test SetManifestMeta", func() { - Convey("RepoMeta not found", func() { - var ( - manifestDigest = godigest.FromString("dig") - manifestBlob = []byte("manifestBlob") - configBlob = []byte("configBlob") - - signatures = mTypes.ManifestSignatures{ - "digest1": []mTypes.SignatureInfo{ - { - SignatureManifestDigest: "signatureDigest", - LayersInfo: []mTypes.LayerInfo{ - { - LayerDigest: "layerDigest", - LayerContent: []byte("layerContent"), - }, - }, - }, - }, - } - ) + imgMulti := CreateRandomMultiarch() - err := metaDB.SetManifestMeta("repo", manifestDigest, mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - DownloadCount: 10, - Signatures: signatures, - }) + for i := range imgMulti.Images { + err = metaDB.SetImageData(imgMulti.Images[i].Digest(), imgMulti.Images[i].AsImageData()) So(err, ShouldBeNil) + } - manifestMeta, err := metaDB.GetManifestMeta("repo", manifestDigest) - So(err, ShouldBeNil) + err = metaDB.SetImageData(imgMulti.Digest(), imgMulti.AsImageData()) + So(err, ShouldBeNil) - So(manifestMeta.ManifestBlob, ShouldResemble, manifestBlob) - So(manifestMeta.ConfigBlob, ShouldResemble, configBlob) - So(manifestMeta.DownloadCount, ShouldEqual, 10) - So(manifestMeta.Signatures, ShouldResemble, signatures) - }) + retrievedImgMultiData, err := metaDB.GetImageData(imgMulti.Digest()) + So(err, ShouldBeNil) + So(imgMulti.AsImageData(), ShouldEqual, retrievedImgMultiData) }) Convey("Test SetRepoReference", func() { - // test behaviours var ( - repo1 = "repo1" - repo2 = "repo2" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - - tag2 = "0.0.2" - manifestDigest2 = godigest.FromString("fake-manifes2") + repo1 = "repo1" + repo2 = "repo2" + tag1 = "0.0.1" + tag2 = "0.0.2" ) - Convey("Setting a good repo", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + img1 := CreateImageWith().RandomLayers(2, 10).RandomConfig(). + Annotations(map[string]string{"test": "annotation"}).Build() + imgData1 := img1.AsImageData() + img1Size := img1.ConfigDescriptor.Size + img1.ManifestDescriptor.Size + 2*10 + + img2 := CreateImageWith().LayerBlobs(img1.Layers).RandomConfig(). + Annotations(map[string]string{"test": "annotation"}).Build() + imgData2 := img2.AsImageData() + img2Size := img2.ConfigDescriptor.Size + img2.ManifestDescriptor.Size + 2*10 + + multiImages := []Image{ + CreateImageWith().RandomLayers(2, 10). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "multi-os1", Architecture: "multi-arch1"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor1"}). + Build(), + CreateImageWith().RandomLayers(2, 10). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "multi-os2", Architecture: "multi-arch2"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor2"}). + Build(), + } - repoMeta, err := metaDB.GetRepoMeta(repo1) - So(err, ShouldBeNil) - So(repoMeta.Name, ShouldResemble, repo1) - So(repoMeta.Tags[tag1].Digest, ShouldEqual, manifestDigest1.String()) + imgMulti := CreateMultiarchWith(). + Images(multiImages). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor1"}).Build() - err = metaDB.SetRepoMeta(repo2, mTypes.RepoMetadata{Tags: map[string]mTypes.Descriptor{ - tag2: { - Digest: manifestDigest2.String(), - }, - }}) + Convey("Setting a good repo", func() { + err := metaDB.SetRepoReference(repo1, tag1, imgData1) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo2) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Name, ShouldResemble, repo2) - So(repoMeta.Tags[tag2].Digest, ShouldEqual, manifestDigest2.String()) + So(repoMeta.Name, ShouldResemble, repo1) + So(repoMeta.Tags[tag1].Digest, ShouldEqual, img1.DigestStr()) }) - Convey("Setting a good repo using a digest", func() { - _, err := metaDB.GetRepoMeta(repo1) + Convey("Setting an index with it's manifests", func() { + _, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldNotBeNil) - digest := godigest.FromString("digest") - err = metaDB.SetRepoReference(repo1, digest.String(), digest, - ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + for i := range imgMulti.Images { + err := metaDB.SetRepoReference(repo1, imgMulti.Images[i].DigestStr(), + imgMulti.Images[i].AsImageData()) + So(err, ShouldBeNil) + } - repoMeta, err := metaDB.GetRepoMeta(repo1) + err = metaDB.SetRepoReference(repo1, tag1, imgMulti.AsImageData()) So(err, ShouldBeNil) - So(repoMeta.Name, ShouldResemble, repo1) - }) - Convey("Set multiple tags for repo", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag2, manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + image1TotalSize := multiImages[0].ManifestDescriptor.Size + multiImages[0].ConfigDescriptor.Size + 2*10 + image2TotalSize := multiImages[1].ManifestDescriptor.Size + multiImages[1].ConfigDescriptor.Size + 2*10 + indexTotalSize := image1TotalSize + image2TotalSize + imgMulti.IndexDescriptor.Size - repoMeta, err := metaDB.GetRepoMeta(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Tags[tag1].Digest, ShouldEqual, manifestDigest1.String()) - So(repoMeta.Tags[tag2].Digest, ShouldEqual, manifestDigest2.String()) + So(repoMeta.Name, ShouldResemble, repo1) + So(repoMeta.Platforms, ShouldContain, ispec.Platform{OS: "multi-os1", Architecture: "multi-arch1"}) + So(repoMeta.Platforms, ShouldContain, ispec.Platform{OS: "multi-os2", Architecture: "multi-arch2"}) + So(repoMeta.Vendors, ShouldContain, "vendor1") + So(repoMeta.Vendors, ShouldContain, "vendor2") + So(repoMeta.Size, ShouldEqual, indexTotalSize) }) Convey("Set multiple repos", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, imgData1) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, tag2, manifestDigest2, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo2, tag1, imgData2) So(err, ShouldBeNil) - repoMeta1, err := metaDB.GetRepoMeta(repo1) + repoMeta1, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - repoMeta2, err := metaDB.GetRepoMeta(repo2) + repoMeta2, err := metaDB.GetRepoMeta(ctx, repo2) So(err, ShouldBeNil) - So(repoMeta1.Tags[tag1].Digest, ShouldResemble, manifestDigest1.String()) - So(repoMeta2.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) + So(repoMeta1.Tags[tag1].Digest, ShouldResemble, imgData1.Digest.String()) + So(repoMeta2.Tags[tag1].Digest, ShouldResemble, imgData2.Digest.String()) + So(repoMeta1.Size, ShouldEqual, img1Size) + So(repoMeta2.Size, ShouldEqual, img2Size) }) - Convey("Setting a repo with invalid fields", func() { - Convey("Repo name is not valid", func() { - err := metaDB.SetRepoReference("", tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldNotBeNil) - }) + Convey("Check repo blobs info for manifest image", func() { + image1 := CreateImageWith().RandomLayers(2, 10). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "os1", Architecture: "arch1"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor1"}). + Build() + imageData1 := image1.AsImageData() - Convey("Tag is not valid", func() { - err := metaDB.SetRepoReference(repo1, "", manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldNotBeNil) - }) - - Convey("Manifest Digest is not valid", func() { - err := metaDB.SetRepoReference(repo1, tag1, "", ispec.MediaTypeImageManifest) - So(err, ShouldNotBeNil) - }) - }) - }) - - Convey("Test GetRepoMeta", func() { - var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") + layersSize := int64(2 * 10) + image1Size := imageData1.Manifests[0].Size + imageData1.Manifests[0].Manifest.Config.Size + layersSize - repo2 = "repo2" - tag2 = "0.0.2" - manifestDigest2 = godigest.FromString("fake-manifest2") + err := metaDB.SetRepoReference(repo1, tag1, imageData1) + So(err, ShouldBeNil) - InexistentRepo = "InexistentRepo" - ) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) + So(err, ShouldBeNil) + So(repoMeta.Vendors, ShouldContain, "vendor1") + So(repoMeta.Platforms, ShouldContain, ispec.Platform{OS: "os1", Architecture: "arch1"}) + So(repoMeta.Size, ShouldEqual, image1Size) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + image2 := CreateImageWith(). + LayerBlobs(image1.Layers). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "os2", Architecture: "arch2"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor2"}). + Build() + imageData2 := image2.AsImageData() - err = metaDB.SetRepoReference(repo2, tag2, manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + // the layers are the same so we add them once + repoSize := image1Size + image2.ManifestDescriptor.Size + image2.ConfigDescriptor.Size - Convey("Get a existent repo", func() { - repoMeta1, err := metaDB.GetRepoMeta(repo1) + err = metaDB.SetRepoReference(repo1, tag2, imageData2) So(err, ShouldBeNil) - So(repoMeta1.Tags[tag1].Digest, ShouldResemble, manifestDigest1.String()) - repoMeta2, err := metaDB.GetRepoMeta(repo2) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta2.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) - }) - - Convey("Get a repo that doesn't exist", func() { - repoMeta, err := metaDB.GetRepoMeta(InexistentRepo) - So(err, ShouldNotBeNil) - So(repoMeta, ShouldBeZeroValue) + So(repoMeta.Vendors, ShouldContain, "vendor1") + So(repoMeta.Vendors, ShouldContain, "vendor2") + So(repoMeta.Platforms, ShouldContain, ispec.Platform{OS: "os1", Architecture: "arch1"}) + So(repoMeta.Platforms, ShouldContain, ispec.Platform{OS: "os2", Architecture: "arch2"}) + So(repoMeta.Size, ShouldEqual, repoSize) }) }) - Convey("Test RemoveRepoReference and DeleteRepoTag", func() { + Convey("Test RemoveRepoReference", func() { var ( - repo = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - tag2 = "0.0.2" - manifestDigest2 = godigest.FromString("fake-manifest2") + repo = "repo1" + tag1 = "0.0.1" + tag2 = "0.0.2" ) - err := metaDB.SetRepoReference(repo, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + layersSize := int64(2 * 10) + + image1 := CreateImageWith(). + RandomLayers(2, 10). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "os1", Architecture: "arch1"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor1"}). + Build() + imageData1 := image1.AsImageData() + image1Size := imageData1.Manifests[0].Size + imageData1.Manifests[0].Manifest.Config.Size + layersSize + + image2 := CreateImageWith(). + LayerBlobs(image1.Layers). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "os2", Architecture: "arch2"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor2", "annotation": "test"}). + Build() + imageData2 := image2.AsImageData() + image2Size := imageData2.Manifests[0].Size + imageData2.Manifests[0].Manifest.Config.Size + layersSize + + totalRepoSize := image1Size + image2Size - layersSize + + err := metaDB.SetRepoReference(repo, tag1, imageData1) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo, tag2, manifestDigest2, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo, tag2, imageData2) So(err, ShouldBeNil) Convey("Delete reference from repo", func() { - _, err := metaDB.GetRepoMeta(repo) - So(err, ShouldBeNil) - - err = metaDB.RemoveRepoReference(repo, tag1, manifestDigest1) + err = metaDB.RemoveRepoReference(repo, tag1, imageData1.Digest) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo) So(err, ShouldBeNil) _, ok := repoMeta.Tags[tag1] So(ok, ShouldBeFalse) - So(repoMeta.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) + So(repoMeta.Size, ShouldEqual, image2Size) + So(repoMeta.Platforms, ShouldNotContain, ispec.Platform{OS: "os1", Architecture: "arch1"}) + So(repoMeta.Vendors, ShouldNotContain, "vendor1") }) - Convey("Delete a reference from repo", func() { - _, err := metaDB.GetRepoMeta(repo) + Convey("Delete a digest from repo", func() { + err = metaDB.RemoveRepoReference(repo, tag2, imageData2.Digest) So(err, ShouldBeNil) - // shouldn't do anything because there is tag1 pointing to it - err = metaDB.RemoveRepoReference(repo, manifestDigest1.String(), manifestDigest1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo) - So(err, ShouldBeNil) - - _, ok := repoMeta.Tags[tag1] + _, ok := repoMeta.Tags[tag2] So(ok, ShouldBeFalse) - So(repoMeta.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) + So(repoMeta.Size, ShouldEqual, image1Size) + So(repoMeta.Platforms, ShouldNotContain, ispec.Platform{OS: "os2", Architecture: "arch2"}) + So(repoMeta.Vendors, ShouldNotContain, "vendor2") }) Convey("Delete inexistent reference from repo", func() { @@ -696,11 +681,12 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func err := metaDB.RemoveRepoReference(repo, inexistentDigest.String(), inexistentDigest) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo) So(err, ShouldBeNil) - So(repoMeta.Tags[tag1].Digest, ShouldResemble, manifestDigest1.String()) - So(repoMeta.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) + So(repoMeta.Tags[tag1].Digest, ShouldResemble, imageData1.Digest.String()) + So(repoMeta.Tags[tag2].Digest, ShouldResemble, imageData2.Digest.String()) + So(repoMeta.Size, ShouldEqual, totalRepoSize) }) Convey("Delete reference from inexistent repo", func() { @@ -709,72 +695,48 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func err := metaDB.RemoveRepoReference("InexistentRepo", inexistentDigest.String(), inexistentDigest) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo) - So(err, ShouldBeNil) - - So(repoMeta.Tags[tag1].Digest, ShouldResemble, manifestDigest1.String()) - So(repoMeta.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) - }) - - Convey("Delete from repo a tag", func() { - _, err := metaDB.GetRepoMeta(repo) - So(err, ShouldBeNil) - - err = metaDB.DeleteRepoTag(repo, tag1) - So(err, ShouldBeNil) - - repoMeta, err := metaDB.GetRepoMeta(repo) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo) So(err, ShouldBeNil) - _, ok := repoMeta.Tags[tag1] - So(ok, ShouldBeFalse) - So(repoMeta.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) - }) - - Convey("Delete inexistent tag from repo", func() { - err := metaDB.DeleteRepoTag(repo, "InexistentTag") - So(err, ShouldBeNil) - - repoMeta, err := metaDB.GetRepoMeta(repo) - So(err, ShouldBeNil) - - So(repoMeta.Tags[tag1].Digest, ShouldResemble, manifestDigest1.String()) - So(repoMeta.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) - }) - - Convey("Delete tag from inexistent repo", func() { - err := metaDB.DeleteRepoTag("InexistentRepo", "InexistentTag") - So(err, ShouldBeNil) - - repoMeta, err := metaDB.GetRepoMeta(repo) - So(err, ShouldBeNil) - - So(repoMeta.Tags[tag1].Digest, ShouldResemble, manifestDigest1.String()) - So(repoMeta.Tags[tag2].Digest, ShouldResemble, manifestDigest2.String()) + So(repoMeta.Tags[tag1].Digest, ShouldResemble, imageData1.Digest.String()) + So(repoMeta.Tags[tag2].Digest, ShouldResemble, imageData2.Digest.String()) + So(repoMeta.Size, ShouldEqual, totalRepoSize) }) }) Convey("Test GetMultipleRepoMeta", func() { var ( - repo1 = "repo1" - repo2 = "repo2" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - tag2 = "0.0.2" - manifestDigest2 = godigest.FromString("fake-manifest2") + repo1 = "repo1" + repo2 = "repo2" + tag1 = "0.0.1" + tag2 = "0.0.2" ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + image1 := CreateImageWith(). + RandomLayers(2, 10). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "os1", Architecture: "arch1"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor1"}). + Build() + imageData1 := image1.AsImageData() + + image2 := CreateImageWith(). + LayerBlobs(image1.Layers). + ImageConfig(ispec.Image{Platform: ispec.Platform{OS: "os2", Architecture: "arch2"}}). + Annotations(map[string]string{ispec.AnnotationVendor: "vendor2"}). + Build() + imageData2 := image2.AsImageData() + + err := metaDB.SetRepoReference(repo1, tag1, imageData1) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag2, manifestDigest2, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, tag2, imageData2) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, tag2, manifestDigest2, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo2, tag2, imageData2) So(err, ShouldBeNil) - Convey("Get all Repometa", func() { - repoMetaSlice, err := metaDB.GetMultipleRepoMeta(context.TODO(), func(repoMeta mTypes.RepoMetadata) bool { + Convey("Get all RepoMeta", func() { + repoMetaSlice, err := metaDB.GetMultipleRepoMeta(context.TODO(), func(repoMeta mTypes.RepoMeta) bool { return true }) So(err, ShouldBeNil) @@ -782,7 +744,7 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func }) Convey("Get repo with a tag", func() { - repoMetaSlice, err := metaDB.GetMultipleRepoMeta(context.TODO(), func(repoMeta mTypes.RepoMetadata) bool { + repoMetaSlice, err := metaDB.GetMultipleRepoMeta(context.TODO(), func(repoMeta mTypes.RepoMeta) bool { for tag := range repoMeta.Tags { if tag == tag1 { return true @@ -793,113 +755,110 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func }) So(err, ShouldBeNil) So(len(repoMetaSlice), ShouldEqual, 1) - So(repoMetaSlice[0].Tags[tag1].Digest == manifestDigest1.String(), ShouldBeTrue) + So(repoMetaSlice[0].Tags[tag1].Digest == imageData1.Digest.String(), ShouldBeTrue) }) }) Convey("Test IncrementRepoStars", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") + repo1 = "repo1" + tag1 = "0.0.1" + imageData = CreateDefaultImage().AsImageData() ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, imageData) So(err, ShouldBeNil) err = metaDB.IncrementRepoStars(repo1) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) err = metaDB.IncrementRepoStars(repo1) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 2) + So(repoMeta.StarCount, ShouldEqual, 2) err = metaDB.IncrementRepoStars(repo1) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 3) + So(repoMeta.StarCount, ShouldEqual, 3) }) Convey("Test DecrementRepoStars", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") + repo1 = "repo1" + tag1 = "0.0.1" + imageData = CreateDefaultImage().AsImageData() ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, imageData) So(err, ShouldBeNil) err = metaDB.IncrementRepoStars(repo1) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) err = metaDB.DecrementRepoStars(repo1) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 0) + So(repoMeta.StarCount, ShouldEqual, 0) err = metaDB.DecrementRepoStars(repo1) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 0) + So(repoMeta.StarCount, ShouldEqual, 0) - _, err = metaDB.GetRepoMeta("badRepo") + _, err = metaDB.GetRepoMeta(ctx, "badRepo") So(err, ShouldNotBeNil) }) - Convey("Test GetRepoStars", func() { + Convey("Test Repo Stars", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") + repo1 = "repo1" + tag1 = "0.0.1" ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, CreateDefaultImage().AsImageData()) So(err, ShouldBeNil) err = metaDB.IncrementRepoStars(repo1) So(err, ShouldBeNil) - - stars, err := metaDB.GetRepoStars(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(stars, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) err = metaDB.IncrementRepoStars(repo1) So(err, ShouldBeNil) - err = metaDB.IncrementRepoStars(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) + So(repoMeta.StarCount, ShouldEqual, 2) - stars, err = metaDB.GetRepoStars(repo1) + err = metaDB.IncrementRepoStars(repo1) So(err, ShouldBeNil) - So(stars, ShouldEqual, 3) - - _, err = metaDB.GetRepoStars("badRepo") - So(err, ShouldNotBeNil) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) + So(err, ShouldBeNil) + So(repoMeta.StarCount, ShouldEqual, 3) }) Convey("Test repo stars for user", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - repo2 = "repo2" + repo1 = "repo1" + tag1 = "0.0.1" + repo2 = "repo2" ) userAc := reqCtx.NewUserAccessControl() @@ -910,7 +869,7 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func }) // "user1" - ctx1 := userAc.DeriveContext(context.Background()) + ctx1 := userAc.DeriveContext(ctx) userAc = reqCtx.NewUserAccessControl() userAc.SetUsername("user2") @@ -920,7 +879,7 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func }) // "user2" - ctx2 := userAc.DeriveContext(context.Background()) + ctx2 := userAc.DeriveContext(ctx) userAc = reqCtx.NewUserAccessControl() userAc.SetGlobPatterns("read", map[string]bool{ @@ -929,21 +888,13 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func }) // anonymous user - ctx3 := userAc.DeriveContext(context.Background()) - - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + ctx3 := userAc.DeriveContext(ctx) - err = metaDB.SetRepoReference(repo2, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, CreateDefaultImage().AsImageData()) So(err, ShouldBeNil) - starCount, err := metaDB.GetRepoStars(repo1) + err = metaDB.SetRepoReference(repo2, tag1, CreateDefaultImage().AsImageData()) So(err, ShouldBeNil) - So(starCount, ShouldEqual, 0) - - starCount, err = metaDB.GetRepoStars(repo2) - So(err, ShouldBeNil) - So(starCount, ShouldEqual, 0) repos, err := metaDB.GetStarredRepos(ctx1) So(err, ShouldBeNil) @@ -962,13 +913,9 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func So(err, ShouldBeNil) So(toggleState, ShouldEqual, mTypes.Added) - repoMeta, err := metaDB.GetRepoMeta(repo1) - So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 1) - - starCount, err = metaDB.GetRepoStars(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(starCount, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) repos, err = metaDB.GetStarredRepos(ctx1) So(err, ShouldBeNil) @@ -988,13 +935,9 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func So(err, ShouldBeNil) So(toggleState, ShouldEqual, mTypes.Added) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 2) - - starCount, err = metaDB.GetRepoStars(repo1) - So(err, ShouldBeNil) - So(starCount, ShouldEqual, 2) + So(repoMeta.StarCount, ShouldEqual, 2) repos, err = metaDB.GetStarredRepos(ctx1) So(err, ShouldBeNil) @@ -1015,13 +958,9 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func So(err, ShouldBeNil) So(toggleState, ShouldEqual, mTypes.Added) - repoMeta, err = metaDB.GetRepoMeta(repo2) - So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 1) - - starCount, err = metaDB.GetRepoStars(repo2) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo2) So(err, ShouldBeNil) - So(starCount, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) repos, err = metaDB.GetStarredRepos(ctx1) So(err, ShouldBeNil) @@ -1043,13 +982,9 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func So(err, ShouldBeNil) So(toggleState, ShouldEqual, mTypes.Removed) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 1) - - starCount, err = metaDB.GetRepoStars(repo1) - So(err, ShouldBeNil) - So(starCount, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) repos, err = metaDB.GetStarredRepos(ctx1) So(err, ShouldBeNil) @@ -1074,21 +1009,13 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func So(err, ShouldBeNil) So(toggleState, ShouldEqual, mTypes.Removed) - repoMeta, err = metaDB.GetRepoMeta(repo1) - So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 1) - - repoMeta, err = metaDB.GetRepoMeta(repo2) - So(err, ShouldBeNil) - So(repoMeta.Stars, ShouldEqual, 1) - - starCount, err = metaDB.GetRepoStars(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(starCount, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) - starCount, err = metaDB.GetRepoStars(repo2) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo2) So(err, ShouldBeNil) - So(starCount, ShouldEqual, 1) + So(repoMeta.StarCount, ShouldEqual, 1) repos, err = metaDB.GetStarredRepos(ctx1) So(err, ShouldBeNil) @@ -1104,15 +1031,11 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func So(err, ShouldBeNil) So(len(repos), ShouldEqual, 0) - // Anonyous user attempts to toggle a star + // Anonymous user attempts to toggle a star toggleState, err = metaDB.ToggleStarRepo(ctx3, repo1) So(err, ShouldNotBeNil) So(toggleState, ShouldEqual, mTypes.NotChanged) - starCount, err = metaDB.GetRepoStars(repo1) - So(err, ShouldBeNil) - So(starCount, ShouldEqual, 1) - repos, err = metaDB.GetStarredRepos(ctx3) So(err, ShouldBeNil) So(len(repos), ShouldEqual, 0) @@ -1122,10 +1045,6 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func So(err, ShouldBeNil) So(toggleState, ShouldEqual, mTypes.Removed) - starCount, err = metaDB.GetRepoStars(repo2) - So(err, ShouldBeNil) - So(starCount, ShouldEqual, 0) - repos, err = metaDB.GetStarredRepos(ctx3) So(err, ShouldBeNil) So(len(repos), ShouldEqual, 0) @@ -1133,10 +1052,10 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func Convey("Test repo bookmarks for user", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - repo2 = "repo2" + repo1 = "repo1" + tag1 = "0.0.1" + repo2 = "repo2" + image1 = CreateRandomImage().AsImageData() ) userAc := reqCtx.NewUserAccessControl() @@ -1168,10 +1087,10 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func // anonymous user ctx3 := userAc.DeriveContext(context.Background()) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, image1) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo2, tag1, image1) So(err, ShouldBeNil) repos, err := metaDB.GetBookmarkedRepos(ctx1) @@ -1277,122 +1196,98 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func Convey("Test IncrementImageDownloads", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" + repo1 = "repo1" + tag1 = "0.0.1" + image1 = CreateRandomImage().AsImageData() ) - configBlob, manifestBlob, err := generateTestImage() - So(err, ShouldBeNil) - - manifestDigest := godigest.FromBytes(manifestBlob) - - err = metaDB.SetRepoReference(repo1, tag1, manifestDigest, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta(repo1, manifestDigest, mTypes.ManifestMetadata{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - }) + err := metaDB.SetRepoReference(repo1, tag1, image1) So(err, ShouldBeNil) err = metaDB.IncrementImageDownloads(repo1, tag1) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Statistics[manifestDigest.String()].DownloadCount, ShouldEqual, 1) + So(repoMeta.Statistics[image1.Digest.String()].DownloadCount, ShouldEqual, 1) err = metaDB.IncrementImageDownloads(repo1, tag1) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Statistics[manifestDigest.String()].DownloadCount, ShouldEqual, 2) - - _, err = metaDB.GetManifestMeta(repo1, "badManiestDigest") - So(err, ShouldNotBeNil) + So(repoMeta.Statistics[image1.Digest.String()].DownloadCount, ShouldEqual, 2) }) Convey("Test AddImageSignature", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") + repo1 = "repo1" + tag1 = "0.0.1" + image1 = CreateRandomImage().AsImageData() ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, image1) So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest1, mTypes.ManifestMetadata{}) - So(err, ShouldBeNil) - - err = metaDB.AddManifestSignature(repo1, manifestDigest1, mTypes.SignatureMetadata{ + err = metaDB.AddManifestSignature(repo1, image1.Digest, mTypes.SignatureMetadata{ SignatureType: "cosign", SignatureDigest: "digest", + LayersInfo: []mTypes.LayerInfo{{LayerDigest: "layer-digest", LayerContent: []byte{10}}}, }) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Signatures[manifestDigest1.String()]["cosign"][0].SignatureManifestDigest, + So(repoMeta.Signatures[image1.Digest.String()]["cosign"][0].SignatureManifestDigest, ShouldResemble, "digest") - _, err = metaDB.GetManifestMeta(repo1, "badDigest") - So(err, ShouldNotBeNil) + imageData, err := metaDB.GetImageData(image1.Digest) + + imageMeta := convert.GetImageMeta(tag1, repoMeta, imageData) + So(err, ShouldBeNil) + So(imageMeta.Signatures["cosign"][0].SignatureManifestDigest, ShouldResemble, "digest") + So(imageMeta.Signatures["cosign"][0].LayersInfo[0].LayerDigest, ShouldResemble, "layer-digest") + So(imageMeta.Signatures["cosign"][0].LayersInfo[0].LayerContent[0], ShouldEqual, 10) }) Convey("Test UpdateSignaturesValidity", func() { Convey("untrusted signature", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("dig") + repo1 = "repo1" + tag1 = "0.0.1" + image1 = CreateRandomImage() ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta(repo1, manifestDigest1, mTypes.ManifestMetadata{ - ManifestBlob: []byte("Bad Manifest"), - ConfigBlob: []byte("Bad Manifest"), - }) + err := metaDB.SetRepoReference(repo1, tag1, image1.AsImageData()) So(err, ShouldBeNil) layerInfo := mTypes.LayerInfo{LayerDigest: "", LayerContent: []byte{}, SignatureKey: ""} - err = metaDB.AddManifestSignature(repo1, manifestDigest1, mTypes.SignatureMetadata{ + err = metaDB.AddManifestSignature(repo1, image1.Digest(), mTypes.SignatureMetadata{ SignatureType: "cosign", - SignatureDigest: string(manifestDigest1), + SignatureDigest: image1.DigestStr(), LayersInfo: []mTypes.LayerInfo{layerInfo}, }) So(err, ShouldBeNil) - err = metaDB.UpdateSignaturesValidity(repo1, manifestDigest1) + err = metaDB.UpdateSignaturesValidity(repo1, image1.Digest()) So(err, ShouldBeNil) - repoData, err := metaDB.GetRepoMeta(repo1) + repoData, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoData.Signatures[string(manifestDigest1)]["cosign"][0].LayersInfo[0].Signer, + So(repoData.Signatures[image1.DigestStr()]["cosign"][0].LayersInfo[0].Signer, ShouldBeEmpty) - So(repoData.Signatures[string(manifestDigest1)]["cosign"][0].LayersInfo[0].Date, + So(repoData.Signatures[image1.DigestStr()]["cosign"][0].LayersInfo[0].Date, ShouldBeZeroValue) }) Convey("trusted signature", func() { - _, _, manifest, _ := deprecated.GetRandomImageComponents(10) //nolint:staticcheck - manifestContent, _ := json.Marshal(manifest) - manifestDigest := godigest.FromBytes(manifestContent) - repo := "repo" + image1 := CreateRandomImage() + repo := "repo1" tag := "0.0.1" - err := metaDB.SetRepoReference(repo, tag, manifestDigest, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta(repo, manifestDigest, mTypes.ManifestMetadata{ - ManifestBlob: manifestContent, - ConfigBlob: []byte("configContent"), - }) + err := metaDB.SetRepoReference(repo, tag, image1.AsImageData()) So(err, ShouldBeNil) mediaType := jws.MediaTypeEnvelope @@ -1424,7 +1319,7 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func signingKeys, err := signature.LoadNotationSigningkeys(tdir) So(err, ShouldBeNil) - idx := test.Index(signingKeys.Keys, keyName) + idx := tCommon.Index(signingKeys.Keys, keyName) So(idx, ShouldBeGreaterThanOrEqualTo, 0) key := signingKeys.Keys[idx] @@ -1435,9 +1330,9 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func } descToSign := ispec.Descriptor{ - MediaType: manifest.MediaType, - Digest: manifestDigest, - Size: int64(len(manifestContent)), + MediaType: image1.Manifest.MediaType, + Digest: image1.Digest(), + Size: image1.ManifestDescriptor.Size, } ctx := context.Background() @@ -1450,7 +1345,7 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func LayerContent: sig, SignatureKey: mediaType, } - err = metaDB.AddManifestSignature(repo, manifestDigest, mTypes.SignatureMetadata{ + err = metaDB.AddManifestSignature(repo, image1.Digest(), mTypes.SignatureMetadata{ SignatureType: "notation", SignatureDigest: string(godigest.FromString("signature digest")), LayersInfo: []mTypes.LayerInfo{layerInfo}, @@ -1471,80 +1366,71 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func err = imagetrust.UploadCertificate(imgTrustStore.NotationStorage, certificateContent, "ca") So(err, ShouldBeNil) - err = metaDB.UpdateSignaturesValidity(repo, manifestDigest) //nolint:contextcheck + err = metaDB.UpdateSignaturesValidity(repo, image1.Digest()) //nolint:contextcheck So(err, ShouldBeNil) - repoData, err := metaDB.GetRepoMeta(repo) + repoData, err := metaDB.GetRepoMeta(ctx, repo) So(err, ShouldBeNil) - So(repoData.Signatures[string(manifestDigest)]["notation"][0].LayersInfo[0].Signer, + So(repoData.Signatures[image1.DigestStr()]["notation"][0].LayersInfo[0].Signer, ShouldNotBeEmpty) - So(repoData.Signatures[string(manifestDigest)]["notation"][0].LayersInfo[0].Date, + So(repoData.Signatures[image1.DigestStr()]["notation"][0].LayersInfo[0].Date, ShouldNotBeZeroValue) }) }) Convey("Test AddImageSignature with inverted order", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") + repo1 = "repo1" + tag1 = "0.0.1" + image1 = CreateRandomImage() ) - err := metaDB.AddManifestSignature(repo1, manifestDigest1, mTypes.SignatureMetadata{ + err := metaDB.AddManifestSignature(repo1, image1.Digest(), mTypes.SignatureMetadata{ SignatureType: "cosign", SignatureDigest: "digest", }) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, tag1, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData(manifestDigest1, mTypes.ManifestData{}) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - - repoMeta, err := metaDB.GetRepoMeta(repo1) - So(err, ShouldBeNil) - So(repoMeta.Signatures[manifestDigest1.String()]["cosign"][0].SignatureManifestDigest, + So(repoMeta.Signatures[image1.DigestStr()]["cosign"][0].SignatureManifestDigest, ShouldResemble, "digest") - - _, err = metaDB.GetManifestMeta(repo1, "badDigest") - So(err, ShouldNotBeNil) }) Convey("Test DeleteSignature", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") + repo1 = "repo1" + tag1 = "0.0.1" + image1 = CreateRandomImage() ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData(manifestDigest1, mTypes.ManifestData{}) - So(err, ShouldBeNil) - - err = metaDB.AddManifestSignature(repo1, manifestDigest1, mTypes.SignatureMetadata{ + err = metaDB.AddManifestSignature(repo1, image1.Digest(), mTypes.SignatureMetadata{ SignatureType: "cosign", SignatureDigest: "digest", }) So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo1) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Signatures[manifestDigest1.String()]["cosign"][0].SignatureManifestDigest, + So(repoMeta.Signatures[image1.DigestStr()]["cosign"][0].SignatureManifestDigest, ShouldResemble, "digest") - err = metaDB.DeleteSignature(repo1, manifestDigest1, mTypes.SignatureMetadata{ + err = metaDB.DeleteSignature(repo1, image1.Digest(), mTypes.SignatureMetadata{ SignatureType: "cosign", SignatureDigest: "digest", }) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo1) + repoMeta, err = metaDB.GetRepoMeta(ctx, repo1) So(err, ShouldBeNil) - So(repoMeta.Signatures[manifestDigest1.String()]["cosign"], ShouldBeEmpty) + So(repoMeta.Signatures[image1.DigestStr()]["cosign"], ShouldBeEmpty) err = metaDB.DeleteSignature(repo1, "badDigest", mTypes.SignatureMetadata{ SignatureType: "cosign", @@ -1555,138 +1441,89 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func Convey("Test SearchRepos", func() { var ( - repo1 = "repo1" - repo2 = "repo2" - repo3 = "repo3" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - tag2 = "0.0.2" - manifestDigest2 = godigest.FromString("fake-manifest2") - tag3 = "0.0.3" - manifestDigest3 = godigest.FromString("fake-manifest3") - ctx = context.Background() - emptyManifest ispec.Manifest - emptyConfig ispec.Manifest + repo1 = "repo1" + repo2 = "repo2" + repo3 = "repo3" + tag1 = "0.0.1" + tag2 = "0.0.2" + tag3 = "0.0.3" + image1 = CreateRandomImage() + image2 = CreateRandomImage() + image3 = CreateRandomImage() + ctx = context.Background() ) - emptyManifestBlob, err := json.Marshal(emptyManifest) - So(err, ShouldBeNil) - - emptyConfigBlob, err := json.Marshal(emptyConfig) - So(err, ShouldBeNil) - - emptyRepoMeta := mTypes.ManifestMetadata{ - ManifestBlob: emptyManifestBlob, - ConfigBlob: emptyConfigBlob, - } - + _ = repo3 Convey("Search all repos", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag2, manifestDigest2, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, tag2, image2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, tag3, manifestDigest3, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo2, tag3, image3.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest1, emptyRepoMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest2, emptyRepoMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest3, emptyRepoMeta) + repoMetaList, err := metaDB.SearchRepos(ctx, "") So(err, ShouldBeNil) + So(len(repoMetaList), ShouldEqual, 2) - repos, manifestMetaMap, _, err := metaDB.SearchRepos(ctx, "") - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 2) - So(len(manifestMetaMap), ShouldEqual, 3) - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest2.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) + So(repoMetaList[0].Tags[tag1].Digest, ShouldResemble, image1.DigestStr()) + So(repoMetaList[0].Tags[tag2].Digest, ShouldResemble, image2.DigestStr()) + So(repoMetaList[1].Tags[tag3].Digest, ShouldResemble, image3.DigestStr()) }) Convey("Search a repo by name", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta(repo1, manifestDigest1, emptyRepoMeta) + err := metaDB.SetRepoReference(repo1, tag1, image1.AsImageData()) So(err, ShouldBeNil) - repos, manifestMetaMap, _, err := metaDB.SearchRepos(ctx, repo1) + repoMetaList, err := metaDB.SearchRepos(ctx, repo1) So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(len(manifestMetaMap), ShouldEqual, 1) - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) + So(len(repoMetaList), ShouldEqual, 1) + So(repoMetaList[0].Tags[tag1].Digest, ShouldResemble, image1.DigestStr()) }) Convey("Search non-existing repo by name", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag2, manifestDigest2, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, tag2, image2.AsImageData()) So(err, ShouldBeNil) - repos, manifestMetaMap, _, err := metaDB.SearchRepos(ctx, "RepoThatDoesntExist") + repoMetaList, err := metaDB.SearchRepos(ctx, "RepoThatDoesntExist") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 0) - So(len(manifestMetaMap), ShouldEqual, 0) + So(len(repoMetaList), ShouldEqual, 0) }) Convey("Search with partial match", func() { - err := metaDB.SetRepoReference("alpine", tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("pine", tag2, manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference("golang", tag3, manifestDigest3, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta("alpine", manifestDigest1, emptyRepoMeta) + err := metaDB.SetRepoReference("alpine", tag1, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestMeta("pine", manifestDigest2, emptyRepoMeta) + err = metaDB.SetRepoReference("pine", tag2, image2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestMeta("golang", manifestDigest3, emptyRepoMeta) + err = metaDB.SetRepoReference("golang", tag3, image3.AsImageData()) So(err, ShouldBeNil) - repos, manifestMetaMap, _, err := metaDB.SearchRepos(ctx, "pine") + repoMetaList, err := metaDB.SearchRepos(ctx, "pine") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 2) - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest2.String()) - So(manifestMetaMap, ShouldNotContainKey, manifestDigest3.String()) + So(len(repoMetaList), ShouldEqual, 2) }) Convey("Search multiple repos that share manifests", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference("alpine", tag1, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, tag2, manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("pine", tag2, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo3, tag3, manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("golang", tag3, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest1, emptyRepoMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo2, manifestDigest1, emptyRepoMeta) + repoMetaList, err := metaDB.SearchRepos(ctx, "") So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo3, manifestDigest1, emptyRepoMeta) - So(err, ShouldBeNil) - - repos, manifestMetaMap, _, err := metaDB.SearchRepos(ctx, "") - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 3) - So(len(manifestMetaMap), ShouldEqual, 1) + So(len(repoMetaList), ShouldEqual, 3) }) Convey("Search repos with access control", func() { - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, tag2, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo3, tag3, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, tag1, image1.AsImageData()) So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta(repo1, manifestDigest1, emptyRepoMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo2, manifestDigest1, emptyRepoMeta) + err = metaDB.SetRepoReference(repo2, tag2, image2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo3, manifestDigest1, emptyRepoMeta) + err = metaDB.SetRepoReference(repo3, tag3, image3.AsImageData()) So(err, ShouldBeNil) userAc := reqCtx.NewUserAccessControl() @@ -1698,153 +1535,84 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func ctx := userAc.DeriveContext(context.Background()) - repos, _, _, err := metaDB.SearchRepos(ctx, "repo") + repoMetaList, err := metaDB.SearchRepos(ctx, "repo") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 2) - for _, k := range repos { + So(len(repoMetaList), ShouldEqual, 2) + for _, k := range repoMetaList { So(k.Name, ShouldBeIn, []string{repo1, repo2}) } }) Convey("Search Repos with Indexes", func() { var ( - tag4 = "0.0.4" - indexDigest = godigest.FromString("Multiarch") - manifestDigest1 = godigest.FromString("manifestDigest1") - manifestDigest2 = godigest.FromString("manifestDigest2") + tag4 = "0.0.4" + subImage1 = CreateRandomImage() + subImage2 = CreateRandomImage() + multiarch = CreateMultiarchWith().Images([]Image{subImage1, subImage2}).Build() - tag5 = "0.0.5" - manifestDigest3 = godigest.FromString("manifestDigest3") + tag5 = "0.0.5" + image1 = CreateRandomImage() ) - err := metaDB.SetManifestData(manifestDigest1, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) - - config := ispec.Image{ - Platform: ispec.Platform{ - Architecture: "arch", - OS: "os", - }, - } - - confBlob, err := json.Marshal(config) + err := metaDB.SetRepoReference("repo", subImage1.DigestStr(), subImage1.AsImageData()) So(err, ShouldBeNil) - - err = metaDB.SetManifestData(manifestDigest2, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: confBlob, - }) + err = metaDB.SetRepoReference("repo", subImage2.DigestStr(), subImage2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData(manifestDigest3, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) + err = metaDB.SetRepoReference("repo", tag4, multiarch.AsImageData()) So(err, ShouldBeNil) - indexContent := ispec.Index{ - MediaType: ispec.MediaTypeImageIndex, - Manifests: []ispec.Descriptor{ - { - Digest: manifestDigest1, - }, - { - Digest: manifestDigest2, - }, - }, - } - - indexBlob, err := json.Marshal(indexContent) + err = metaDB.SetRepoReference("repo", tag5, image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetIndexData(indexDigest, mTypes.IndexData{ - IndexBlob: indexBlob, - }) - So(err, ShouldBeNil) - - err = metaDB.SetRepoReference("repo", tag4, indexDigest, ispec.MediaTypeImageIndex) - So(err, ShouldBeNil) - - err = metaDB.SetRepoReference("repo", tag5, manifestDigest3, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - repos, manifestMetaMap, indexDataMap, err := metaDB.SearchRepos(ctx, "repo") + repoMetaList, err := metaDB.SearchRepos(ctx, "repo") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(repos[0].Name, ShouldResemble, "repo") - So(repos[0].Tags, ShouldContainKey, tag4) - So(repos[0].Tags, ShouldContainKey, tag5) - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest2.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) - So(indexDataMap, ShouldContainKey, indexDigest.String()) + So(len(repoMetaList), ShouldEqual, 1) + So(repoMetaList[0].Name, ShouldResemble, "repo") + So(repoMetaList[0].Tags, ShouldContainKey, tag4) + So(repoMetaList[0].Tags[tag4].MediaType, ShouldResemble, ispec.MediaTypeImageIndex) + So(repoMetaList[0].Tags, ShouldContainKey, tag5) + So(repoMetaList[0].Tags[tag5].MediaType, ShouldResemble, ispec.MediaTypeImageManifest) }) }) Convey("Test SearchTags", func() { var ( - repo1 = "repo1" - repo2 = "repo2" - manifestDigest1 = godigest.FromString("fake-manifest1") - manifestDigest2 = godigest.FromString("fake-manifest2") - manifestDigest3 = godigest.FromString("fake-manifest3") - ctx = context.Background() - emptyManifest ispec.Manifest - emptyConfig ispec.Manifest + repo1 = "repo1" + repo2 = "repo2" + image1 = CreateRandomImage() + image2 = CreateRandomImage() + image3 = CreateRandomImage() + ctx = context.Background() ) - emptyManifestBlob, err := json.Marshal(emptyManifest) + err := metaDB.SetRepoReference(repo1, "0.0.1", image1.AsImageData()) So(err, ShouldBeNil) - - emptyConfigBlob, err := json.Marshal(emptyConfig) + err = metaDB.SetRepoReference(repo1, "0.0.2", image3.AsImageData()) So(err, ShouldBeNil) - - emptyRepoMeta := mTypes.ManifestMetadata{ - ManifestBlob: emptyManifestBlob, - ConfigBlob: emptyConfigBlob, - } - - err = metaDB.SetRepoReference(repo1, "0.0.1", manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "0.1.0", image2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "0.0.2", manifestDigest3, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "1.0.0", image2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "0.1.0", manifestDigest2, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "1.0.1", image2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "1.0.0", manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "1.0.1", manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, "0.0.1", manifestDigest3, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta(repo1, manifestDigest1, emptyRepoMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest2, emptyRepoMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest3, emptyRepoMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo2, manifestDigest3, emptyRepoMeta) + err = metaDB.SetRepoReference(repo2, "0.0.1", image3.AsImageData()) So(err, ShouldBeNil) Convey("With exact match", func() { - repos, manifestMetaMap, _, err := metaDB.SearchTags(ctx, "repo1:0.0.1") + imageDataList, err := metaDB.SearchTags(ctx, "repo1:0.0.1") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(len(repos[0].Tags), ShouldEqual, 1) - So(repos[0].Tags, ShouldContainKey, "0.0.1") - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) + So(len(imageDataList), ShouldEqual, 1) + So(imageDataList[0].Digest.String(), ShouldResemble, image1.DigestStr()) }) Convey("With no match", func() { - repos, _, _, err := metaDB.SearchTags(ctx, "repo1:badtag") + imageDataList, err := metaDB.SearchTags(ctx, "repo1:badtag") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 0) + So(len(imageDataList), ShouldEqual, 0) }) - Convey("With no permision", func() { + Convey("With no permission", func() { userAc := reqCtx.NewUserAccessControl() userAc.SetUsername("user1") userAc.SetGlobPatterns("read", @@ -1855,76 +1623,51 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func ) ctx1 := userAc.DeriveContext(context.Background()) - repos, _, _, err := metaDB.SearchTags(ctx1, "repo1:0.0.1") + imageDataList, err := metaDB.SearchTags(ctx1, "repo1:0.0.1") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 0) + So(len(imageDataList), ShouldEqual, 0) }) Convey("With partial repo path", func() { - repos, manifestMetaMap, _, err := metaDB.SearchTags(ctx, "repo:0.0.1") + imageDataList, err := metaDB.SearchTags(ctx, "repo:0.0.1") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 0) - So(len(manifestMetaMap), ShouldEqual, 0) + So(len(imageDataList), ShouldEqual, 0) }) Convey("With partial tag", func() { - repos, manifestMetaMap, _, err := metaDB.SearchTags(ctx, "repo1:0.0") - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(len(repos[0].Tags), ShouldEqual, 2) - So(repos[0].Tags, ShouldContainKey, "0.0.2") - So(repos[0].Tags, ShouldContainKey, "0.0.1") - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) - - repos, manifestMetaMap, _, err = metaDB.SearchTags(ctx, "repo1:0.") - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(len(repos[0].Tags), ShouldEqual, 3) - So(repos[0].Tags, ShouldContainKey, "0.0.1") - So(repos[0].Tags, ShouldContainKey, "0.0.2") - So(repos[0].Tags, ShouldContainKey, "0.1.0") - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest2.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) - }) + imageDataList, err := metaDB.SearchTags(ctx, "repo1:0.0") + So(err, ShouldBeNil) + So(len(imageDataList), ShouldEqual, 2) - Convey("With bad query", func() { - repos, manifestMetaMap, _, err := metaDB.SearchTags(ctx, "repo:0.0.1:test") - So(err, ShouldNotBeNil) - So(len(repos), ShouldEqual, 0) - So(len(manifestMetaMap), ShouldEqual, 0) - }) + tags := map[string]struct{}{} + for _, imageData := range imageDataList { + tags[imageData.Tag] = struct{}{} + } - Convey("Search with access control", func() { - var ( - repo1 = "repo1" - repo2 = "repo2" - repo3 = "repo3" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - tag2 = "0.0.2" - tag3 = "0.0.3" - ) + So(tags, ShouldContainKey, "0.0.2") + So(tags, ShouldContainKey, "0.0.1") - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, tag2, manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo3, tag3, manifestDigest1, ispec.MediaTypeImageManifest) + imageDataList, err = metaDB.SearchTags(ctx, "repo1:0.") So(err, ShouldBeNil) + So(len(imageDataList), ShouldEqual, 3) - config := ispec.Image{} - configBlob, err := json.Marshal(config) - So(err, ShouldBeNil) + tags = map[string]struct{}{} + for _, imageData := range imageDataList { + tags[imageData.Tag] = struct{}{} + } - err = metaDB.SetManifestMeta(repo1, manifestDigest1, mTypes.ManifestMetadata{ConfigBlob: configBlob}) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo2, manifestDigest1, mTypes.ManifestMetadata{ConfigBlob: configBlob}) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo3, manifestDigest1, mTypes.ManifestMetadata{ConfigBlob: configBlob}) - So(err, ShouldBeNil) + So(tags, ShouldContainKey, "0.0.1") + So(tags, ShouldContainKey, "0.0.2") + So(tags, ShouldContainKey, "0.1.0") + }) + Convey("With bad query", func() { + imageDataList, err := metaDB.SearchTags(ctx, "repo:0.0.1:test") + So(err, ShouldNotBeNil) + So(len(imageDataList), ShouldEqual, 0) + }) + + Convey("Search with access control", func() { userAc := reqCtx.NewUserAccessControl() userAc.SetUsername("username") userAc.SetGlobPatterns("read", map[string]bool{ @@ -1934,297 +1677,193 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func ctx := userAc.DeriveContext(context.Background()) - repos, _, _, err := metaDB.SearchTags(ctx, "repo1:") + imageDataList, err := metaDB.SearchTags(ctx, "repo1:") So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(repos[0].Name, ShouldResemble, repo1) + So(len(imageDataList), ShouldEqual, 5) + So(imageDataList[0].Repo, ShouldResemble, repo1) - repos, _, _, err = metaDB.SearchTags(ctx, "repo2:") + imageDataList, err = metaDB.SearchTags(ctx, "repo2:") So(err, ShouldBeNil) - So(repos, ShouldBeEmpty) + So(imageDataList, ShouldBeEmpty) }) Convey("Search Tags with Indexes", func() { var ( - tag4 = "0.0.4" - indexDigest = godigest.FromString("Multiarch") - manifestDigest1 = godigest.FromString("manifestDigest1") - manifestDigest2 = godigest.FromString("manifestDigest2") + tag4 = "0.0.4" + subImage1 = CreateRandomImage() + subImage2 = CreateRandomImage() + multiarch = CreateMultiarchWith().Images([]Image{subImage1, subImage2}).Build() - tag5 = "0.0.5" - manifestDigest3 = godigest.FromString("manifestDigest3") + tag5 = "0.0.5" + image5 = CreateRandomImage() - tag6 = "6.0.0" - manifestDigest4 = godigest.FromString("manifestDigest4") + tag6 = "6.0.0" + image6 = CreateRandomImage() ) - err := metaDB.SetManifestData(manifestDigest1, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) - - config := ispec.Image{ - Platform: ispec.Platform{ - Architecture: "arch", - OS: "os", - }, - } - - confBlob, err := json.Marshal(config) + err = metaDB.SetRepoReference("repo", subImage1.DigestStr(), subImage1.AsImageData()) So(err, ShouldBeNil) - - err = metaDB.SetManifestData(manifestDigest2, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: confBlob, - }) + err = metaDB.SetRepoReference("repo", subImage2.DigestStr(), subImage2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData(manifestDigest3, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) + err = metaDB.SetRepoReference("repo", tag4, multiarch.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetManifestData(manifestDigest4, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) + err = metaDB.SetRepoReference("repo", tag5, image5.AsImageData()) So(err, ShouldBeNil) - indexBlob, err := GetIndexBlobWithManifests( - []godigest.Digest{ - manifestDigest1, - manifestDigest2, - }, - ) + err = metaDB.SetRepoReference("repo", tag6, image6.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetIndexData(indexDigest, mTypes.IndexData{ - IndexBlob: indexBlob, - }) + imageDataList, err := metaDB.SearchTags(ctx, "repo:0.0") So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo", tag4, indexDigest, ispec.MediaTypeImageIndex) - So(err, ShouldBeNil) + tags := map[string]struct{}{} + for _, imageData := range imageDataList { + tags[imageData.Tag] = struct{}{} + } - err = metaDB.SetRepoReference("repo", tag5, manifestDigest3, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + So(len(imageDataList), ShouldEqual, 2) + So(tags, ShouldContainKey, tag4) + So(tags, ShouldContainKey, tag5) + So(tags, ShouldNotContainKey, tag6) - err = metaDB.SetRepoReference("repo", tag6, manifestDigest4, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + multiarchImageData := mTypes.ImageMeta{} + found := false - repos, manifestMetaMap, indexDataMap, err := metaDB.SearchTags(ctx, "repo:0.0") - So(err, ShouldBeNil) + for _, imageData := range imageDataList { + if imageData.MediaType == ispec.MediaTypeImageIndex { + multiarchImageData = imageData + found = true + } + } + + So(found, ShouldBeTrue) + So(len(multiarchImageData.Manifests), ShouldEqual, 2) + + digests := []string{} + for _, manifest := range multiarchImageData.Manifests { + digests = append(digests, manifest.Digest.String()) + } - So(len(repos), ShouldEqual, 1) - So(repos[0].Name, ShouldResemble, "repo") - So(repos[0].Tags, ShouldContainKey, tag4) - So(repos[0].Tags, ShouldContainKey, tag5) - So(repos[0].Tags, ShouldNotContainKey, tag6) - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest2.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) - So(manifestMetaMap, ShouldNotContainKey, manifestDigest4.String()) - So(indexDataMap, ShouldContainKey, indexDigest.String()) + So(digests, ShouldContain, subImage1.DigestStr()) + So(digests, ShouldContain, subImage2.DigestStr()) }) }) - Convey("Paginated tag search", func() { + Convey("Test FilterTags", func() { var ( - repo1 = "repo1" - tag1 = "0.0.1" - manifestDigest1 = godigest.FromString("fake-manifest1") - tag2 = "0.0.2" - tag3 = "0.0.3" - tag4 = "0.0.4" - tag5 = "0.0.5" + repo1 = "repo1" + repo2 = "repo2" + image1 = CreateRandomImage() + image2 = CreateRandomImage() + image3 = CreateRandomImage() + + subImage1 = CreateRandomImage() + subImage2 = CreateRandomImage() + multiarch = CreateMultiarchWith().Images([]Image{subImage1, subImage2}).Build() + ctx = context.Background() ) - err := metaDB.SetRepoReference(repo1, tag1, manifestDigest1, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo1, subImage1.DigestStr(), subImage1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag2, manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, subImage2.DigestStr(), subImage2.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag3, manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "2.0.0", multiarch.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag4, manifestDigest1, ispec.MediaTypeImageManifest) + + err = metaDB.SetRepoReference(repo1, "0.0.1", image1.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, tag5, manifestDigest1, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo1, "0.0.2", image3.AsImageData()) So(err, ShouldBeNil) - - config := ispec.Image{} - configBlob, err := json.Marshal(config) + err = metaDB.SetRepoReference(repo1, "0.1.0", image2.AsImageData()) So(err, ShouldBeNil) - - err = metaDB.SetManifestMeta(repo1, manifestDigest1, mTypes.ManifestMetadata{ConfigBlob: configBlob}) + err = metaDB.SetRepoReference(repo1, "1.0.0", image2.AsImageData()) So(err, ShouldBeNil) - - repos, _, _, err := metaDB.SearchTags(context.TODO(), "repo1:") - + err = metaDB.SetRepoReference(repo1, "1.0.1", image2.AsImageData()) + So(err, ShouldBeNil) + err = metaDB.SetRepoReference(repo2, "0.0.1", image3.AsImageData()) So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - keys := make([]string, 0, len(repos[0].Tags)) - for k := range repos[0].Tags { - keys = append(keys, k) - } - repos, _, _, err = metaDB.SearchTags(context.TODO(), "repo1:") + Convey("Return all tags", func() { + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, mTypes.AcceptAllImageData) - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - for k := range repos[0].Tags { - keys = append(keys, k) - } + So(err, ShouldBeNil) + So(len(imageDataList), ShouldEqual, 7) - repos, _, _, err = metaDB.SearchTags(context.TODO(), "repo1:") + tags := []string{} + indexImage := mTypes.ImageMeta{} - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - for k := range repos[0].Tags { - keys = append(keys, k) - } + for _, imageData := range imageDataList { + tags = append(tags, imageData.Tag) - So(keys, ShouldContain, tag1) - So(keys, ShouldContain, tag2) - So(keys, ShouldContain, tag3) - }) + if imageData.MediaType == ispec.MediaTypeImageIndex { + indexImage = imageData + } + } - Convey("Test FilterTags", func() { - var ( - repo1 = "repo1" - repo2 = "repo2" - manifestDigest1 = godigest.FromString("fake-manifest1") - manifestDigest2 = godigest.FromString("fake-manifest2") - manifestDigest3 = godigest.FromString("fake-manifest3") - indexDigest = godigest.FromString("index-digest") - manifestFromIndexDigest1 = godigest.FromString("fake-manifestFromIndexDigest1") - manifestFromIndexDigest2 = godigest.FromString("fake-manifestFromIndexDigest2") - - emptyManifest ispec.Manifest - emptyConfig ispec.Image - ctx = context.Background() - ) + So(zcommon.Contains(tags, "0.0.1"), ShouldBeTrue) + So(zcommon.Contains(tags, "0.0.2"), ShouldBeTrue) + So(zcommon.Contains(tags, "0.1.0"), ShouldBeTrue) + So(zcommon.Contains(tags, "1.0.0"), ShouldBeTrue) + So(zcommon.Contains(tags, "1.0.1"), ShouldBeTrue) + So(zcommon.Contains(tags, "2.0.0"), ShouldBeTrue) + So(zcommon.Contains(tags, "0.0.1"), ShouldBeTrue) - emptyManifestBlob, err := json.Marshal(emptyManifest) - So(err, ShouldBeNil) + So(indexImage.Digest.String(), ShouldResemble, multiarch.DigestStr()) - emptyConfigBlob, err := json.Marshal(emptyConfig) - So(err, ShouldBeNil) + digests := []string{} + for _, manifest := range indexImage.Manifests { + digests = append(digests, manifest.Digest.String()) + } - emptyManifestMeta := mTypes.ManifestMetadata{ - ManifestBlob: emptyManifestBlob, - ConfigBlob: emptyConfigBlob, - } + So(digests, ShouldContain, subImage1.DigestStr()) + So(digests, ShouldContain, subImage2.DigestStr()) + }) - emptyManifestData := mTypes.ManifestData{ - ManifestBlob: emptyManifestBlob, - ConfigBlob: emptyConfigBlob, - } + Convey("Return all tags in a specific repo", func() { + imageDataList, err := metaDB.FilterTags(ctx, func(repo, tag string) bool { return repo == repo1 }, + mTypes.AcceptAllImageData) - err = metaDB.SetRepoReference(repo1, "2.0.0", indexDigest, ispec.MediaTypeImageIndex) - So(err, ShouldBeNil) + So(err, ShouldBeNil) + So(len(imageDataList), ShouldEqual, 6) - indexBlob, err := GetIndexBlobWithManifests([]godigest.Digest{ - manifestFromIndexDigest1, - manifestFromIndexDigest2, - }) - So(err, ShouldBeNil) + tags := map[string]struct{}{} + indexImage := mTypes.ImageMeta{} - err = metaDB.SetIndexData(indexDigest, mTypes.IndexData{ - IndexBlob: indexBlob, - }) - So(err, ShouldBeNil) + for _, imageData := range imageDataList { + tags[imageData.Tag] = struct{}{} - err = metaDB.SetRepoReference(repo1, "0.0.1", manifestDigest1, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "0.0.2", manifestDigest3, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "0.1.0", manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "1.0.0", manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo1, "1.0.1", manifestDigest2, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo2, "0.0.1", manifestDigest3, ispec.MediaTypeImageManifest) - So(err, ShouldBeNil) + if imageData.MediaType == ispec.MediaTypeImageIndex { + indexImage = imageData + } + } - err = metaDB.SetManifestMeta(repo1, manifestDigest1, emptyManifestMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest2, emptyManifestMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo1, manifestDigest3, emptyManifestMeta) - So(err, ShouldBeNil) - err = metaDB.SetManifestMeta(repo2, manifestDigest3, emptyManifestMeta) - So(err, ShouldBeNil) + So(tags, ShouldContainKey, "0.0.1") + So(tags, ShouldContainKey, "0.0.2") + So(tags, ShouldContainKey, "0.1.0") + So(tags, ShouldContainKey, "1.0.0") + So(tags, ShouldContainKey, "1.0.1") + So(tags, ShouldContainKey, "2.0.0") - err = metaDB.SetManifestData(manifestFromIndexDigest1, emptyManifestData) - So(err, ShouldBeNil) - err = metaDB.SetManifestData(manifestFromIndexDigest2, emptyManifestData) - So(err, ShouldBeNil) + So(indexImage.Digest.String(), ShouldResemble, multiarch.DigestStr()) - Convey("Return all tags", func() { - repos, manifestMetaMap, indexDataMap, err := metaDB.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return true - }) - - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 2) - So(repos[0].Name, ShouldEqual, "repo1") - So(repos[1].Name, ShouldEqual, "repo2") - So(len(repos[0].Tags), ShouldEqual, 6) - So(len(repos[1].Tags), ShouldEqual, 1) - So(repos[0].Tags, ShouldContainKey, "0.0.1") - So(repos[0].Tags, ShouldContainKey, "0.0.2") - So(repos[0].Tags, ShouldContainKey, "0.1.0") - So(repos[0].Tags, ShouldContainKey, "1.0.0") - So(repos[0].Tags, ShouldContainKey, "1.0.1") - So(repos[0].Tags, ShouldContainKey, "2.0.0") - So(repos[1].Tags, ShouldContainKey, "0.0.1") - - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest2.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) - - So(indexDataMap, ShouldContainKey, indexDigest.String()) - So(manifestMetaMap, ShouldContainKey, manifestFromIndexDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestFromIndexDigest2.String()) - }) + digests := []string{} + for _, manifest := range indexImage.Manifests { + digests = append(digests, manifest.Digest.String()) + } - Convey("Return all tags in a specific repo", func() { - repos, manifestMetaMap, indexDataMap, err := metaDB.FilterTags( - ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return repoMeta.Name == repo1 - }) - - So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(repos[0].Name, ShouldEqual, repo1) - So(len(repos[0].Tags), ShouldEqual, 6) - So(repos[0].Tags, ShouldContainKey, "0.0.1") - So(repos[0].Tags, ShouldContainKey, "0.0.2") - So(repos[0].Tags, ShouldContainKey, "0.1.0") - So(repos[0].Tags, ShouldContainKey, "1.0.0") - So(repos[0].Tags, ShouldContainKey, "1.0.1") - So(repos[0].Tags, ShouldContainKey, "2.0.0") - So(manifestMetaMap, ShouldContainKey, manifestDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest2.String()) - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) - So(indexDataMap, ShouldContainKey, indexDigest.String()) - So(manifestMetaMap, ShouldContainKey, manifestFromIndexDigest1.String()) - So(manifestMetaMap, ShouldContainKey, manifestFromIndexDigest2.String()) + So(digests, ShouldContain, subImage1.DigestStr()) + So(digests, ShouldContain, subImage2.DigestStr()) }) Convey("Filter everything out", func() { - repos, manifestMetaMap, _, err := metaDB.FilterTags( - ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return false - }) + imageDataList, err := metaDB.FilterTags(ctx, + func(repo, tag string) bool { return false }, + func(repoMeta mTypes.RepoMeta, imageData mTypes.ImageData) bool { return false }, + ) So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 0) - So(len(manifestMetaMap), ShouldEqual, 0) + So(len(imageDataList), ShouldEqual, 0) }) Convey("Search with access control", func() { @@ -2237,243 +1876,288 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func ctx := userAc.DeriveContext(context.Background()) - repos, manifestMetaMap, _, err := metaDB.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { - return true - }) + imageDataList, err := metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, mTypes.AcceptAllImageData) So(err, ShouldBeNil) - So(len(repos), ShouldEqual, 1) - So(repos[0].Name, ShouldResemble, repo2) - So(len(repos[0].Tags), ShouldEqual, 1) - So(repos[0].Tags, ShouldContainKey, "0.0.1") - So(manifestMetaMap, ShouldContainKey, manifestDigest3.String()) + So(len(imageDataList), ShouldEqual, 1) + So(imageDataList[0].Repo, ShouldResemble, repo2) + So(imageDataList[0].Tag, ShouldResemble, "0.0.1") }) }) - Convey("Test index logic", func() { - multiArch, err := deprecated.GetRandomMultiarchImage("tag1") //nolint:staticcheck - So(err, ShouldBeNil) + Convey("Test Referrers", func() { + image1 := CreateRandomImage() - indexDigest := multiArch.Digest() + err := metaDB.SetRepoReference("repo", "tag", image1.AsImageData()) + So(err, ShouldBeNil) - indexData := multiArch.IndexData() + // Artifact 1 with artifact type in Manifest + artifact1 := CreateImageWith(). + RandomLayers(10, 2).DefaultConfig(). + ArtifactType("art-type1"). + Subject(image1.DescriptorRef()). + Build() - err = metaDB.SetIndexData(indexDigest, indexData) + err = metaDB.SetRepoReference("repo", artifact1.DigestStr(), artifact1.AsImageData()) So(err, ShouldBeNil) - result, err := metaDB.GetIndexData(indexDigest) - So(err, ShouldBeNil) - So(result, ShouldResemble, indexData) + // Artifact 2 with artifact type in Config media type + artifact2 := CreateImageWith(). + RandomLayers(10, 2). + ArtifactConfig("art-type2"). + Subject(image1.DescriptorRef()). + Build() - _, err = metaDB.GetIndexData(godigest.FromString("inexistent")) - So(err, ShouldNotBeNil) - }) - - Convey("Test Referrers", func() { - image, err := deprecated.GetRandomImage() //nolint:staticcheck + err = metaDB.SetRepoReference("repo", artifact2.DigestStr(), artifact2.AsImageData()) So(err, ShouldBeNil) - referredDigest := image.Digest() - - manifestBlob, err := json.Marshal(image.Manifest) + // GetReferrers + referrers, err := metaDB.GetReferrersInfo("repo", image1.Digest(), nil) + So(len(referrers), ShouldEqual, 2) + So(referrers, ShouldContain, mTypes.ReferrerInfo{ + Digest: artifact1.DigestStr(), + MediaType: ispec.MediaTypeImageManifest, + ArtifactType: "art-type1", + Size: int(artifact1.ManifestDescriptor.Size), + }) + So(referrers, ShouldContain, mTypes.ReferrerInfo{ + Digest: artifact2.DigestStr(), + MediaType: ispec.MediaTypeImageManifest, + ArtifactType: "art-type2", + Size: int(artifact2.ManifestDescriptor.Size), + }) So(err, ShouldBeNil) - configBlob, err := json.Marshal(image.Config) + // Delete the Referrers + err = metaDB.RemoveRepoReference("repo", artifact1.DigestStr(), artifact1.Digest()) So(err, ShouldBeNil) - manifestData := mTypes.ManifestData{ - ManifestBlob: manifestBlob, - ConfigBlob: configBlob, - } + referrers, err = metaDB.GetReferrersInfo("repo", image1.Digest(), nil) + So(err, ShouldBeNil) + So(len(referrers), ShouldEqual, 1) - err = metaDB.SetManifestData(referredDigest, manifestData) + err = metaDB.RemoveRepoReference("repo", artifact2.DigestStr(), artifact2.Digest()) So(err, ShouldBeNil) - err = metaDB.SetRepoReference("repo", "tag", referredDigest, ispec.MediaTypeImageManifest) + referrers, err = metaDB.GetReferrersInfo("repo", image1.Digest(), nil) So(err, ShouldBeNil) + So(len(referrers), ShouldEqual, 0) + }) - // ------- Add Artifact 1 + Convey("Test Referrers add same one twice but with different tags - delete by tag then digest", func() { + _, err := metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldNotBeNil) - artifact1, err := deprecated.GetImageWithSubject( //nolint:staticcheck - referredDigest, - ispec.MediaTypeImageManifest, - ) - So(err, ShouldBeNil) + tag := "tag1" + refTag := "refTag" + image := CreateRandomImage() + referrer := CreateRandomImageWith().Subject(image.DescriptorRef()).Build() - artifactDigest1 := artifact1.Digest() + err = metaDB.SetRepoReference("repo", tag, image.AsImageData()) + So(err, ShouldBeNil) - err = metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: artifactDigest1.String(), - MediaType: ispec.MediaTypeImageManifest, - }) + err = metaDB.SetRepoReference("repo", refTag, referrer.AsImageData()) So(err, ShouldBeNil) - // ------- Add Artifact 2 + err = metaDB.SetRepoReference("repo", referrer.DigestStr(), referrer.AsImageData()) + So(err, ShouldBeNil) - artifact2, err := deprecated.GetImageWithSubject( //nolint:staticcheck - referredDigest, - ispec.MediaTypeImageManifest, - ) + repoMeta, err := metaDB.GetRepoMeta(ctx, "repo") So(err, ShouldBeNil) + So(len(repoMeta.Referrers[image.DigestStr()]), ShouldEqual, 1) - artifactDigest2 := artifact2.Digest() + err = metaDB.RemoveRepoReference("repo", refTag, referrer.Digest()) + So(err, ShouldBeNil) - err = metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: artifactDigest2.String(), - MediaType: ispec.MediaTypeImageManifest, - }) + // we still have the untagged manifest + repoMeta, err = metaDB.GetRepoMeta(ctx, "repo") So(err, ShouldBeNil) + So(len(repoMeta.Referrers[image.DigestStr()]), ShouldEqual, 1) - // ------ GetReferrers + err = metaDB.RemoveRepoReference("repo", referrer.DigestStr(), referrer.Digest()) + So(err, ShouldBeNil) - referrers, err := metaDB.GetReferrersInfo("repo", referredDigest, nil) - So(len(referrers), ShouldEqual, 2) - So(referrers, ShouldContain, mTypes.ReferrerInfo{ - Digest: artifactDigest1.String(), - MediaType: ispec.MediaTypeImageManifest, - }) - So(referrers, ShouldContain, mTypes.ReferrerInfo{ - Digest: artifactDigest2.String(), - MediaType: ispec.MediaTypeImageManifest, - }) + repoMeta, err = metaDB.GetRepoMeta(ctx, "repo") So(err, ShouldBeNil) + So(len(repoMeta.Referrers[image.DigestStr()]), ShouldEqual, 0) + }) - // ------ DeleteReferrers + Convey("Test Referrers add same one twice but with different tags - delete by digest", func() { + _, err := metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldNotBeNil) - err = metaDB.DeleteReferrer("repo", referredDigest, artifactDigest1) - So(err, ShouldBeNil) + tag := "tag-1" + refTag := "refTag" + image := CreateRandomImage() + referrer := CreateRandomImageWith().Subject(image.DescriptorRef()).Build() - err = metaDB.DeleteReferrer("repo", referredDigest, artifactDigest2) + err = metaDB.SetRepoReference("repo", tag, image.AsImageData()) So(err, ShouldBeNil) - referrers, err = metaDB.GetReferrersInfo("repo", referredDigest, nil) + err = metaDB.SetRepoReference("repo", refTag, referrer.AsImageData()) So(err, ShouldBeNil) - So(len(referrers), ShouldEqual, 0) - }) - Convey("Test Referrers on empty Repo", func() { - repoMeta, err := metaDB.GetRepoMeta("repo") - So(err, ShouldNotBeNil) - So(repoMeta, ShouldResemble, mTypes.RepoMetadata{}) + err = metaDB.SetRepoReference("repo", referrer.DigestStr(), referrer.AsImageData()) + So(err, ShouldBeNil) - referredDigest := godigest.FromString("referredDigest") - referrerDigest := godigest.FromString("referrerDigest") + repoMeta, err := metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldBeNil) + So(len(repoMeta.Referrers[image.DigestStr()]), ShouldEqual, 1) - err = metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: referrerDigest.String(), - MediaType: ispec.MediaTypeImageManifest, - }) + // this should delete all references + err = metaDB.RemoveRepoReference("repo", referrer.DigestStr(), referrer.Digest()) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta("repo") + repoMeta, err = metaDB.GetRepoMeta(ctx, "repo") So(err, ShouldBeNil) - So(repoMeta.Referrers[referredDigest.String()][0].Digest, ShouldResemble, referrerDigest.String()) + So(len(repoMeta.Referrers[image.DigestStr()]), ShouldEqual, 0) }) Convey("Test Referrers add same one twice", func() { - repoMeta, err := metaDB.GetRepoMeta("repo") + _, err := metaDB.GetRepoMeta(ctx, "repo") So(err, ShouldNotBeNil) - So(repoMeta, ShouldResemble, mTypes.RepoMetadata{}) - referredDigest := godigest.FromString("referredDigest") - referrerDigest := godigest.FromString("referrerDigest") + tag := "tag-ref" + image := CreateRandomImage() + referrer := CreateRandomImageWith().Subject(image.DescriptorRef()).Build() - err = metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: referrerDigest.String(), - MediaType: ispec.MediaTypeImageManifest, - }) + err = metaDB.SetRepoReference("repo", tag, image.AsImageData()) So(err, ShouldBeNil) - err = metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: referrerDigest.String(), - MediaType: ispec.MediaTypeImageManifest, - }) + err = metaDB.SetRepoReference("repo", referrer.DigestStr(), referrer.AsImageData()) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta("repo") + err = metaDB.SetRepoReference("repo", referrer.DigestStr(), referrer.AsImageData()) So(err, ShouldBeNil) - So(len(repoMeta.Referrers[referredDigest.String()]), ShouldEqual, 1) + + repoMeta, err := metaDB.GetRepoMeta(ctx, "repo") + So(err, ShouldBeNil) + So(len(repoMeta.Referrers[image.DigestStr()]), ShouldEqual, 1) }) Convey("GetReferrersInfo", func() { - referredDigest := godigest.FromString("referredDigest") + repo := "repo" + tag := "tag" - err := metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: "inexistendManifestDigest", - MediaType: ispec.MediaTypeImageManifest, - }) + image := CreateRandomImage() + err := metaDB.SetRepoReference(repo, tag, image.AsImageData()) So(err, ShouldBeNil) - // ------- Set existent manifest and artifact manifest - err = metaDB.SetManifestData("goodManifest", mTypes.ManifestData{ - ManifestBlob: []byte(`{"artifactType": "unwantedType"}`), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) + referrerWantedType := CreateRandomImageWith(). + ArtifactType("wanted-type"). + Subject(image.DescriptorRef()).Build() - err = metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: "goodManifestUnwanted", - MediaType: ispec.MediaTypeImageManifest, - ArtifactType: "unwantedType", - }) - So(err, ShouldBeNil) + referrerNotWantedType := CreateRandomImageWith(). + ArtifactType("not-wanted-type"). + Subject(image.DescriptorRef()).Build() - err = metaDB.SetReferrer("repo", referredDigest, mTypes.ReferrerInfo{ - Digest: "goodManifest", - MediaType: ispec.MediaTypeImageManifest, - ArtifactType: "wantedType", - }) + err = metaDB.SetRepoReference(repo, referrerWantedType.DigestStr(), referrerWantedType.AsImageData()) + So(err, ShouldBeNil) + err = metaDB.SetRepoReference(repo, referrerNotWantedType.DigestStr(), referrerNotWantedType.AsImageData()) So(err, ShouldBeNil) - referrerInfo, err := metaDB.GetReferrersInfo("repo", referredDigest, []string{"wantedType"}) + referrerInfo, err := metaDB.GetReferrersInfo("repo", image.Digest(), []string{"wanted-type"}) So(err, ShouldBeNil) So(len(referrerInfo), ShouldEqual, 1) - So(referrerInfo[0].ArtifactType, ShouldResemble, "wantedType") - So(referrerInfo[0].Digest, ShouldResemble, "goodManifest") + So(referrerInfo[0].ArtifactType, ShouldResemble, "wanted-type") + So(referrerInfo[0].Digest, ShouldResemble, referrerWantedType.DigestStr()) }) - Convey("FilterRepos", func() { - img, err := deprecated.GetRandomImage() //nolint:staticcheck - So(err, ShouldBeNil) - imgDigest := img.Digest() + Convey("FilterImageData", func() { + repo := "repo" + tag := "tag" - manifestData, err := NewManifestData(img.Manifest, img.Config) - So(err, ShouldBeNil) + Convey("Just manifests", func() { + image := CreateRandomImage() + err := metaDB.SetRepoReference(repo, tag, image.AsImageData()) + So(err, ShouldBeNil) - err = metaDB.SetManifestData(imgDigest, manifestData) - So(err, ShouldBeNil) + imageData, err := metaDB.FilterImageData(ctx, []string{image.DigestStr()}) + So(err, ShouldBeNil) + So(imageData, ShouldContainKey, image.DigestStr()) - multiarch, err := deprecated.GetRandomMultiarchImage("multi") //nolint:staticcheck + _, err = metaDB.FilterImageData(ctx, []string{image.DigestStr(), "bad-digest"}) + So(err, ShouldNotBeNil) + }) + + Convey("Index", func() { + multi := CreateRandomMultiarch() + digests := []string{} + + for i := range multi.Images { + err := metaDB.SetRepoReference(repo, multi.Images[i].DigestStr(), multi.Images[i].AsImageData()) + So(err, ShouldBeNil) + + digests = append(digests, multi.Images[i].DigestStr()) + } + + err := metaDB.SetRepoReference(repo, tag, multi.AsImageData()) + So(err, ShouldBeNil) + + imageData, err := metaDB.FilterImageData(ctx, []string{multi.DigestStr()}) + So(err, ShouldBeNil) + So(imageData, ShouldContainKey, multi.DigestStr()) + + actualDigests := tCommon.AccumulateField(imageData[multi.DigestStr()].Manifests, getManifestDigest) + So(tCommon.ContainSameElements(actualDigests, digests), ShouldBeTrue) + }) + }) + + Convey("ResetDB", func() { + repo := "repo-reset" + tag := "tag-reset" + + image := CreateRandomImage() + referrer := CreateRandomImageWith().Subject(image.DescriptorRef()).Build() + err := metaDB.SetRepoReference(repo, tag, image.AsImageData()) + So(err, ShouldBeNil) + err = metaDB.SetRepoReference(repo, tag, referrer.AsImageData()) So(err, ShouldBeNil) - multiarchDigest := multiarch.Digest() - indexData, err := NewIndexData(multiarch.Index) + repoMeta, err := metaDB.GetRepoMeta(ctx, repo) So(err, ShouldBeNil) + So(repoMeta.Tags, ShouldNotBeEmpty) + So(repoMeta.Statistics, ShouldNotBeEmpty) + So(repoMeta.Referrers, ShouldNotBeEmpty) - err = metaDB.SetIndexData(multiarchDigest, indexData) + err = metaDB.ResetDB() So(err, ShouldBeNil) - for _, img := range multiarch.Images { - digest := img.Digest() + _, err = metaDB.GetRepoMeta(ctx, repo) + So(err, ShouldNotBeNil) + }) - indManData1, err := NewManifestData(multiarch.Images[0].Manifest, multiarch.Images[0].Config) - So(err, ShouldBeNil) + Convey("FilterRepos", func() { + repo := "repoFilter" + tag1 := "tag1" + tag2 := "tag2" + + image := CreateImageWith().DefaultLayers().PlatformConfig("image-platform", "image-os").Build() + err := metaDB.SetRepoReference(repo, tag1, image.AsImageData()) + So(err, ShouldBeNil) + + multiarch := CreateMultiarchWith(). + Images([]Image{ + CreateImageWith().DefaultLayers().PlatformConfig("sub-platform1", "sub-os1").Build(), + CreateImageWith().DefaultLayers().PlatformConfig("sub-platform2", "sub-os2").Build(), + }).Build() - err = metaDB.SetManifestData(digest, indManData1) + for _, img := range multiarch.Images { + err := metaDB.SetRepoReference(repo, img.DigestStr(), img.AsImageData()) So(err, ShouldBeNil) } - err = metaDB.SetRepoReference("repo", img.DigestStr(), imgDigest, img.Manifest.MediaType) - So(err, ShouldBeNil) - - err = metaDB.SetRepoReference("repo", multiarch.DigestStr(), multiarchDigest, ispec.MediaTypeImageIndex) + err = metaDB.SetRepoReference(repo, tag2, multiarch.AsImageData()) So(err, ShouldBeNil) - repoMetas, _, _, err := metaDB.FilterRepos(context.Background(), - func(repoMeta mTypes.RepoMetadata) bool { return true }) + repoMetaList, err := metaDB.FilterRepos(context.Background(), mTypes.AcceptAllRepoNames, + mTypes.AcceptAllRepoMeta) So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) + So(len(repoMetaList), ShouldEqual, 1) + repoMeta := repoMetaList[0] + So(repoMeta.Platforms, ShouldContain, ispec.Platform{Architecture: "image-platform", OS: "image-os"}) + So(repoMeta.Platforms, ShouldContain, ispec.Platform{Architecture: "sub-platform1", OS: "sub-os1"}) + So(repoMeta.Platforms, ShouldContain, ispec.Platform{Architecture: "sub-platform2", OS: "sub-os2"}) }) Convey("Test bookmarked/starred field present in returned RepoMeta", func() { @@ -2486,42 +2170,34 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func ctx := userAc.DeriveContext(context.Background()) - manifestDigest := godigest.FromString("dig") - err := metaDB.SetManifestData(manifestDigest, mTypes.ManifestData{ - ManifestBlob: []byte("{}"), - ConfigBlob: []byte("{}"), - }) - So(err, ShouldBeNil) + image := CreateRandomImage() - err = metaDB.SetRepoReference(repo99, "tag", manifestDigest, ispec.MediaTypeImageManifest) + err := metaDB.SetRepoReference(repo99, "tag", image.AsImageData()) So(err, ShouldBeNil) - repoMetas, _, _, err := metaDB.SearchRepos(ctx, repo99) + repoMetaList, err := metaDB.SearchRepos(ctx, repo99) So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeFalse) - So(repoMetas[0].IsStarred, ShouldBeFalse) + So(len(repoMetaList), ShouldEqual, 1) + So(repoMetaList[0].IsBookmarked, ShouldBeFalse) + So(repoMetaList[0].IsStarred, ShouldBeFalse) - repoMetas, _, _, err = metaDB.SearchTags(ctx, repo99+":") + imageDataList, err := metaDB.SearchTags(ctx, repo99+":") So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeFalse) - So(repoMetas[0].IsStarred, ShouldBeFalse) + So(len(imageDataList), ShouldEqual, 1) + So(imageDataList[0].IsBookmarked, ShouldBeFalse) + So(imageDataList[0].IsStarred, ShouldBeFalse) - repoMetas, _, _, err = metaDB.FilterRepos(ctx, func(repoMeta mTypes.RepoMetadata) bool { - return true - }) + repoMetaList, err = metaDB.FilterRepos(ctx, mTypes.AcceptAllRepoNames, mTypes.AcceptAllRepoMeta) So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeFalse) - So(repoMetas[0].IsStarred, ShouldBeFalse) + So(len(repoMetaList), ShouldEqual, 1) + So(repoMetaList[0].IsBookmarked, ShouldBeFalse) + So(repoMetaList[0].IsStarred, ShouldBeFalse) - repoMetas, _, _, err = metaDB.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) + imageDataList, err = metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, mTypes.AcceptAllImageData) So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeFalse) - So(repoMetas[0].IsStarred, ShouldBeFalse) + So(len(imageDataList), ShouldEqual, 1) + So(imageDataList[0].IsBookmarked, ShouldBeFalse) + So(imageDataList[0].IsStarred, ShouldBeFalse) _, err = metaDB.ToggleBookmarkRepo(ctx, repo99) So(err, ShouldBeNil) @@ -2529,35 +2205,35 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func _, err = metaDB.ToggleStarRepo(ctx, repo99) So(err, ShouldBeNil) - repoMetas, _, _, err = metaDB.SearchRepos(ctx, repo99) + repoMetaList, err = metaDB.SearchRepos(ctx, repo99) So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeTrue) - So(repoMetas[0].IsStarred, ShouldBeTrue) + So(len(repoMetaList), ShouldEqual, 1) + So(repoMetaList[0].IsBookmarked, ShouldBeTrue) + So(repoMetaList[0].IsStarred, ShouldBeTrue) - repoMetas, _, _, err = metaDB.SearchTags(ctx, repo99+":") + imageDataList, err = metaDB.SearchTags(ctx, repo99+":") So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeTrue) - So(repoMetas[0].IsStarred, ShouldBeTrue) + So(len(imageDataList), ShouldEqual, 1) + So(imageDataList[0].IsBookmarked, ShouldBeTrue) + So(imageDataList[0].IsStarred, ShouldBeTrue) - repoMetas, _, _, err = metaDB.FilterRepos(ctx, func(repoMeta mTypes.RepoMetadata) bool { - return true - }) + repoMetaList, err = metaDB.FilterRepos(ctx, mTypes.AcceptAllRepoNames, mTypes.AcceptAllRepoMeta) So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeTrue) - So(repoMetas[0].IsStarred, ShouldBeTrue) + So(len(repoMetaList), ShouldEqual, 1) + So(repoMetaList[0].IsBookmarked, ShouldBeTrue) + So(repoMetaList[0].IsStarred, ShouldBeTrue) - repoMetas, _, _, err = metaDB.FilterTags(ctx, - func(repoMeta mTypes.RepoMetadata, manifestMeta mTypes.ManifestMetadata) bool { return true }) + imageDataList, err = metaDB.FilterTags(ctx, mTypes.AcceptAllRepoTag, mTypes.AcceptAllImageData) So(err, ShouldBeNil) - So(len(repoMetas), ShouldEqual, 1) - So(repoMetas[0].IsBookmarked, ShouldBeTrue) - So(repoMetas[0].IsStarred, ShouldBeTrue) + So(len(imageDataList), ShouldEqual, 1) + So(imageDataList[0].IsBookmarked, ShouldBeTrue) + So(imageDataList[0].IsStarred, ShouldBeTrue) }) Convey("Test GetUserRepoMeta", func() { + err := metaDB.ResetDB() + So(err, ShouldBeNil) + userAc := reqCtx.NewUserAccessControl() userAc.SetUsername("user1") userAc.SetGlobPatterns("read", map[string]bool{ @@ -2566,9 +2242,7 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func ctx := userAc.DeriveContext(context.Background()) - digest := godigest.FromString("1") - - err := metaDB.SetRepoReference("repo", "tag", digest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference("repo", "tag", CreateDefaultImage().AsImageData()) So(err, ShouldBeNil) _, err = metaDB.ToggleBookmarkRepo(ctx, "repo") @@ -2577,7 +2251,7 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func _, err = metaDB.ToggleStarRepo(ctx, "repo") So(err, ShouldBeNil) - repoMeta, err := metaDB.GetUserRepoMeta(ctx, "repo") + repoMeta, err := metaDB.GetRepoMeta(ctx, "repo") So(err, ShouldBeNil) So(repoMeta.IsBookmarked, ShouldBeTrue) So(repoMeta.IsStarred, ShouldBeTrue) @@ -2586,28 +2260,6 @@ func RunMetaDBTests(t *testing.T, metaDB mTypes.MetaDB, preparationFuncs ...func }) } -func NewManifestData(manifest ispec.Manifest, config ispec.Image) (mTypes.ManifestData, error) { - configBlob, err := json.Marshal(config) - if err != nil { - return mTypes.ManifestData{}, err - } - - manifest.Config.Digest = godigest.FromBytes(configBlob) - - manifestBlob, err := json.Marshal(manifest) - if err != nil { - return mTypes.ManifestData{}, err - } - - return mTypes.ManifestData{ManifestBlob: manifestBlob, ConfigBlob: configBlob}, nil -} - -func NewIndexData(index ispec.Index) (mTypes.IndexData, error) { - indexBlob, err := json.Marshal(index) - - return mTypes.IndexData{IndexBlob: indexBlob}, err -} - func TestRelevanceSorting(t *testing.T) { Convey("Test Relevance Sorting", t, func() { So(common.RankRepoName("alpine", "alpine"), ShouldEqual, 0) @@ -2629,78 +2281,19 @@ func TestRelevanceSorting(t *testing.T) { }) } -func generateTestImage() ([]byte, []byte, error) { - config := ispec.Image{ - Platform: ispec.Platform{ - Architecture: "amd64", - OS: LINUX, - }, - RootFS: ispec.RootFS{ - Type: "layers", - DiffIDs: []godigest.Digest{}, - }, - Author: "ZotUser", - } - - configBlob, err := json.Marshal(config) - if err != nil { - return []byte{}, []byte{}, err - } - - configDigest := godigest.FromBytes(configBlob) - - layers := [][]byte{ - make([]byte, 100), - } - - // init layers with random values - for i := range layers { - //nolint:gosec - _, err := rand.Read(layers[i]) //nolint:staticcheck - if err != nil { - return []byte{}, []byte{}, err - } - } - - manifest := ispec.Manifest{ - Versioned: specs.Versioned{ - SchemaVersion: 2, - }, - Config: ispec.Descriptor{ - MediaType: "application/vnd.oci.image.config.v1+json", - Digest: configDigest, - Size: int64(len(configBlob)), - }, - Layers: []ispec.Descriptor{ - { - MediaType: "application/vnd.oci.image.layer.v1.tar", - Digest: godigest.FromBytes(layers[0]), - Size: int64(len(layers[0])), - }, - }, - } - - manifestBlob, err := json.Marshal(manifest) - if err != nil { - return []byte{}, []byte{}, err - } - - return configBlob, manifestBlob, nil -} - func TestCreateDynamo(t *testing.T) { tskip.SkipDynamo(t) Convey("Create", t, func() { dynamoDBDriverParams := mdynamodb.DBDriverParameters{ - Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), - RepoMetaTablename: "RepoMetadataTable", - ManifestDataTablename: "ManifestDataTable", - IndexDataTablename: "IndexDataTable", - UserDataTablename: "UserDataTable", - APIKeyTablename: "ApiKeyTable", - VersionTablename: "Version", - Region: "us-east-2", + Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), + RepoMetaTablename: "RepoMetadataTable", + RepoBlobsInfoTablename: "RepoBlobs", + ImageDataTablename: "ImageData", + UserDataTablename: "UserDataTable", + APIKeyTablename: "ApiKeyTable", + VersionTablename: "Version", + Region: "us-east-2", } client, err := mdynamodb.GetDynamoClient(dynamoDBDriverParams) diff --git a/pkg/meta/parse.go b/pkg/meta/parse.go index 9984d50ec5..a1ab4688d1 100644 --- a/pkg/meta/parse.go +++ b/pkg/meta/parse.go @@ -4,7 +4,6 @@ import ( "encoding/json" "errors" "fmt" - "regexp" "time" godigest "github.com/opencontainers/go-digest" @@ -19,6 +18,11 @@ import ( storageTypes "zotregistry.io/zot/pkg/storage/types" ) +const ( + CosignType = "cosign" + NotationType = "notation" +) + // ParseStorage will sync all repos found in the rootdirectory of the oci layout that zot was deployed on with the // ParseStorage database. func ParseStorage(metaDB mTypes.MetaDB, storeController storage.StoreController, log log.Logger) error { @@ -33,7 +37,10 @@ func ParseStorage(metaDB mTypes.MetaDB, storeController storage.StoreController, return err } - for _, repo := range allRepos { + for i, repo := range allRepos { + log.Info().Int("total", len(allRepos)).Int("progress", i).Str("current-repo", repo). + Msgf("parsing next repo '%s'", repo) + err := ParseRepo(repo, metaDB, storeController, log) if err != nil { log.Error().Err(err).Str("repository", repo).Msg("load-local-layout: failed to sync repo") @@ -72,24 +79,23 @@ func ParseRepo(repo string, metaDB mTypes.MetaDB, storeController storage.StoreC return err } - err = resetRepoMeta(repo, metaDB, log) + err = metaDB.ResetRepoReferences(repo) if err != nil && !errors.Is(err, zerr.ErrRepoMetaNotFound) { log.Error().Err(err).Str("repository", repo).Msg("load-repo: failed to reset tag field in RepoMetadata for repo") return err } - for _, descriptor := range indexContent.Manifests { - tag := descriptor.Annotations[ispec.AnnotationRefName] + for _, manifest := range indexContent.Manifests { + tag := manifest.Annotations[ispec.AnnotationRefName] if zcommon.IsReferrersTag(tag) { continue } - // TODO take info from RepoDB if it exists already - descriptorBlob, _, _, err := imageStore.GetImageManifest(repo, descriptor.Digest.String()) + manifestBlob, _, _, err := imageStore.GetImageManifest(repo, manifest.Digest.String()) if err != nil { - log.Error().Err(err).Str("repository", repo).Str("digest", descriptor.Digest.String()). + log.Error().Err(err).Str("repository", repo).Str("digest", manifest.Digest.String()). Msg("load-repo: failed to get blob for image") return err @@ -98,10 +104,10 @@ func ParseRepo(repo string, metaDB mTypes.MetaDB, storeController storage.StoreC reference := tag if tag == "" { - reference = descriptor.Digest.String() + reference = manifest.Digest.String() } - err = ProtoSetImageMetaFromInput(repo, reference, descriptor.MediaType, descriptor.Digest, descriptorBlob, + err = SetImageMetaFromInput(repo, reference, manifest.MediaType, manifest.Digest, manifestBlob, imageStore, metaDB, log) if err != nil { log.Error().Err(err).Str("repository", repo).Str("tag", tag). @@ -114,32 +120,6 @@ func ParseRepo(repo string, metaDB mTypes.MetaDB, storeController storage.StoreC return nil } -// resetRepoMeta will delete all tags and non-user related information from a RepoMetadata. -// It is used to recalculate and keep MetaDB consistent with the layout in case of unexpected changes. -func resetRepoMeta(repo string, metaDB mTypes.MetaDB, log log.Logger) error { - repoMeta, err := metaDB.ProtoGetRepoMeta(repo) - if err != nil && !errors.Is(err, zerr.ErrRepoMetaNotFound) { - log.Error().Err(err).Str("repository", repo).Msg("load-repo: failed to get RepoMeta for repo") - - return err - } - - if errors.Is(err, zerr.ErrRepoMetaNotFound) { - log.Info().Str("repository", repo).Msg("load-repo: RepoMeta not found for repo, new RepoMeta will be created") - - return nil - } - - return metaDB.ProtoSetRepoMeta(repo, mTypes.RepoMetadata2{ - Name: repoMeta.Name, - Tags: map[string]mTypes.Descriptor{"": {}}, - Statistics: repoMeta.Statistics, - Signatures: map[string]mTypes.ManifestSignatures{"": {"": {}}}, - Referrers: map[string][]mTypes.ReferrerInfo{"": {}}, - Stars: repoMeta.Stars, - }) -} - func getAllRepos(storeController storage.StoreController) ([]string, error) { allRepos, err := storeController.DefaultStore.GetRepositories() if err != nil { @@ -160,43 +140,6 @@ func getAllRepos(storeController storage.StoreController) ([]string, error) { return allRepos, nil } -func getCachedBlob(repo string, descriptor ispec.Descriptor, metaDB mTypes.MetaDB, - imageStore storageTypes.ImageStore, log log.Logger, -) ([]byte, error) { - digest := descriptor.Digest - - descriptorBlob, err := getCachedBlobFromMetaDB(descriptor, metaDB) - - if err != nil || len(descriptorBlob) == 0 { - descriptorBlob, _, _, err = imageStore.GetImageManifest(repo, digest.String()) - if err != nil { - log.Error().Err(err).Str("repository", repo).Str("digest", digest.String()). - Msg("load-repo: failed to get blob for image") - - return nil, err - } - - return descriptorBlob, nil - } - - return descriptorBlob, nil -} - -func getCachedBlobFromMetaDB(descriptor ispec.Descriptor, metaDB mTypes.MetaDB) ([]byte, error) { - switch descriptor.MediaType { - case ispec.MediaTypeImageManifest: - manifestData, err := metaDB.GetManifestData(descriptor.Digest) - - return manifestData.ManifestBlob, err - case ispec.MediaTypeImageIndex: - indexData, err := metaDB.GetIndexData(descriptor.Digest) - - return indexData.IndexBlob, err - } - - return nil, nil -} - func GetSignatureLayersInfo(repo, tag, manifestDigest, signatureType string, manifestBlob []byte, imageStore storageTypes.ImageStore, log log.Logger, ) ([]mTypes.LayerInfo, error) { @@ -304,136 +247,43 @@ func getNotationSignatureLayersInfo( return layers, nil } -// NewManifestMeta takes raw data about an image and createa a new ManifestMetadate object. -func NewManifestData(repoName string, manifestBlob []byte, imageStore storageTypes.ImageStore, -) (mTypes.ManifestData, error) { - var ( - manifestContent ispec.Manifest - configContent ispec.Image - manifestData mTypes.ManifestData - ) - - err := json.Unmarshal(manifestBlob, &manifestContent) - if err != nil { - return mTypes.ManifestData{}, err - } - - var lockLatency time.Time - - imageStore.RLock(&lockLatency) - defer imageStore.RUnlock(&lockLatency) - - configBlob, err := imageStore.GetBlobContent(repoName, manifestContent.Config.Digest) - if err != nil { - return mTypes.ManifestData{}, err - } - - if manifestContent.Config.MediaType == ispec.MediaTypeImageConfig { - err = json.Unmarshal(configBlob, &configContent) - if err != nil { - return mTypes.ManifestData{}, err - } - } - - manifestData.ManifestBlob = manifestBlob - manifestData.ConfigBlob = configBlob - - return manifestData, nil -} - -func NewIndexData(repoName string, indexBlob []byte, imageStore storageTypes.ImageStore, -) mTypes.IndexData { - indexData := mTypes.IndexData{} - - indexData.IndexBlob = indexBlob - - return indexData -} - -// SetMetadataFromInput tries to set manifest metadata and update repo metadata by adding the current tag -// (in case the reference is a tag). The function expects image manifests and indexes (multi arch images). -func SetImageMetaFromInput(repo, reference, mediaType string, digest godigest.Digest, descriptorBlob []byte, - imageStore storageTypes.ImageStore, metaDB mTypes.MetaDB, log log.Logger, -) error { - switch mediaType { - case ispec.MediaTypeImageManifest: - imageData, err := NewManifestData(repo, descriptorBlob, imageStore) - if err != nil { - return err - } - - err = metaDB.SetManifestData(digest, imageData) - if err != nil { - log.Error().Err(err).Msg("metadb: error while putting manifest meta") - - return err - } - case ispec.MediaTypeImageIndex: - indexData := NewIndexData(repo, descriptorBlob, imageStore) - - err := metaDB.SetIndexData(digest, indexData) - if err != nil { - log.Error().Err(err).Msg("metadb: error while putting index data") - - return err - } - } - - referredDigest, referrerInfo, hasSubject, err := GetReferredInfo(descriptorBlob, digest.String(), mediaType) - if hasSubject && err == nil { - err := metaDB.SetReferrer(repo, referredDigest, referrerInfo) - if err != nil { - log.Error().Err(err).Msg("metadb: error while settingg referrer") - - return err - } - } - - err = metaDB.SetRepoReference(repo, reference, digest, mediaType) - if err != nil { - log.Error().Err(err).Msg("metadb: error while putting repo meta") - - return err - } - - return nil -} - // SetMetadataFromInput tries to set manifest metadata and update repo metadata by adding the current tag // (in case the reference is a tag). The function expects image manifests and indexes (multi arch images). -func ProtoSetImageMetaFromInput(repo, reference, mediaType string, digest godigest.Digest, descriptorBlob []byte, +func SetImageMetaFromInput(repo, reference, mediaType string, digest godigest.Digest, blob []byte, imageStore storageTypes.ImageStore, metaDB mTypes.MetaDB, log log.Logger, ) error { - imageData := mTypes.ImageData2{} + var imageData mTypes.ImageData switch mediaType { case ispec.MediaTypeImageManifest: manifestContent := ispec.Manifest{} configContent := ispec.Image{} - err := json.Unmarshal(descriptorBlob, &manifestContent) + err := json.Unmarshal(blob, &manifestContent) if err != nil { return err } - configBlob, err := imageStore.GetBlobContent(repo, manifestContent.Config.Digest) - if err != nil { - return err - } + if manifestContent.Config.MediaType == ispec.MediaTypeImageConfig { + configBlob, err := imageStore.GetBlobContent(repo, manifestContent.Config.Digest) + if err != nil { + return err + } - err = json.Unmarshal(configBlob, &configContent) - if err != nil { - return err + err = json.Unmarshal(configBlob, &configContent) + if err != nil { + return err + } } - if isSig, sigType, signedManifestDigest := isSingature(reference, manifestContent); isSig { + if isSig, sigType, signedManifestDigest := isSignature(reference, manifestContent); isSig { layers, err := GetSignatureLayersInfo(repo, reference, digest.String(), sigType, - descriptorBlob, imageStore, log) + blob, imageStore, log) if err != nil { return err } - err = metaDB.ProtoAddManifestSignature(repo, signedManifestDigest, + err = metaDB.AddManifestSignature(repo, signedManifestDigest, mTypes.SignatureMetadata{ SignatureType: sigType, SignatureDigest: digest.String(), @@ -447,10 +297,10 @@ func ProtoSetImageMetaFromInput(repo, reference, mediaType string, digest godige return err } - err = metaDB.ProtoUpdateSignaturesValidity(repo, signedManifestDigest) + err = metaDB.UpdateSignaturesValidity(repo, signedManifestDigest) if err != nil { - log.Error().Err(err).Str("repository", repo).Str("reference", reference).Str("digest", signedManifestDigest.String()).Msg( - "load-repo: failed verify signatures validity for signed image") + log.Error().Err(err).Str("repository", repo).Str("reference", reference).Str("digest", + signedManifestDigest.String()).Msg("load-repo: failed verify signatures validity for signed image") return err } @@ -458,19 +308,21 @@ func ProtoSetImageMetaFromInput(repo, reference, mediaType string, digest godige return nil } - imageData = convert.GetImageManifestData(manifestContent, configContent, int64(len(descriptorBlob)), digest) + imageData = convert.GetImageManifestData(manifestContent, configContent, int64(len(blob)), digest) case ispec.MediaTypeImageIndex: indexContent := ispec.Index{} - err := json.Unmarshal(descriptorBlob, &indexContent) + err := json.Unmarshal(blob, &indexContent) if err != nil { return err } - imageData = convert.GetImageIndexData(indexContent, int64(len(descriptorBlob)), digest) + imageData = convert.GetImageIndexData(indexContent, int64(len(blob)), digest) + default: + return nil } - err := metaDB.ProtoSetRepoReference(repo, reference, imageData) + err := metaDB.SetRepoReference(repo, reference, imageData) if err != nil { log.Error().Err(err).Msg("metadb: error while putting repo meta") @@ -480,12 +332,7 @@ func ProtoSetImageMetaFromInput(repo, reference, mediaType string, digest godige return nil } -const ( - CosignType = "cosign" - NotationType = "notation" -) - -func isSingature(reference string, manifestContent ispec.Manifest) (bool, string, godigest.Digest) { +func isSignature(reference string, manifestContent ispec.Manifest) (bool, string, godigest.Digest) { manifestArtifactType := zcommon.GetManifestArtifactType(manifestContent) // check notation signature @@ -493,10 +340,7 @@ func isSingature(reference string, manifestContent ispec.Manifest) (bool, string return true, NotationType, manifestContent.Subject.Digest } - // check cosign - cosignTagRule := regexp.MustCompile(`sha256\-.+\.sig`) - - if tag := reference; cosignTagRule.MatchString(reference) { + if tag := reference; zcommon.IsCosignTag(reference) { prefixLen := len("sha256-") digestLen := 64 signedImageManifestDigestEncoded := tag[prefixLen : prefixLen+digestLen] @@ -562,9 +406,3 @@ func GetReferredInfo(descriptorBlob []byte, referrerDigest, mediaType string, return referrerSubject.Digest, referrerInfo, true, nil } - -func ref[T any](input T) *T { - ref := input - - return &ref -} diff --git a/pkg/meta/parse_test.go b/pkg/meta/parse_test.go index 103d847e42..b290bdb965 100644 --- a/pkg/meta/parse_test.go +++ b/pkg/meta/parse_test.go @@ -13,7 +13,6 @@ import ( ispec "github.com/opencontainers/image-spec/specs-go/v1" . "github.com/smartystreets/goconvey/convey" - zerr "zotregistry.io/zot/errors" zcommon "zotregistry.io/zot/pkg/common" "zotregistry.io/zot/pkg/extensions/monitoring" "zotregistry.io/zot/pkg/log" @@ -97,50 +96,6 @@ func TestParseStorageErrors(t *testing.T) { So(err, ShouldNotBeNil) }) - Convey("resetRepoMetaTags errors", func() { - imageStore.GetIndexContentFn = func(repo string) ([]byte, error) { - return []byte("{}"), nil - } - - Convey("metaDB.GetRepoMeta errors", func() { - metaDB.GetRepoMetaFn = func(repo string) (mTypes.RepoMetadata, error) { - return mTypes.RepoMetadata{}, ErrTestError - } - - err := meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - }) - }) - - Convey("isManifestMetaPresent errors", func() { - indexContent := ispec.Index{ - Manifests: []ispec.Descriptor{ - { - Digest: godigest.FromString("manifest1"), - MediaType: ispec.MediaTypeImageManifest, - Annotations: map[string]string{ - ispec.AnnotationRefName: "tag1", - }, - }, - }, - } - indexBlob, err := json.Marshal(indexContent) - So(err, ShouldBeNil) - - imageStore.GetIndexContentFn = func(repo string) ([]byte, error) { - return indexBlob, nil - } - - Convey("metaDB.GetManifestMeta errors", func() { - metaDB.GetManifestMetaFn = func(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - return mTypes.ManifestMetadata{}, ErrTestError - } - - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - }) - }) - Convey("manifestMetaIsPresent true", func() { indexContent := ispec.Index{ Manifests: []ispec.Descriptor{ @@ -161,7 +116,7 @@ func TestParseStorageErrors(t *testing.T) { } Convey("metaDB.SetRepoReference", func() { - metaDB.SetRepoReferenceFn = func(repo, tag string, manifestDigest godigest.Digest, mediaType string) error { + metaDB.SetRepoReferenceFn = func(repo, reference string, imageData mTypes.ImageData) error { return ErrTestError } @@ -169,212 +124,6 @@ func TestParseStorageErrors(t *testing.T) { So(err, ShouldNotBeNil) }) }) - - Convey("manifestMetaIsPresent false", func() { - indexContent := ispec.Index{ - Manifests: []ispec.Descriptor{ - { - Digest: godigest.FromString("manifest1"), - MediaType: ispec.MediaTypeImageManifest, - Annotations: map[string]string{ - ispec.AnnotationRefName: "tag1", - }, - }, - }, - } - indexBlob, err := json.Marshal(indexContent) - So(err, ShouldBeNil) - - imageStore.GetIndexContentFn = func(repo string) ([]byte, error) { - return indexBlob, nil - } - - metaDB.GetManifestMetaFn = func(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - return mTypes.ManifestMetadata{}, zerr.ErrManifestMetaNotFound - } - - Convey("GetImageManifest errors", func() { - imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) { - return nil, "", "", ErrTestError - } - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - }) - - Convey("CheckIsImageSignature errors", func() { - // CheckIsImageSignature will fail because of a invalid json - imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) { - return []byte("Invalid JSON"), "", "", nil - } - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - }) - Convey("CheckIsImageSignature -> not signature", func() { - 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 - } - - Convey("imgStore.GetBlobContent errors", func() { - imageStore.GetBlobContentFn = func(repo string, digest godigest.Digest) ([]byte, error) { - return nil, ErrTestError - } - - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - }) - }) - - Convey("CheckIsImageSignature -> is signature", func() { - manifestContent := ispec.Manifest{ - Subject: &ispec.Descriptor{ - Digest: "123", - }, - ArtifactType: "application/vnd.cncf.notary.signature", - Layers: []ispec.Descriptor{{MediaType: ispec.MediaTypeImageLayer}}, - } - - manifestBlob, err := json.Marshal(manifestContent) - So(err, ShouldBeNil) - - imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) { - return manifestBlob, "", "", nil - } - - metaDB.AddManifestSignatureFn = func(repo string, signedManifestDigest godigest.Digest, - sm mTypes.SignatureMetadata, - ) error { - return ErrTestError - } - - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - - metaDB.AddManifestSignatureFn = func(repo string, signedManifestDigest godigest.Digest, - sm mTypes.SignatureMetadata, - ) error { - return nil - } - - metaDB.UpdateSignaturesValidityFn = func(repo string, signedManifestDigest godigest.Digest, - ) error { - return ErrTestError - } - - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - }) - - Convey("GetSignatureLayersInfo errors", func() { - // get notation signature layers info - badNotationManifestContent := ispec.Manifest{ - Subject: &ispec.Descriptor{ - Digest: "123", - }, - ArtifactType: "application/vnd.cncf.notary.signature", - } - - badNotationManifestBlob, err := json.Marshal(badNotationManifestContent) - So(err, ShouldBeNil) - - imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) { - return badNotationManifestBlob, "", "", nil - } - - // wrong number of layers of notation manifest - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - - notationManifestContent := ispec.Manifest{ - Subject: &ispec.Descriptor{ - Digest: "123", - }, - ArtifactType: "application/vnd.cncf.notary.signature", - Layers: []ispec.Descriptor{{MediaType: ispec.MediaTypeImageLayer}}, - } - - 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{}, ErrTestError - } - - // unable to get layer content - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - - _, _, cosignManifestContent, _ := deprecated.GetRandomImageComponents(10) //nolint:staticcheck - _, _, signedManifest, _ := deprecated.GetRandomImageComponents(10) //nolint:staticcheck - signatureTag, err := signature.GetCosignSignatureTagForManifest(signedManifest) - So(err, ShouldBeNil) - - cosignManifestContent.Annotations = map[string]string{ispec.AnnotationRefName: signatureTag} - - cosignManifestBlob, err := json.Marshal(cosignManifestContent) - So(err, ShouldBeNil) - - imageStore.GetImageManifestFn = func(repo, reference string) ([]byte, godigest.Digest, string, error) { - return cosignManifestBlob, "", "", nil - } - - indexContent := ispec.Index{ - Manifests: []ispec.Descriptor{ - { - Digest: godigest.FromString("cosignSig"), - MediaType: ispec.MediaTypeImageManifest, - Annotations: map[string]string{ - ispec.AnnotationRefName: signatureTag, - }, - }, - }, - } - indexBlob, err := json.Marshal(indexContent) - So(err, ShouldBeNil) - - imageStore.GetIndexContentFn = func(repo string) ([]byte, error) { - return indexBlob, nil - } - - // unable to get layer content - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldNotBeNil) - }) - - Convey("IsReferrersTag -> true", func() { - indexContent := ispec.Index{ - Manifests: []ispec.Descriptor{ - { - Digest: godigest.FromString("indx1"), - MediaType: ispec.MediaTypeImageIndex, - Annotations: map[string]string{ - ispec.AnnotationRefName: "sha256-123", - }, - }, - }, - } - indexBlob, err := json.Marshal(indexContent) - So(err, ShouldBeNil) - - imageStore.GetIndexContentFn = func(repo string) ([]byte, error) { - return indexBlob, nil - } - - metaDB.SetIndexDataFn = func(digest godigest.Digest, indexData mTypes.IndexData) error { - return ErrTestError - } - - err = meta.ParseRepo("repo", metaDB, storeController, log) - So(err, ShouldBeNil) - }) - }) }) } @@ -401,14 +150,14 @@ func TestParseStorageDynamoWrapper(t *testing.T) { rootDir := t.TempDir() params := dynamodb.DBDriverParameters{ - Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), - Region: "us-east-2", - RepoMetaTablename: "RepoMetadataTable", - ManifestDataTablename: "ManifestDataTable", - IndexDataTablename: "IndexDataTable", - UserDataTablename: "UserDataTable", - APIKeyTablename: "ApiKeyTable", - VersionTablename: "Version", + Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), + Region: "us-east-2", + RepoMetaTablename: "RepoMetadataTable", + RepoBlobsInfoTablename: "RepoBlobsInfoTablename", + ImageDataTablename: "ImageDataTablename", + UserDataTablename: "UserDataTable", + APIKeyTablename: "ApiKeyTable", + VersionTablename: "Version", } dynamoClient, err := dynamodb.GetDynamoClient(params) @@ -417,10 +166,13 @@ func TestParseStorageDynamoWrapper(t *testing.T) { dynamoWrapper, err := dynamodb.New(dynamoClient, params, log.NewLogger("debug", "")) So(err, ShouldBeNil) - err = dynamoWrapper.ResetManifestDataTable() + err = dynamoWrapper.ResetTable(dynamoWrapper.RepoMetaTablename) So(err, ShouldBeNil) - err = dynamoWrapper.ResetRepoMetaTable() + err = dynamoWrapper.ResetTable(dynamoWrapper.RepoBlobsTablename) + So(err, ShouldBeNil) + + err = dynamoWrapper.ResetTable(dynamoWrapper.ImageDataTablename) So(err, ShouldBeNil) RunParseStorageTests(rootDir, dynamoWrapper) @@ -496,20 +248,20 @@ func RunParseStorageTests(rootDir string, metaDB mTypes.MetaDB) { So(err, ShouldBeNil) repos, err := metaDB.GetMultipleRepoMeta(context.Background(), - func(repoMeta mTypes.RepoMetadata) bool { return true }) + func(repoMeta mTypes.RepoMeta) bool { return true }) So(err, ShouldBeNil) So(len(repos), ShouldEqual, 1) So(len(repos[0].Tags), ShouldEqual, 2) - for _, descriptor := range repos[0].Tags { - manifestMeta, err := metaDB.GetManifestMeta(repo, godigest.Digest(descriptor.Digest)) + ctx := context.Background() + + for tag, descriptor := range repos[0].Tags { + imageManifestData, err := metaDB.GetImageMeta(ctx, repo, tag) So(err, ShouldBeNil) - So(manifestMeta.ManifestBlob, ShouldNotBeNil) - So(manifestMeta.ConfigBlob, ShouldNotBeNil) if descriptor.Digest == signedManifestDigest.String() { - So(manifestMeta.Signatures, ShouldNotBeEmpty) + So(imageManifestData.Signatures, ShouldNotBeEmpty) } } }) @@ -555,10 +307,8 @@ func RunParseStorageTests(rootDir string, metaDB mTypes.MetaDB) { err = meta.ParseStorage(metaDB, storeController, log.NewLogger("debug", "")) So(err, ShouldBeNil) - repos, err := metaDB.GetMultipleRepoMeta( - context.Background(), - func(repoMeta mTypes.RepoMetadata) bool { return true }, - ) + repos, err := metaDB.GetMultipleRepoMeta(context.Background(), + func(repoMeta mTypes.RepoMeta) bool { return true }) So(err, ShouldBeNil) for _, desc := range repos[0].Tags { @@ -577,15 +327,12 @@ func RunParseStorageTests(rootDir string, metaDB mTypes.MetaDB) { storeController := storage.StoreController{DefaultStore: imageStore} // add an image - image, err := deprecated.GetRandomImage() //nolint:staticcheck - So(err, ShouldBeNil) - - manifestDigest := image.Digest() + image := CreateRandomImage() //nolint:staticcheck - err = WriteImageToFileSystem(image, repo, "tag", storeController) + err := WriteImageToFileSystem(image, repo, "tag", storeController) So(err, ShouldBeNil) - err = metaDB.SetRepoReference(repo, "tag", manifestDigest, ispec.MediaTypeImageManifest) + err = metaDB.SetRepoReference(repo, "tag", image.AsImageData()) So(err, ShouldBeNil) err = metaDB.IncrementRepoStars(repo) @@ -597,20 +344,20 @@ func RunParseStorageTests(rootDir string, metaDB mTypes.MetaDB) { err = metaDB.IncrementImageDownloads(repo, "tag") So(err, ShouldBeNil) - repoMeta, err := metaDB.GetRepoMeta(repo) + repoMeta, err := metaDB.GetRepoMeta(context.Background(), repo) So(err, ShouldBeNil) - So(repoMeta.Statistics[manifestDigest.String()].DownloadCount, ShouldEqual, 3) - So(repoMeta.Stars, ShouldEqual, 1) + So(repoMeta.Statistics[image.DigestStr()].DownloadCount, ShouldEqual, 3) + So(repoMeta.StarCount, ShouldEqual, 1) err = meta.ParseStorage(metaDB, storeController, log.NewLogger("debug", "")) So(err, ShouldBeNil) - repoMeta, err = metaDB.GetRepoMeta(repo) + repoMeta, err = metaDB.GetRepoMeta(context.Background(), repo) So(err, ShouldBeNil) - So(repoMeta.Statistics[manifestDigest.String()].DownloadCount, ShouldEqual, 3) - So(repoMeta.Stars, ShouldEqual, 1) + So(repoMeta.Statistics[image.DigestStr()].DownloadCount, ShouldEqual, 3) + So(repoMeta.StarCount, ShouldEqual, 1) }) } diff --git a/pkg/meta/proto/config.proto b/pkg/meta/proto/config.proto index 7d9c3f21b2..aff9271b95 100644 --- a/pkg/meta/proto/config.proto +++ b/pkg/meta/proto/config.proto @@ -6,41 +6,40 @@ import "descriptor.proto"; // https://github.com/opencontainers/image-spec/blob/main/specs-go/v1/config.go +message Image { + optional google.protobuf.Timestamp Created = 1; + optional string Author = 2; + Platform Platform = 3; + optional ImageConfig Config = 4; + optional RootFS RootFS = 5; + repeated History History = 6; +} + message ImageConfig { - string user = 1; - map exposedports = 2; - repeated string env = 3; - repeated string entrypoint = 4; - repeated string cmd = 5; - map volumes = 6; - optional string workingdir = 7; - map labels = 8; - optional string stopsignal = 9; - bool argsescaped = 10; + map ExposedPorts = 1; + map Volumes = 2; + map Labels = 3; + string User = 4; + repeated string Env = 5; + repeated string Entrypoint = 6; + repeated string Cmd = 7; + optional string WorkingDir = 8; + optional string StopSignal = 9; + bool ArgsEscaped = 10; } message RootFS { - string type = 1; - repeated string diffids = 2; + string Type = 1; + repeated string DiffIDs = 2; } message History { - optional google.protobuf.Timestamp created = 1; - optional string createdby = 2; - optional string author = 3; - optional string comment = 4; - optional bool emptylayer = 5; -} + optional google.protobuf.Timestamp Created = 1; -message ConfigData { - optional google.protobuf.Timestamp created = 1; - optional string author = 2; - Platform platform = 3; - optional ImageConfig config = 4; - optional RootFS rootfs = 5; - repeated History history = 6; - string Digest = 7; - int64 Size = 8; + optional string CreatedBy = 2; + optional string Author = 3; + optional string Comment = 4; + optional bool EmptyLayer = 5; } -message EmptyMessage{} +message EmptyMessage{} \ No newline at end of file diff --git a/pkg/meta/proto/descriptor.proto b/pkg/meta/proto/descriptor.proto index 6a4b5757eb..5e8cb0583b 100644 --- a/pkg/meta/proto/descriptor.proto +++ b/pkg/meta/proto/descriptor.proto @@ -4,17 +4,20 @@ package v1; // https://github.com/opencontainers/image-spec/blob/main/specs-go/v1/descriptor.go message Descriptor { - string mediaType = 1; - string digest = 2; - int64 size = 3; - repeated string urls = 4; - map annotations = 5; - bytes data = 6; - optional Platform platform = 7; - optional string artifactType = 8; + string MediaType = 1; + string Digest = 2; + int64 Size = 3; + repeated string URLs = 4; + bytes Data = 5; + optional Platform Platform = 6; + optional string ArtifactType = 7; + map Annotations = 8; } message Platform { - string Os = 1; - string Arch = 2; -} + string Architecture = 1; + string OS = 2; + optional string OSVersion = 3; + repeated string OSFeatures = 4; + optional string Variant = 5; +} \ No newline at end of file diff --git a/pkg/meta/proto/index.proto b/pkg/meta/proto/index.proto index 281cb84a5f..92fcf67f55 100644 --- a/pkg/meta/proto/index.proto +++ b/pkg/meta/proto/index.proto @@ -7,10 +7,10 @@ import "versioned.proto"; // https://github.com/opencontainers/image-spec/blob/main/specs-go/v1/index.go message Index { - Versioned versioned = 1; - optional string mediatype = 2; - optional string artifacttype = 3; - repeated Descriptor manifests = 4; - optional Descriptor subject = 5; - map annotations = 6; -} + Versioned Versioned = 1; + optional string MediaType = 2; + optional string ArtifactType = 3; + repeated Descriptor Manifests = 4; + optional Descriptor Subject = 5; + map Annotations = 6; +} \ No newline at end of file diff --git a/pkg/meta/proto/manifest.proto b/pkg/meta/proto/manifest.proto index 6d0e835c22..dbdce83efc 100644 --- a/pkg/meta/proto/manifest.proto +++ b/pkg/meta/proto/manifest.proto @@ -7,11 +7,11 @@ import "versioned.proto"; // https://github.com/opencontainers/image-spec/blob/main/specs-go/v1/manifest.go message Manifest { - Versioned versioned = 1; - optional string mediatype = 2; - optional string artifacttype = 3; - Descriptor config = 4; - repeated Descriptor layers = 5; - optional Descriptor subject = 6; - map annotations = 7; -} + Versioned Versioned = 1; + optional string MediaType = 2; + optional string ArtifactType = 3; + Descriptor Config = 4; + repeated Descriptor Layers = 5; + optional Descriptor Subject = 6; + map Annotations = 7; +} \ No newline at end of file diff --git a/pkg/meta/proto/imageData.proto b/pkg/meta/proto/meta.proto similarity index 50% rename from pkg/meta/proto/imageData.proto rename to pkg/meta/proto/meta.proto index 515194a469..e959413527 100644 --- a/pkg/meta/proto/imageData.proto +++ b/pkg/meta/proto/meta.proto @@ -1,8 +1,9 @@ syntax = "proto3"; package v1; -import "versioned.proto"; import "config.proto"; +import "manifest.proto"; +import "index.proto"; import "timestamp.proto"; import "descriptor.proto"; @@ -12,52 +13,44 @@ message TagDescriptor { } message ImageData { - Versioned Versioned = 1; - string MediaType = 2; - string ArtifacType = 3; - repeated ManifestData Manifests = 4; - optional Descriptor Subject = 5; - map Annotations = 6; - int64 Size = 7; - string Digest = 8; - - optional string Repo = 9; - optional string Tag = 10; + string MediaType = 1; + repeated ManifestData Manifests = 2; + optional IndexData Index = 3; } message ManifestData { - Versioned versioned = 1; - string Digest = 2; - optional string MediaType = 4; - optional string ArtifactType = 5; - repeated Descriptor Layers = 6; - optional Descriptor Subject = 7; - map Annotations = 8; - - int64 Size = 9; - ConfigData Config = 10; + string Digest = 1; + int64 Size = 2; + Manifest Manifest = 3; + Image Config = 4; +} + +message IndexData { + string Digest = 1; + int64 Size = 2; + Index Index = 3; } message RepoLastUpdatedImage { optional google.protobuf.Timestamp LastUpdated = 1; string MediaType = 2; - string Digest = 3; - string Tag = 4; + string Digest = 3; + string Tag = 4; } -message ProtoRepoMeta { +message RepoMeta { string Name = 1; - map Tags = 2; + map Tags = 2; - map Statistics = 3; - map Signatures = 4; - map Referrers = 5; + map Statistics = 3; + map Signatures = 4; + map Referrers = 5; - bool IsStarred = 6; - bool IsBookmarked = 7; - int32 Rank = 8; + bool IsStarred = 6; + bool IsBookmarked = 7; + int32 Rank = 8; - int32 Stars = 9; + int32 Stars = 9; int32 Size = 10; repeated string Vendors = 11; @@ -91,11 +84,12 @@ message ReferrersInfo { message ReferrerInfo { string Digest = 1; - string MediaType = 2; - string ArtifactType = 3; - int64 Size = 4; + int64 Count = 2; + string MediaType = 3; + string ArtifactType = 4; + int64 Size = 5; - map Annotations = 5; + map Annotations = 6; } message ManifestSignatures { @@ -112,10 +106,10 @@ message SignatureInfo { } message LayersInfo { - string LayerDigest = 1; - bytes LayerContent = 2; - string SignatureKey = 3; - string Signer = 4; + string LayerDigest = 1; + bytes LayerContent = 2; + string SignatureKey = 3; + string Signer = 4; - google.protobuf.Timestamp Date = 5; + google.protobuf.Timestamp Date = 5; } \ No newline at end of file diff --git a/pkg/meta/proto/versioned.proto b/pkg/meta/proto/versioned.proto index f9c82bca84..25d2a926d5 100644 --- a/pkg/meta/proto/versioned.proto +++ b/pkg/meta/proto/versioned.proto @@ -4,5 +4,5 @@ package v1; // https://github.com/opencontainers/image-spec/blob/main/specs-go/versioned.go message Versioned { - int32 schemaversion = 1; + int32 SchemaVersion = 1; } diff --git a/pkg/meta/proto_go/config.pb.go b/pkg/meta/proto_go/config.pb.go index e95795392c..df42ec154e 100644 --- a/pkg/meta/proto_go/config.pb.go +++ b/pkg/meta/proto_go/config.pb.go @@ -21,25 +21,21 @@ const ( _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) ) -type ImageConfig struct { +type Image struct { state protoimpl.MessageState sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - User string `protobuf:"bytes,1,opt,name=user,proto3" json:"user,omitempty"` - Exposedports map[string]*EmptyMessage `protobuf:"bytes,2,rep,name=exposedports,proto3" json:"exposedports,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Env []string `protobuf:"bytes,3,rep,name=env,proto3" json:"env,omitempty"` - Entrypoint []string `protobuf:"bytes,4,rep,name=entrypoint,proto3" json:"entrypoint,omitempty"` - Cmd []string `protobuf:"bytes,5,rep,name=cmd,proto3" json:"cmd,omitempty"` - Volumes map[string]*EmptyMessage `protobuf:"bytes,6,rep,name=volumes,proto3" json:"volumes,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Workingdir *string `protobuf:"bytes,7,opt,name=workingdir,proto3,oneof" json:"workingdir,omitempty"` - Labels map[string]string `protobuf:"bytes,8,rep,name=labels,proto3" json:"labels,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Stopsignal *string `protobuf:"bytes,9,opt,name=stopsignal,proto3,oneof" json:"stopsignal,omitempty"` - Argsescaped bool `protobuf:"varint,10,opt,name=argsescaped,proto3" json:"argsescaped,omitempty"` + Created *timestamppb.Timestamp `protobuf:"bytes,1,opt,name=Created,proto3,oneof" json:"Created,omitempty"` + Author *string `protobuf:"bytes,2,opt,name=Author,proto3,oneof" json:"Author,omitempty"` + Platform *Platform `protobuf:"bytes,3,opt,name=Platform,proto3" json:"Platform,omitempty"` + Config *ImageConfig `protobuf:"bytes,4,opt,name=Config,proto3,oneof" json:"Config,omitempty"` + RootFS *RootFS `protobuf:"bytes,5,opt,name=RootFS,proto3,oneof" json:"RootFS,omitempty"` + History []*History `protobuf:"bytes,6,rep,name=History,proto3" json:"History,omitempty"` } -func (x *ImageConfig) Reset() { - *x = ImageConfig{} +func (x *Image) Reset() { + *x = Image{} if protoimpl.UnsafeEnabled { mi := &file_config_proto_msgTypes[0] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) @@ -47,13 +43,13 @@ func (x *ImageConfig) Reset() { } } -func (x *ImageConfig) String() string { +func (x *Image) String() string { return protoimpl.X.MessageStringOf(x) } -func (*ImageConfig) ProtoMessage() {} +func (*Image) ProtoMessage() {} -func (x *ImageConfig) ProtoReflect() protoreflect.Message { +func (x *Image) ProtoReflect() protoreflect.Message { mi := &file_config_proto_msgTypes[0] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) @@ -65,42 +61,105 @@ func (x *ImageConfig) ProtoReflect() protoreflect.Message { return mi.MessageOf(x) } -// Deprecated: Use ImageConfig.ProtoReflect.Descriptor instead. -func (*ImageConfig) Descriptor() ([]byte, []int) { +// Deprecated: Use Image.ProtoReflect.Descriptor instead. +func (*Image) Descriptor() ([]byte, []int) { return file_config_proto_rawDescGZIP(), []int{0} } -func (x *ImageConfig) GetUser() string { +func (x *Image) GetCreated() *timestamppb.Timestamp { if x != nil { - return x.User + return x.Created + } + return nil +} + +func (x *Image) GetAuthor() string { + if x != nil && x.Author != nil { + return *x.Author } return "" } -func (x *ImageConfig) GetExposedports() map[string]*EmptyMessage { +func (x *Image) GetPlatform() *Platform { if x != nil { - return x.Exposedports + return x.Platform } return nil } -func (x *ImageConfig) GetEnv() []string { +func (x *Image) GetConfig() *ImageConfig { if x != nil { - return x.Env + return x.Config } return nil } -func (x *ImageConfig) GetEntrypoint() []string { +func (x *Image) GetRootFS() *RootFS { if x != nil { - return x.Entrypoint + return x.RootFS } return nil } -func (x *ImageConfig) GetCmd() []string { +func (x *Image) GetHistory() []*History { if x != nil { - return x.Cmd + return x.History + } + return nil +} + +type ImageConfig struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + ExposedPorts map[string]*EmptyMessage `protobuf:"bytes,1,rep,name=ExposedPorts,proto3" json:"ExposedPorts,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + Volumes map[string]*EmptyMessage `protobuf:"bytes,2,rep,name=Volumes,proto3" json:"Volumes,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + Labels map[string]string `protobuf:"bytes,3,rep,name=Labels,proto3" json:"Labels,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + User string `protobuf:"bytes,4,opt,name=User,proto3" json:"User,omitempty"` + Env []string `protobuf:"bytes,5,rep,name=Env,proto3" json:"Env,omitempty"` + Entrypoint []string `protobuf:"bytes,6,rep,name=Entrypoint,proto3" json:"Entrypoint,omitempty"` + Cmd []string `protobuf:"bytes,7,rep,name=Cmd,proto3" json:"Cmd,omitempty"` + WorkingDir *string `protobuf:"bytes,8,opt,name=WorkingDir,proto3,oneof" json:"WorkingDir,omitempty"` + StopSignal *string `protobuf:"bytes,9,opt,name=StopSignal,proto3,oneof" json:"StopSignal,omitempty"` + ArgsEscaped bool `protobuf:"varint,10,opt,name=ArgsEscaped,proto3" json:"ArgsEscaped,omitempty"` +} + +func (x *ImageConfig) Reset() { + *x = ImageConfig{} + if protoimpl.UnsafeEnabled { + mi := &file_config_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ImageConfig) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ImageConfig) ProtoMessage() {} + +func (x *ImageConfig) ProtoReflect() protoreflect.Message { + mi := &file_config_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ImageConfig.ProtoReflect.Descriptor instead. +func (*ImageConfig) Descriptor() ([]byte, []int) { + return file_config_proto_rawDescGZIP(), []int{1} +} + +func (x *ImageConfig) GetExposedPorts() map[string]*EmptyMessage { + if x != nil { + return x.ExposedPorts } return nil } @@ -112,30 +171,58 @@ func (x *ImageConfig) GetVolumes() map[string]*EmptyMessage { return nil } -func (x *ImageConfig) GetWorkingdir() string { - if x != nil && x.Workingdir != nil { - return *x.Workingdir +func (x *ImageConfig) GetLabels() map[string]string { + if x != nil { + return x.Labels + } + return nil +} + +func (x *ImageConfig) GetUser() string { + if x != nil { + return x.User } return "" } -func (x *ImageConfig) GetLabels() map[string]string { +func (x *ImageConfig) GetEnv() []string { if x != nil { - return x.Labels + return x.Env + } + return nil +} + +func (x *ImageConfig) GetEntrypoint() []string { + if x != nil { + return x.Entrypoint + } + return nil +} + +func (x *ImageConfig) GetCmd() []string { + if x != nil { + return x.Cmd } return nil } -func (x *ImageConfig) GetStopsignal() string { - if x != nil && x.Stopsignal != nil { - return *x.Stopsignal +func (x *ImageConfig) GetWorkingDir() string { + if x != nil && x.WorkingDir != nil { + return *x.WorkingDir } return "" } -func (x *ImageConfig) GetArgsescaped() bool { +func (x *ImageConfig) GetStopSignal() string { + if x != nil && x.StopSignal != nil { + return *x.StopSignal + } + return "" +} + +func (x *ImageConfig) GetArgsEscaped() bool { if x != nil { - return x.Argsescaped + return x.ArgsEscaped } return false } @@ -145,14 +232,14 @@ type RootFS struct { sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - Type string `protobuf:"bytes,1,opt,name=type,proto3" json:"type,omitempty"` - Diffids []string `protobuf:"bytes,2,rep,name=diffids,proto3" json:"diffids,omitempty"` + Type string `protobuf:"bytes,1,opt,name=Type,proto3" json:"Type,omitempty"` + DiffIDs []string `protobuf:"bytes,2,rep,name=DiffIDs,proto3" json:"DiffIDs,omitempty"` } func (x *RootFS) Reset() { *x = RootFS{} if protoimpl.UnsafeEnabled { - mi := &file_config_proto_msgTypes[1] + mi := &file_config_proto_msgTypes[2] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -165,7 +252,7 @@ func (x *RootFS) String() string { func (*RootFS) ProtoMessage() {} func (x *RootFS) ProtoReflect() protoreflect.Message { - mi := &file_config_proto_msgTypes[1] + mi := &file_config_proto_msgTypes[2] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -178,7 +265,7 @@ func (x *RootFS) ProtoReflect() protoreflect.Message { // Deprecated: Use RootFS.ProtoReflect.Descriptor instead. func (*RootFS) Descriptor() ([]byte, []int) { - return file_config_proto_rawDescGZIP(), []int{1} + return file_config_proto_rawDescGZIP(), []int{2} } func (x *RootFS) GetType() string { @@ -188,9 +275,9 @@ func (x *RootFS) GetType() string { return "" } -func (x *RootFS) GetDiffids() []string { +func (x *RootFS) GetDiffIDs() []string { if x != nil { - return x.Diffids + return x.DiffIDs } return nil } @@ -200,17 +287,17 @@ type History struct { sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - Created *timestamppb.Timestamp `protobuf:"bytes,1,opt,name=created,proto3,oneof" json:"created,omitempty"` - Createdby *string `protobuf:"bytes,2,opt,name=createdby,proto3,oneof" json:"createdby,omitempty"` - Author *string `protobuf:"bytes,3,opt,name=author,proto3,oneof" json:"author,omitempty"` - Comment *string `protobuf:"bytes,4,opt,name=comment,proto3,oneof" json:"comment,omitempty"` - Emptylayer *bool `protobuf:"varint,5,opt,name=emptylayer,proto3,oneof" json:"emptylayer,omitempty"` + Created *timestamppb.Timestamp `protobuf:"bytes,1,opt,name=Created,proto3,oneof" json:"Created,omitempty"` + CreatedBy *string `protobuf:"bytes,2,opt,name=CreatedBy,proto3,oneof" json:"CreatedBy,omitempty"` + Author *string `protobuf:"bytes,3,opt,name=Author,proto3,oneof" json:"Author,omitempty"` + Comment *string `protobuf:"bytes,4,opt,name=Comment,proto3,oneof" json:"Comment,omitempty"` + EmptyLayer *bool `protobuf:"varint,5,opt,name=EmptyLayer,proto3,oneof" json:"EmptyLayer,omitempty"` } func (x *History) Reset() { *x = History{} if protoimpl.UnsafeEnabled { - mi := &file_config_proto_msgTypes[2] + mi := &file_config_proto_msgTypes[3] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -223,7 +310,7 @@ func (x *History) String() string { func (*History) ProtoMessage() {} func (x *History) ProtoReflect() protoreflect.Message { - mi := &file_config_proto_msgTypes[2] + mi := &file_config_proto_msgTypes[3] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -236,7 +323,7 @@ func (x *History) ProtoReflect() protoreflect.Message { // Deprecated: Use History.ProtoReflect.Descriptor instead. func (*History) Descriptor() ([]byte, []int) { - return file_config_proto_rawDescGZIP(), []int{2} + return file_config_proto_rawDescGZIP(), []int{3} } func (x *History) GetCreated() *timestamppb.Timestamp { @@ -246,9 +333,9 @@ func (x *History) GetCreated() *timestamppb.Timestamp { return nil } -func (x *History) GetCreatedby() string { - if x != nil && x.Createdby != nil { - return *x.Createdby +func (x *History) GetCreatedBy() string { + if x != nil && x.CreatedBy != nil { + return *x.CreatedBy } return "" } @@ -267,116 +354,13 @@ func (x *History) GetComment() string { return "" } -func (x *History) GetEmptylayer() bool { - if x != nil && x.Emptylayer != nil { - return *x.Emptylayer +func (x *History) GetEmptyLayer() bool { + if x != nil && x.EmptyLayer != nil { + return *x.EmptyLayer } return false } -type ConfigData struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Created *timestamppb.Timestamp `protobuf:"bytes,1,opt,name=created,proto3,oneof" json:"created,omitempty"` - Author *string `protobuf:"bytes,2,opt,name=author,proto3,oneof" json:"author,omitempty"` - Platform *Platform `protobuf:"bytes,3,opt,name=platform,proto3" json:"platform,omitempty"` - Config *ImageConfig `protobuf:"bytes,4,opt,name=config,proto3,oneof" json:"config,omitempty"` - Rootfs *RootFS `protobuf:"bytes,5,opt,name=rootfs,proto3,oneof" json:"rootfs,omitempty"` - History []*History `protobuf:"bytes,6,rep,name=history,proto3" json:"history,omitempty"` - Digest string `protobuf:"bytes,7,opt,name=Digest,proto3" json:"Digest,omitempty"` - Size int64 `protobuf:"varint,8,opt,name=Size,proto3" json:"Size,omitempty"` -} - -func (x *ConfigData) Reset() { - *x = ConfigData{} - if protoimpl.UnsafeEnabled { - mi := &file_config_proto_msgTypes[3] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *ConfigData) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*ConfigData) ProtoMessage() {} - -func (x *ConfigData) ProtoReflect() protoreflect.Message { - mi := &file_config_proto_msgTypes[3] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use ConfigData.ProtoReflect.Descriptor instead. -func (*ConfigData) Descriptor() ([]byte, []int) { - return file_config_proto_rawDescGZIP(), []int{3} -} - -func (x *ConfigData) GetCreated() *timestamppb.Timestamp { - if x != nil { - return x.Created - } - return nil -} - -func (x *ConfigData) GetAuthor() string { - if x != nil && x.Author != nil { - return *x.Author - } - return "" -} - -func (x *ConfigData) GetPlatform() *Platform { - if x != nil { - return x.Platform - } - return nil -} - -func (x *ConfigData) GetConfig() *ImageConfig { - if x != nil { - return x.Config - } - return nil -} - -func (x *ConfigData) GetRootfs() *RootFS { - if x != nil { - return x.Rootfs - } - return nil -} - -func (x *ConfigData) GetHistory() []*History { - if x != nil { - return x.History - } - return nil -} - -func (x *ConfigData) GetDigest() string { - if x != nil { - return x.Digest - } - return "" -} - -func (x *ConfigData) GetSize() int64 { - if x != nil { - return x.Size - } - return 0 -} - type EmptyMessage struct { state protoimpl.MessageState sizeCache protoimpl.SizeCache @@ -421,91 +405,88 @@ var file_config_proto_rawDesc = []byte{ 0x0a, 0x0c, 0x63, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x02, 0x76, 0x31, 0x1a, 0x0f, 0x74, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x10, 0x64, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x2e, - 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0xff, 0x04, 0x0a, 0x0b, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, - 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x12, 0x12, 0x0a, 0x04, 0x75, 0x73, 0x65, 0x72, 0x18, 0x01, 0x20, - 0x01, 0x28, 0x09, 0x52, 0x04, 0x75, 0x73, 0x65, 0x72, 0x12, 0x45, 0x0a, 0x0c, 0x65, 0x78, 0x70, - 0x6f, 0x73, 0x65, 0x64, 0x70, 0x6f, 0x72, 0x74, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0b, 0x32, - 0x21, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, - 0x2e, 0x45, 0x78, 0x70, 0x6f, 0x73, 0x65, 0x64, 0x70, 0x6f, 0x72, 0x74, 0x73, 0x45, 0x6e, 0x74, - 0x72, 0x79, 0x52, 0x0c, 0x65, 0x78, 0x70, 0x6f, 0x73, 0x65, 0x64, 0x70, 0x6f, 0x72, 0x74, 0x73, - 0x12, 0x10, 0x0a, 0x03, 0x65, 0x6e, 0x76, 0x18, 0x03, 0x20, 0x03, 0x28, 0x09, 0x52, 0x03, 0x65, - 0x6e, 0x76, 0x12, 0x1e, 0x0a, 0x0a, 0x65, 0x6e, 0x74, 0x72, 0x79, 0x70, 0x6f, 0x69, 0x6e, 0x74, - 0x18, 0x04, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0a, 0x65, 0x6e, 0x74, 0x72, 0x79, 0x70, 0x6f, 0x69, - 0x6e, 0x74, 0x12, 0x10, 0x0a, 0x03, 0x63, 0x6d, 0x64, 0x18, 0x05, 0x20, 0x03, 0x28, 0x09, 0x52, - 0x03, 0x63, 0x6d, 0x64, 0x12, 0x36, 0x0a, 0x07, 0x76, 0x6f, 0x6c, 0x75, 0x6d, 0x65, 0x73, 0x18, - 0x06, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1c, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, - 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x2e, 0x56, 0x6f, 0x6c, 0x75, 0x6d, 0x65, 0x73, 0x45, 0x6e, - 0x74, 0x72, 0x79, 0x52, 0x07, 0x76, 0x6f, 0x6c, 0x75, 0x6d, 0x65, 0x73, 0x12, 0x23, 0x0a, 0x0a, - 0x77, 0x6f, 0x72, 0x6b, 0x69, 0x6e, 0x67, 0x64, 0x69, 0x72, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, - 0x48, 0x00, 0x52, 0x0a, 0x77, 0x6f, 0x72, 0x6b, 0x69, 0x6e, 0x67, 0x64, 0x69, 0x72, 0x88, 0x01, - 0x01, 0x12, 0x33, 0x0a, 0x06, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x18, 0x08, 0x20, 0x03, 0x28, - 0x0b, 0x32, 0x1b, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, 0x6f, 0x6e, 0x66, - 0x69, 0x67, 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x06, - 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x12, 0x23, 0x0a, 0x0a, 0x73, 0x74, 0x6f, 0x70, 0x73, 0x69, - 0x67, 0x6e, 0x61, 0x6c, 0x18, 0x09, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x0a, 0x73, 0x74, - 0x6f, 0x70, 0x73, 0x69, 0x67, 0x6e, 0x61, 0x6c, 0x88, 0x01, 0x01, 0x12, 0x20, 0x0a, 0x0b, 0x61, - 0x72, 0x67, 0x73, 0x65, 0x73, 0x63, 0x61, 0x70, 0x65, 0x64, 0x18, 0x0a, 0x20, 0x01, 0x28, 0x08, - 0x52, 0x0b, 0x61, 0x72, 0x67, 0x73, 0x65, 0x73, 0x63, 0x61, 0x70, 0x65, 0x64, 0x1a, 0x51, 0x0a, - 0x11, 0x45, 0x78, 0x70, 0x6f, 0x73, 0x65, 0x64, 0x70, 0x6f, 0x72, 0x74, 0x73, 0x45, 0x6e, 0x74, - 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, - 0x03, 0x6b, 0x65, 0x79, 0x12, 0x26, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, - 0x01, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x76, 0x31, 0x2e, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4d, 0x65, - 0x73, 0x73, 0x61, 0x67, 0x65, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, - 0x1a, 0x4c, 0x0a, 0x0c, 0x56, 0x6f, 0x6c, 0x75, 0x6d, 0x65, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, - 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, - 0x65, 0x79, 0x12, 0x26, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, - 0x0b, 0x32, 0x10, 0x2e, 0x76, 0x31, 0x2e, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4d, 0x65, 0x73, 0x73, - 0x61, 0x67, 0x65, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x39, - 0x0a, 0x0b, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, - 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, - 0x14, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, - 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x42, 0x0d, 0x0a, 0x0b, 0x5f, 0x77, 0x6f, - 0x72, 0x6b, 0x69, 0x6e, 0x67, 0x64, 0x69, 0x72, 0x42, 0x0d, 0x0a, 0x0b, 0x5f, 0x73, 0x74, 0x6f, - 0x70, 0x73, 0x69, 0x67, 0x6e, 0x61, 0x6c, 0x22, 0x36, 0x0a, 0x06, 0x52, 0x6f, 0x6f, 0x74, 0x46, - 0x53, 0x12, 0x12, 0x0a, 0x04, 0x74, 0x79, 0x70, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, - 0x04, 0x74, 0x79, 0x70, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x64, 0x69, 0x66, 0x66, 0x69, 0x64, 0x73, - 0x18, 0x02, 0x20, 0x03, 0x28, 0x09, 0x52, 0x07, 0x64, 0x69, 0x66, 0x66, 0x69, 0x64, 0x73, 0x22, - 0x88, 0x02, 0x0a, 0x07, 0x48, 0x69, 0x73, 0x74, 0x6f, 0x72, 0x79, 0x12, 0x39, 0x0a, 0x07, 0x63, - 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, - 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, - 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x48, 0x00, 0x52, 0x07, 0x63, 0x72, 0x65, 0x61, - 0x74, 0x65, 0x64, 0x88, 0x01, 0x01, 0x12, 0x21, 0x0a, 0x09, 0x63, 0x72, 0x65, 0x61, 0x74, 0x65, - 0x64, 0x62, 0x79, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x09, 0x63, 0x72, 0x65, - 0x61, 0x74, 0x65, 0x64, 0x62, 0x79, 0x88, 0x01, 0x01, 0x12, 0x1b, 0x0a, 0x06, 0x61, 0x75, 0x74, - 0x68, 0x6f, 0x72, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x48, 0x02, 0x52, 0x06, 0x61, 0x75, 0x74, - 0x68, 0x6f, 0x72, 0x88, 0x01, 0x01, 0x12, 0x1d, 0x0a, 0x07, 0x63, 0x6f, 0x6d, 0x6d, 0x65, 0x6e, - 0x74, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x48, 0x03, 0x52, 0x07, 0x63, 0x6f, 0x6d, 0x6d, 0x65, - 0x6e, 0x74, 0x88, 0x01, 0x01, 0x12, 0x23, 0x0a, 0x0a, 0x65, 0x6d, 0x70, 0x74, 0x79, 0x6c, 0x61, - 0x79, 0x65, 0x72, 0x18, 0x05, 0x20, 0x01, 0x28, 0x08, 0x48, 0x04, 0x52, 0x0a, 0x65, 0x6d, 0x70, - 0x74, 0x79, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x88, 0x01, 0x01, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x63, - 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x42, 0x0c, 0x0a, 0x0a, 0x5f, 0x63, 0x72, 0x65, 0x61, 0x74, - 0x65, 0x64, 0x62, 0x79, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x61, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x42, - 0x0a, 0x0a, 0x08, 0x5f, 0x63, 0x6f, 0x6d, 0x6d, 0x65, 0x6e, 0x74, 0x42, 0x0d, 0x0a, 0x0b, 0x5f, - 0x65, 0x6d, 0x70, 0x74, 0x79, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x22, 0xe5, 0x02, 0x0a, 0x0a, 0x43, - 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x44, 0x61, 0x74, 0x61, 0x12, 0x39, 0x0a, 0x07, 0x63, 0x72, 0x65, - 0x61, 0x74, 0x65, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, - 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, - 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x48, 0x00, 0x52, 0x07, 0x63, 0x72, 0x65, 0x61, 0x74, 0x65, - 0x64, 0x88, 0x01, 0x01, 0x12, 0x1b, 0x0a, 0x06, 0x61, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x18, 0x02, - 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x06, 0x61, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x88, 0x01, - 0x01, 0x12, 0x28, 0x0a, 0x08, 0x70, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x18, 0x03, 0x20, - 0x01, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, - 0x6d, 0x52, 0x08, 0x70, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x12, 0x2c, 0x0a, 0x06, 0x63, - 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0f, 0x2e, 0x76, 0x31, - 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x48, 0x02, 0x52, 0x06, - 0x63, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x88, 0x01, 0x01, 0x12, 0x27, 0x0a, 0x06, 0x72, 0x6f, 0x6f, - 0x74, 0x66, 0x73, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0a, 0x2e, 0x76, 0x31, 0x2e, 0x52, - 0x6f, 0x6f, 0x74, 0x46, 0x53, 0x48, 0x03, 0x52, 0x06, 0x72, 0x6f, 0x6f, 0x74, 0x66, 0x73, 0x88, - 0x01, 0x01, 0x12, 0x25, 0x0a, 0x07, 0x68, 0x69, 0x73, 0x74, 0x6f, 0x72, 0x79, 0x18, 0x06, 0x20, - 0x03, 0x28, 0x0b, 0x32, 0x0b, 0x2e, 0x76, 0x31, 0x2e, 0x48, 0x69, 0x73, 0x74, 0x6f, 0x72, 0x79, - 0x52, 0x07, 0x68, 0x69, 0x73, 0x74, 0x6f, 0x72, 0x79, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, - 0x65, 0x73, 0x74, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, - 0x74, 0x12, 0x12, 0x0a, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x18, 0x08, 0x20, 0x01, 0x28, 0x03, 0x52, - 0x04, 0x53, 0x69, 0x7a, 0x65, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x63, 0x72, 0x65, 0x61, 0x74, 0x65, - 0x64, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x61, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x42, 0x09, 0x0a, 0x07, - 0x5f, 0x63, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x72, 0x6f, 0x6f, 0x74, - 0x66, 0x73, 0x22, 0x0e, 0x0a, 0x0c, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4d, 0x65, 0x73, 0x73, 0x61, - 0x67, 0x65, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, + 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0xb4, 0x02, 0x0a, 0x05, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x12, + 0x39, 0x0a, 0x07, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, + 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, + 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x48, 0x00, 0x52, 0x07, + 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x88, 0x01, 0x01, 0x12, 0x1b, 0x0a, 0x06, 0x41, 0x75, + 0x74, 0x68, 0x6f, 0x72, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x06, 0x41, 0x75, + 0x74, 0x68, 0x6f, 0x72, 0x88, 0x01, 0x01, 0x12, 0x28, 0x0a, 0x08, 0x50, 0x6c, 0x61, 0x74, 0x66, + 0x6f, 0x72, 0x6d, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, 0x31, 0x2e, 0x50, + 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x52, 0x08, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, + 0x6d, 0x12, 0x2c, 0x0a, 0x06, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, + 0x0b, 0x32, 0x0f, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, 0x6f, 0x6e, 0x66, + 0x69, 0x67, 0x48, 0x02, 0x52, 0x06, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x88, 0x01, 0x01, 0x12, + 0x27, 0x0a, 0x06, 0x52, 0x6f, 0x6f, 0x74, 0x46, 0x53, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, + 0x0a, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x6f, 0x6f, 0x74, 0x46, 0x53, 0x48, 0x03, 0x52, 0x06, 0x52, + 0x6f, 0x6f, 0x74, 0x46, 0x53, 0x88, 0x01, 0x01, 0x12, 0x25, 0x0a, 0x07, 0x48, 0x69, 0x73, 0x74, + 0x6f, 0x72, 0x79, 0x18, 0x06, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0b, 0x2e, 0x76, 0x31, 0x2e, 0x48, + 0x69, 0x73, 0x74, 0x6f, 0x72, 0x79, 0x52, 0x07, 0x48, 0x69, 0x73, 0x74, 0x6f, 0x72, 0x79, 0x42, + 0x0a, 0x0a, 0x08, 0x5f, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x42, 0x09, 0x0a, 0x07, 0x5f, + 0x41, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x43, 0x6f, 0x6e, 0x66, 0x69, + 0x67, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x52, 0x6f, 0x6f, 0x74, 0x46, 0x53, 0x22, 0xff, 0x04, 0x0a, + 0x0b, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x12, 0x45, 0x0a, 0x0c, + 0x45, 0x78, 0x70, 0x6f, 0x73, 0x65, 0x64, 0x50, 0x6f, 0x72, 0x74, 0x73, 0x18, 0x01, 0x20, 0x03, + 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, 0x6f, 0x6e, + 0x66, 0x69, 0x67, 0x2e, 0x45, 0x78, 0x70, 0x6f, 0x73, 0x65, 0x64, 0x50, 0x6f, 0x72, 0x74, 0x73, + 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0c, 0x45, 0x78, 0x70, 0x6f, 0x73, 0x65, 0x64, 0x50, 0x6f, + 0x72, 0x74, 0x73, 0x12, 0x36, 0x0a, 0x07, 0x56, 0x6f, 0x6c, 0x75, 0x6d, 0x65, 0x73, 0x18, 0x02, + 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1c, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, + 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x2e, 0x56, 0x6f, 0x6c, 0x75, 0x6d, 0x65, 0x73, 0x45, 0x6e, 0x74, + 0x72, 0x79, 0x52, 0x07, 0x56, 0x6f, 0x6c, 0x75, 0x6d, 0x65, 0x73, 0x12, 0x33, 0x0a, 0x06, 0x4c, + 0x61, 0x62, 0x65, 0x6c, 0x73, 0x18, 0x03, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x76, 0x31, + 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x2e, 0x4c, 0x61, 0x62, + 0x65, 0x6c, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x06, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, + 0x12, 0x12, 0x0a, 0x04, 0x55, 0x73, 0x65, 0x72, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, + 0x55, 0x73, 0x65, 0x72, 0x12, 0x10, 0x0a, 0x03, 0x45, 0x6e, 0x76, 0x18, 0x05, 0x20, 0x03, 0x28, + 0x09, 0x52, 0x03, 0x45, 0x6e, 0x76, 0x12, 0x1e, 0x0a, 0x0a, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x70, + 0x6f, 0x69, 0x6e, 0x74, 0x18, 0x06, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0a, 0x45, 0x6e, 0x74, 0x72, + 0x79, 0x70, 0x6f, 0x69, 0x6e, 0x74, 0x12, 0x10, 0x0a, 0x03, 0x43, 0x6d, 0x64, 0x18, 0x07, 0x20, + 0x03, 0x28, 0x09, 0x52, 0x03, 0x43, 0x6d, 0x64, 0x12, 0x23, 0x0a, 0x0a, 0x57, 0x6f, 0x72, 0x6b, + 0x69, 0x6e, 0x67, 0x44, 0x69, 0x72, 0x18, 0x08, 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, 0x52, 0x0a, + 0x57, 0x6f, 0x72, 0x6b, 0x69, 0x6e, 0x67, 0x44, 0x69, 0x72, 0x88, 0x01, 0x01, 0x12, 0x23, 0x0a, + 0x0a, 0x53, 0x74, 0x6f, 0x70, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x6c, 0x18, 0x09, 0x20, 0x01, 0x28, + 0x09, 0x48, 0x01, 0x52, 0x0a, 0x53, 0x74, 0x6f, 0x70, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x6c, 0x88, + 0x01, 0x01, 0x12, 0x20, 0x0a, 0x0b, 0x41, 0x72, 0x67, 0x73, 0x45, 0x73, 0x63, 0x61, 0x70, 0x65, + 0x64, 0x18, 0x0a, 0x20, 0x01, 0x28, 0x08, 0x52, 0x0b, 0x41, 0x72, 0x67, 0x73, 0x45, 0x73, 0x63, + 0x61, 0x70, 0x65, 0x64, 0x1a, 0x51, 0x0a, 0x11, 0x45, 0x78, 0x70, 0x6f, 0x73, 0x65, 0x64, 0x50, + 0x6f, 0x72, 0x74, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, + 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x26, 0x0a, 0x05, 0x76, + 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x76, 0x31, 0x2e, + 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65, 0x52, 0x05, 0x76, 0x61, + 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x4c, 0x0a, 0x0c, 0x56, 0x6f, 0x6c, 0x75, 0x6d, + 0x65, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x26, 0x0a, 0x05, 0x76, 0x61, 0x6c, + 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x76, 0x31, 0x2e, 0x45, 0x6d, + 0x70, 0x74, 0x79, 0x4d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, + 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x39, 0x0a, 0x0b, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x45, + 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, + 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, + 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, + 0x42, 0x0d, 0x0a, 0x0b, 0x5f, 0x57, 0x6f, 0x72, 0x6b, 0x69, 0x6e, 0x67, 0x44, 0x69, 0x72, 0x42, + 0x0d, 0x0a, 0x0b, 0x5f, 0x53, 0x74, 0x6f, 0x70, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x6c, 0x22, 0x36, + 0x0a, 0x06, 0x52, 0x6f, 0x6f, 0x74, 0x46, 0x53, 0x12, 0x12, 0x0a, 0x04, 0x54, 0x79, 0x70, 0x65, + 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x54, 0x79, 0x70, 0x65, 0x12, 0x18, 0x0a, 0x07, + 0x44, 0x69, 0x66, 0x66, 0x49, 0x44, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x09, 0x52, 0x07, 0x44, + 0x69, 0x66, 0x66, 0x49, 0x44, 0x73, 0x22, 0x88, 0x02, 0x0a, 0x07, 0x48, 0x69, 0x73, 0x74, 0x6f, + 0x72, 0x79, 0x12, 0x39, 0x0a, 0x07, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x18, 0x01, 0x20, + 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, + 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x48, + 0x00, 0x52, 0x07, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x88, 0x01, 0x01, 0x12, 0x21, 0x0a, + 0x09, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x42, 0x79, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, + 0x48, 0x01, 0x52, 0x09, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x42, 0x79, 0x88, 0x01, 0x01, + 0x12, 0x1b, 0x0a, 0x06, 0x41, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, + 0x48, 0x02, 0x52, 0x06, 0x41, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x88, 0x01, 0x01, 0x12, 0x1d, 0x0a, + 0x07, 0x43, 0x6f, 0x6d, 0x6d, 0x65, 0x6e, 0x74, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x48, 0x03, + 0x52, 0x07, 0x43, 0x6f, 0x6d, 0x6d, 0x65, 0x6e, 0x74, 0x88, 0x01, 0x01, 0x12, 0x23, 0x0a, 0x0a, + 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x18, 0x05, 0x20, 0x01, 0x28, 0x08, + 0x48, 0x04, 0x52, 0x0a, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x88, 0x01, + 0x01, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x42, 0x0c, 0x0a, + 0x0a, 0x5f, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x42, 0x79, 0x42, 0x09, 0x0a, 0x07, 0x5f, + 0x41, 0x75, 0x74, 0x68, 0x6f, 0x72, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x43, 0x6f, 0x6d, 0x6d, 0x65, + 0x6e, 0x74, 0x42, 0x0d, 0x0a, 0x0b, 0x5f, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4c, 0x61, 0x79, 0x65, + 0x72, 0x22, 0x0e, 0x0a, 0x0c, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x4d, 0x65, 0x73, 0x73, 0x61, 0x67, + 0x65, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, } var ( @@ -522,28 +503,28 @@ func file_config_proto_rawDescGZIP() []byte { var file_config_proto_msgTypes = make([]protoimpl.MessageInfo, 8) var file_config_proto_goTypes = []interface{}{ - (*ImageConfig)(nil), // 0: v1.ImageConfig - (*RootFS)(nil), // 1: v1.RootFS - (*History)(nil), // 2: v1.History - (*ConfigData)(nil), // 3: v1.ConfigData + (*Image)(nil), // 0: v1.Image + (*ImageConfig)(nil), // 1: v1.ImageConfig + (*RootFS)(nil), // 2: v1.RootFS + (*History)(nil), // 3: v1.History (*EmptyMessage)(nil), // 4: v1.EmptyMessage - nil, // 5: v1.ImageConfig.ExposedportsEntry + nil, // 5: v1.ImageConfig.ExposedPortsEntry nil, // 6: v1.ImageConfig.VolumesEntry nil, // 7: v1.ImageConfig.LabelsEntry (*timestamppb.Timestamp)(nil), // 8: google.protobuf.Timestamp (*Platform)(nil), // 9: v1.Platform } var file_config_proto_depIdxs = []int32{ - 5, // 0: v1.ImageConfig.exposedports:type_name -> v1.ImageConfig.ExposedportsEntry - 6, // 1: v1.ImageConfig.volumes:type_name -> v1.ImageConfig.VolumesEntry - 7, // 2: v1.ImageConfig.labels:type_name -> v1.ImageConfig.LabelsEntry - 8, // 3: v1.History.created:type_name -> google.protobuf.Timestamp - 8, // 4: v1.ConfigData.created:type_name -> google.protobuf.Timestamp - 9, // 5: v1.ConfigData.platform:type_name -> v1.Platform - 0, // 6: v1.ConfigData.config:type_name -> v1.ImageConfig - 1, // 7: v1.ConfigData.rootfs:type_name -> v1.RootFS - 2, // 8: v1.ConfigData.history:type_name -> v1.History - 4, // 9: v1.ImageConfig.ExposedportsEntry.value:type_name -> v1.EmptyMessage + 8, // 0: v1.Image.Created:type_name -> google.protobuf.Timestamp + 9, // 1: v1.Image.Platform:type_name -> v1.Platform + 1, // 2: v1.Image.Config:type_name -> v1.ImageConfig + 2, // 3: v1.Image.RootFS:type_name -> v1.RootFS + 3, // 4: v1.Image.History:type_name -> v1.History + 5, // 5: v1.ImageConfig.ExposedPorts:type_name -> v1.ImageConfig.ExposedPortsEntry + 6, // 6: v1.ImageConfig.Volumes:type_name -> v1.ImageConfig.VolumesEntry + 7, // 7: v1.ImageConfig.Labels:type_name -> v1.ImageConfig.LabelsEntry + 8, // 8: v1.History.Created:type_name -> google.protobuf.Timestamp + 4, // 9: v1.ImageConfig.ExposedPortsEntry.value:type_name -> v1.EmptyMessage 4, // 10: v1.ImageConfig.VolumesEntry.value:type_name -> v1.EmptyMessage 11, // [11:11] is the sub-list for method output_type 11, // [11:11] is the sub-list for method input_type @@ -560,7 +541,7 @@ func file_config_proto_init() { file_descriptor_proto_init() if !protoimpl.UnsafeEnabled { file_config_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ImageConfig); i { + switch v := v.(*Image); i { case 0: return &v.state case 1: @@ -572,7 +553,7 @@ func file_config_proto_init() { } } file_config_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*RootFS); i { + switch v := v.(*ImageConfig); i { case 0: return &v.state case 1: @@ -584,7 +565,7 @@ func file_config_proto_init() { } } file_config_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*History); i { + switch v := v.(*RootFS); i { case 0: return &v.state case 1: @@ -596,7 +577,7 @@ func file_config_proto_init() { } } file_config_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ConfigData); i { + switch v := v.(*History); i { case 0: return &v.state case 1: @@ -621,7 +602,7 @@ func file_config_proto_init() { } } file_config_proto_msgTypes[0].OneofWrappers = []interface{}{} - file_config_proto_msgTypes[2].OneofWrappers = []interface{}{} + file_config_proto_msgTypes[1].OneofWrappers = []interface{}{} file_config_proto_msgTypes[3].OneofWrappers = []interface{}{} type x struct{} out := protoimpl.TypeBuilder{ diff --git a/pkg/meta/proto_go/descriptor.pb.go b/pkg/meta/proto_go/descriptor.pb.go index a7c056b8b9..8a39bc8105 100644 --- a/pkg/meta/proto_go/descriptor.pb.go +++ b/pkg/meta/proto_go/descriptor.pb.go @@ -25,14 +25,14 @@ type Descriptor struct { sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - MediaType string `protobuf:"bytes,1,opt,name=mediaType,proto3" json:"mediaType,omitempty"` - Digest string `protobuf:"bytes,2,opt,name=digest,proto3" json:"digest,omitempty"` - Size int64 `protobuf:"varint,3,opt,name=size,proto3" json:"size,omitempty"` - Urls []string `protobuf:"bytes,4,rep,name=urls,proto3" json:"urls,omitempty"` - Annotations map[string]string `protobuf:"bytes,5,rep,name=annotations,proto3" json:"annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Data []byte `protobuf:"bytes,6,opt,name=data,proto3" json:"data,omitempty"` - Platform *Platform `protobuf:"bytes,7,opt,name=platform,proto3,oneof" json:"platform,omitempty"` - ArtifactType *string `protobuf:"bytes,8,opt,name=artifactType,proto3,oneof" json:"artifactType,omitempty"` + MediaType string `protobuf:"bytes,1,opt,name=MediaType,proto3" json:"MediaType,omitempty"` + Digest string `protobuf:"bytes,2,opt,name=Digest,proto3" json:"Digest,omitempty"` + Size int64 `protobuf:"varint,3,opt,name=Size,proto3" json:"Size,omitempty"` + URLs []string `protobuf:"bytes,4,rep,name=URLs,proto3" json:"URLs,omitempty"` + Data []byte `protobuf:"bytes,5,opt,name=Data,proto3" json:"Data,omitempty"` + Platform *Platform `protobuf:"bytes,6,opt,name=Platform,proto3,oneof" json:"Platform,omitempty"` + ArtifactType *string `protobuf:"bytes,7,opt,name=ArtifactType,proto3,oneof" json:"ArtifactType,omitempty"` + Annotations map[string]string `protobuf:"bytes,8,rep,name=Annotations,proto3" json:"Annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` } func (x *Descriptor) Reset() { @@ -88,16 +88,9 @@ func (x *Descriptor) GetSize() int64 { return 0 } -func (x *Descriptor) GetUrls() []string { +func (x *Descriptor) GetURLs() []string { if x != nil { - return x.Urls - } - return nil -} - -func (x *Descriptor) GetAnnotations() map[string]string { - if x != nil { - return x.Annotations + return x.URLs } return nil } @@ -123,13 +116,23 @@ func (x *Descriptor) GetArtifactType() string { return "" } +func (x *Descriptor) GetAnnotations() map[string]string { + if x != nil { + return x.Annotations + } + return nil +} + type Platform struct { state protoimpl.MessageState sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - Os string `protobuf:"bytes,1,opt,name=Os,proto3" json:"Os,omitempty"` - Arch string `protobuf:"bytes,2,opt,name=Arch,proto3" json:"Arch,omitempty"` + Architecture string `protobuf:"bytes,1,opt,name=Architecture,proto3" json:"Architecture,omitempty"` + OS string `protobuf:"bytes,2,opt,name=OS,proto3" json:"OS,omitempty"` + OSVersion *string `protobuf:"bytes,3,opt,name=OSVersion,proto3,oneof" json:"OSVersion,omitempty"` + OSFeatures []string `protobuf:"bytes,4,rep,name=OSFeatures,proto3" json:"OSFeatures,omitempty"` + Variant *string `protobuf:"bytes,5,opt,name=Variant,proto3,oneof" json:"Variant,omitempty"` } func (x *Platform) Reset() { @@ -164,16 +167,37 @@ func (*Platform) Descriptor() ([]byte, []int) { return file_descriptor_proto_rawDescGZIP(), []int{1} } -func (x *Platform) GetOs() string { +func (x *Platform) GetArchitecture() string { if x != nil { - return x.Os + return x.Architecture } return "" } -func (x *Platform) GetArch() string { +func (x *Platform) GetOS() string { if x != nil { - return x.Arch + return x.OS + } + return "" +} + +func (x *Platform) GetOSVersion() string { + if x != nil && x.OSVersion != nil { + return *x.OSVersion + } + return "" +} + +func (x *Platform) GetOSFeatures() []string { + if x != nil { + return x.OSFeatures + } + return nil +} + +func (x *Platform) GetVariant() string { + if x != nil && x.Variant != nil { + return *x.Variant } return "" } @@ -183,33 +207,42 @@ var File_descriptor_proto protoreflect.FileDescriptor var file_descriptor_proto_rawDesc = []byte{ 0x0a, 0x10, 0x64, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x02, 0x76, 0x31, 0x22, 0xf7, 0x02, 0x0a, 0x0a, 0x44, 0x65, 0x73, 0x63, 0x72, - 0x69, 0x70, 0x74, 0x6f, 0x72, 0x12, 0x1c, 0x0a, 0x09, 0x6d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, - 0x70, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x6d, 0x65, 0x64, 0x69, 0x61, 0x54, - 0x79, 0x70, 0x65, 0x12, 0x16, 0x0a, 0x06, 0x64, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x02, 0x20, - 0x01, 0x28, 0x09, 0x52, 0x06, 0x64, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x12, 0x0a, 0x04, 0x73, - 0x69, 0x7a, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, 0x04, 0x73, 0x69, 0x7a, 0x65, 0x12, - 0x12, 0x0a, 0x04, 0x75, 0x72, 0x6c, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x09, 0x52, 0x04, 0x75, - 0x72, 0x6c, 0x73, 0x12, 0x41, 0x0a, 0x0b, 0x61, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, - 0x6e, 0x73, 0x18, 0x05, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1f, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, - 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x2e, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, - 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x61, 0x6e, 0x6e, 0x6f, 0x74, - 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x12, 0x12, 0x0a, 0x04, 0x64, 0x61, 0x74, 0x61, 0x18, 0x06, - 0x20, 0x01, 0x28, 0x0c, 0x52, 0x04, 0x64, 0x61, 0x74, 0x61, 0x12, 0x2d, 0x0a, 0x08, 0x70, 0x6c, - 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x18, 0x07, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, - 0x31, 0x2e, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x48, 0x00, 0x52, 0x08, 0x70, 0x6c, - 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x88, 0x01, 0x01, 0x12, 0x27, 0x0a, 0x0c, 0x61, 0x72, 0x74, - 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x18, 0x08, 0x20, 0x01, 0x28, 0x09, 0x48, - 0x01, 0x52, 0x0c, 0x61, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x88, - 0x01, 0x01, 0x1a, 0x3e, 0x0a, 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, + 0x69, 0x70, 0x74, 0x6f, 0x72, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, + 0x70, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, + 0x79, 0x70, 0x65, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x02, 0x20, + 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x12, 0x0a, 0x04, 0x53, + 0x69, 0x7a, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, + 0x12, 0x0a, 0x04, 0x55, 0x52, 0x4c, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x09, 0x52, 0x04, 0x55, + 0x52, 0x4c, 0x73, 0x12, 0x12, 0x0a, 0x04, 0x44, 0x61, 0x74, 0x61, 0x18, 0x05, 0x20, 0x01, 0x28, + 0x0c, 0x52, 0x04, 0x44, 0x61, 0x74, 0x61, 0x12, 0x2d, 0x0a, 0x08, 0x50, 0x6c, 0x61, 0x74, 0x66, + 0x6f, 0x72, 0x6d, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, 0x31, 0x2e, 0x50, + 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x48, 0x00, 0x52, 0x08, 0x50, 0x6c, 0x61, 0x74, 0x66, + 0x6f, 0x72, 0x6d, 0x88, 0x01, 0x01, 0x12, 0x27, 0x0a, 0x0c, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, + 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x0c, + 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, + 0x41, 0x0a, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x08, + 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1f, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, + 0x70, 0x74, 0x6f, 0x72, 0x2e, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, + 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, + 0x6e, 0x73, 0x1a, 0x3e, 0x0a, 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, - 0x38, 0x01, 0x42, 0x0b, 0x0a, 0x09, 0x5f, 0x70, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x42, - 0x0f, 0x0a, 0x0d, 0x5f, 0x61, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, - 0x22, 0x2e, 0x0a, 0x08, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x12, 0x0e, 0x0a, 0x02, - 0x4f, 0x73, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x4f, 0x73, 0x12, 0x12, 0x0a, 0x04, - 0x41, 0x72, 0x63, 0x68, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x41, 0x72, 0x63, 0x68, - 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, + 0x38, 0x01, 0x42, 0x0b, 0x0a, 0x09, 0x5f, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x42, + 0x0f, 0x0a, 0x0d, 0x5f, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, + 0x22, 0xba, 0x01, 0x0a, 0x08, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x12, 0x22, 0x0a, + 0x0c, 0x41, 0x72, 0x63, 0x68, 0x69, 0x74, 0x65, 0x63, 0x74, 0x75, 0x72, 0x65, 0x18, 0x01, 0x20, + 0x01, 0x28, 0x09, 0x52, 0x0c, 0x41, 0x72, 0x63, 0x68, 0x69, 0x74, 0x65, 0x63, 0x74, 0x75, 0x72, + 0x65, 0x12, 0x0e, 0x0a, 0x02, 0x4f, 0x53, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x4f, + 0x53, 0x12, 0x21, 0x0a, 0x09, 0x4f, 0x53, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x03, + 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, 0x52, 0x09, 0x4f, 0x53, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, + 0x6e, 0x88, 0x01, 0x01, 0x12, 0x1e, 0x0a, 0x0a, 0x4f, 0x53, 0x46, 0x65, 0x61, 0x74, 0x75, 0x72, + 0x65, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0a, 0x4f, 0x53, 0x46, 0x65, 0x61, 0x74, + 0x75, 0x72, 0x65, 0x73, 0x12, 0x1d, 0x0a, 0x07, 0x56, 0x61, 0x72, 0x69, 0x61, 0x6e, 0x74, 0x18, + 0x05, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x07, 0x56, 0x61, 0x72, 0x69, 0x61, 0x6e, 0x74, + 0x88, 0x01, 0x01, 0x42, 0x0c, 0x0a, 0x0a, 0x5f, 0x4f, 0x53, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, + 0x6e, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x56, 0x61, 0x72, 0x69, 0x61, 0x6e, 0x74, 0x62, 0x06, 0x70, + 0x72, 0x6f, 0x74, 0x6f, 0x33, } var ( @@ -231,8 +264,8 @@ var file_descriptor_proto_goTypes = []interface{}{ nil, // 2: v1.Descriptor.AnnotationsEntry } var file_descriptor_proto_depIdxs = []int32{ - 2, // 0: v1.Descriptor.annotations:type_name -> v1.Descriptor.AnnotationsEntry - 1, // 1: v1.Descriptor.platform:type_name -> v1.Platform + 1, // 0: v1.Descriptor.Platform:type_name -> v1.Platform + 2, // 1: v1.Descriptor.Annotations:type_name -> v1.Descriptor.AnnotationsEntry 2, // [2:2] is the sub-list for method output_type 2, // [2:2] is the sub-list for method input_type 2, // [2:2] is the sub-list for extension type_name @@ -272,6 +305,7 @@ func file_descriptor_proto_init() { } } file_descriptor_proto_msgTypes[0].OneofWrappers = []interface{}{} + file_descriptor_proto_msgTypes[1].OneofWrappers = []interface{}{} type x struct{} out := protoimpl.TypeBuilder{ File: protoimpl.DescBuilder{ diff --git a/pkg/meta/proto_go/imageData.pb.go b/pkg/meta/proto_go/imageData.pb.go deleted file mode 100644 index f8d2afd3af..0000000000 --- a/pkg/meta/proto_go/imageData.pb.go +++ /dev/null @@ -1,1566 +0,0 @@ -// Code generated by protoc-gen-go. DO NOT EDIT. -// versions: -// protoc-gen-go v1.31.0 -// protoc v3.15.8 -// source: imageData.proto - -package proto_go - -import ( - protoreflect "google.golang.org/protobuf/reflect/protoreflect" - protoimpl "google.golang.org/protobuf/runtime/protoimpl" - timestamppb "google.golang.org/protobuf/types/known/timestamppb" - reflect "reflect" - sync "sync" -) - -const ( - // Verify that this generated code is sufficiently up-to-date. - _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) - // Verify that runtime/protoimpl is sufficiently up-to-date. - _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) -) - -type TagDescriptor struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - MediaType string `protobuf:"bytes,1,opt,name=MediaType,proto3" json:"MediaType,omitempty"` - Digest string `protobuf:"bytes,2,opt,name=Digest,proto3" json:"Digest,omitempty"` -} - -func (x *TagDescriptor) Reset() { - *x = TagDescriptor{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[0] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *TagDescriptor) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*TagDescriptor) ProtoMessage() {} - -func (x *TagDescriptor) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[0] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use TagDescriptor.ProtoReflect.Descriptor instead. -func (*TagDescriptor) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{0} -} - -func (x *TagDescriptor) GetMediaType() string { - if x != nil { - return x.MediaType - } - return "" -} - -func (x *TagDescriptor) GetDigest() string { - if x != nil { - return x.Digest - } - return "" -} - -type ImageData struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Versioned *Versioned `protobuf:"bytes,1,opt,name=Versioned,proto3" json:"Versioned,omitempty"` - MediaType string `protobuf:"bytes,2,opt,name=MediaType,proto3" json:"MediaType,omitempty"` - ArtifacType string `protobuf:"bytes,3,opt,name=ArtifacType,proto3" json:"ArtifacType,omitempty"` - Manifests []*ManifestData `protobuf:"bytes,4,rep,name=Manifests,proto3" json:"Manifests,omitempty"` - Subject *Descriptor `protobuf:"bytes,5,opt,name=Subject,proto3,oneof" json:"Subject,omitempty"` - Annotations map[string]string `protobuf:"bytes,6,rep,name=Annotations,proto3" json:"Annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Size int64 `protobuf:"varint,7,opt,name=Size,proto3" json:"Size,omitempty"` - Digest string `protobuf:"bytes,8,opt,name=Digest,proto3" json:"Digest,omitempty"` - Repo *string `protobuf:"bytes,9,opt,name=Repo,proto3,oneof" json:"Repo,omitempty"` - Tag *string `protobuf:"bytes,10,opt,name=Tag,proto3,oneof" json:"Tag,omitempty"` -} - -func (x *ImageData) Reset() { - *x = ImageData{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[1] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *ImageData) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*ImageData) ProtoMessage() {} - -func (x *ImageData) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[1] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use ImageData.ProtoReflect.Descriptor instead. -func (*ImageData) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{1} -} - -func (x *ImageData) GetVersioned() *Versioned { - if x != nil { - return x.Versioned - } - return nil -} - -func (x *ImageData) GetMediaType() string { - if x != nil { - return x.MediaType - } - return "" -} - -func (x *ImageData) GetArtifacType() string { - if x != nil { - return x.ArtifacType - } - return "" -} - -func (x *ImageData) GetManifests() []*ManifestData { - if x != nil { - return x.Manifests - } - return nil -} - -func (x *ImageData) GetSubject() *Descriptor { - if x != nil { - return x.Subject - } - return nil -} - -func (x *ImageData) GetAnnotations() map[string]string { - if x != nil { - return x.Annotations - } - return nil -} - -func (x *ImageData) GetSize() int64 { - if x != nil { - return x.Size - } - return 0 -} - -func (x *ImageData) GetDigest() string { - if x != nil { - return x.Digest - } - return "" -} - -func (x *ImageData) GetRepo() string { - if x != nil && x.Repo != nil { - return *x.Repo - } - return "" -} - -func (x *ImageData) GetTag() string { - if x != nil && x.Tag != nil { - return *x.Tag - } - return "" -} - -type ManifestData struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Versioned *Versioned `protobuf:"bytes,1,opt,name=versioned,proto3" json:"versioned,omitempty"` - Digest string `protobuf:"bytes,2,opt,name=Digest,proto3" json:"Digest,omitempty"` - MediaType *string `protobuf:"bytes,4,opt,name=MediaType,proto3,oneof" json:"MediaType,omitempty"` - ArtifactType *string `protobuf:"bytes,5,opt,name=ArtifactType,proto3,oneof" json:"ArtifactType,omitempty"` - Layers []*Descriptor `protobuf:"bytes,6,rep,name=Layers,proto3" json:"Layers,omitempty"` - Subject *Descriptor `protobuf:"bytes,7,opt,name=Subject,proto3,oneof" json:"Subject,omitempty"` - Annotations map[string]string `protobuf:"bytes,8,rep,name=Annotations,proto3" json:"Annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Size int64 `protobuf:"varint,9,opt,name=Size,proto3" json:"Size,omitempty"` - Config *ConfigData `protobuf:"bytes,10,opt,name=Config,proto3" json:"Config,omitempty"` -} - -func (x *ManifestData) Reset() { - *x = ManifestData{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[2] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *ManifestData) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*ManifestData) ProtoMessage() {} - -func (x *ManifestData) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[2] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use ManifestData.ProtoReflect.Descriptor instead. -func (*ManifestData) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{2} -} - -func (x *ManifestData) GetVersioned() *Versioned { - if x != nil { - return x.Versioned - } - return nil -} - -func (x *ManifestData) GetDigest() string { - if x != nil { - return x.Digest - } - return "" -} - -func (x *ManifestData) GetMediaType() string { - if x != nil && x.MediaType != nil { - return *x.MediaType - } - return "" -} - -func (x *ManifestData) GetArtifactType() string { - if x != nil && x.ArtifactType != nil { - return *x.ArtifactType - } - return "" -} - -func (x *ManifestData) GetLayers() []*Descriptor { - if x != nil { - return x.Layers - } - return nil -} - -func (x *ManifestData) GetSubject() *Descriptor { - if x != nil { - return x.Subject - } - return nil -} - -func (x *ManifestData) GetAnnotations() map[string]string { - if x != nil { - return x.Annotations - } - return nil -} - -func (x *ManifestData) GetSize() int64 { - if x != nil { - return x.Size - } - return 0 -} - -func (x *ManifestData) GetConfig() *ConfigData { - if x != nil { - return x.Config - } - return nil -} - -type RepoLastUpdatedImage struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - LastUpdated *timestamppb.Timestamp `protobuf:"bytes,1,opt,name=LastUpdated,proto3,oneof" json:"LastUpdated,omitempty"` - MediaType string `protobuf:"bytes,2,opt,name=MediaType,proto3" json:"MediaType,omitempty"` - Digest string `protobuf:"bytes,3,opt,name=Digest,proto3" json:"Digest,omitempty"` - Tag string `protobuf:"bytes,4,opt,name=Tag,proto3" json:"Tag,omitempty"` -} - -func (x *RepoLastUpdatedImage) Reset() { - *x = RepoLastUpdatedImage{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[3] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *RepoLastUpdatedImage) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*RepoLastUpdatedImage) ProtoMessage() {} - -func (x *RepoLastUpdatedImage) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[3] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use RepoLastUpdatedImage.ProtoReflect.Descriptor instead. -func (*RepoLastUpdatedImage) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{3} -} - -func (x *RepoLastUpdatedImage) GetLastUpdated() *timestamppb.Timestamp { - if x != nil { - return x.LastUpdated - } - return nil -} - -func (x *RepoLastUpdatedImage) GetMediaType() string { - if x != nil { - return x.MediaType - } - return "" -} - -func (x *RepoLastUpdatedImage) GetDigest() string { - if x != nil { - return x.Digest - } - return "" -} - -func (x *RepoLastUpdatedImage) GetTag() string { - if x != nil { - return x.Tag - } - return "" -} - -type ProtoRepoMeta struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Name string `protobuf:"bytes,1,opt,name=Name,proto3" json:"Name,omitempty"` - Tags map[string]*TagDescriptor `protobuf:"bytes,2,rep,name=Tags,proto3" json:"Tags,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Statistics map[string]*DescriptorStatistics `protobuf:"bytes,3,rep,name=Statistics,proto3" json:"Statistics,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Signatures map[string]*ManifestSignatures `protobuf:"bytes,4,rep,name=Signatures,proto3" json:"Signatures,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - Referrers map[string]*ReferrersInfo `protobuf:"bytes,5,rep,name=Referrers,proto3" json:"Referrers,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - IsStarred bool `protobuf:"varint,6,opt,name=IsStarred,proto3" json:"IsStarred,omitempty"` - IsBookmarked bool `protobuf:"varint,7,opt,name=IsBookmarked,proto3" json:"IsBookmarked,omitempty"` - Rank int32 `protobuf:"varint,8,opt,name=Rank,proto3" json:"Rank,omitempty"` - Stars int32 `protobuf:"varint,9,opt,name=Stars,proto3" json:"Stars,omitempty"` - Size int32 `protobuf:"varint,10,opt,name=Size,proto3" json:"Size,omitempty"` - Vendors []string `protobuf:"bytes,11,rep,name=Vendors,proto3" json:"Vendors,omitempty"` - Platforms []*Platform `protobuf:"bytes,12,rep,name=Platforms,proto3" json:"Platforms,omitempty"` - LastUpdatedImage *RepoLastUpdatedImage `protobuf:"bytes,13,opt,name=LastUpdatedImage,proto3,oneof" json:"LastUpdatedImage,omitempty"` -} - -func (x *ProtoRepoMeta) Reset() { - *x = ProtoRepoMeta{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[4] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *ProtoRepoMeta) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*ProtoRepoMeta) ProtoMessage() {} - -func (x *ProtoRepoMeta) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[4] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use ProtoRepoMeta.ProtoReflect.Descriptor instead. -func (*ProtoRepoMeta) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{4} -} - -func (x *ProtoRepoMeta) GetName() string { - if x != nil { - return x.Name - } - return "" -} - -func (x *ProtoRepoMeta) GetTags() map[string]*TagDescriptor { - if x != nil { - return x.Tags - } - return nil -} - -func (x *ProtoRepoMeta) GetStatistics() map[string]*DescriptorStatistics { - if x != nil { - return x.Statistics - } - return nil -} - -func (x *ProtoRepoMeta) GetSignatures() map[string]*ManifestSignatures { - if x != nil { - return x.Signatures - } - return nil -} - -func (x *ProtoRepoMeta) GetReferrers() map[string]*ReferrersInfo { - if x != nil { - return x.Referrers - } - return nil -} - -func (x *ProtoRepoMeta) GetIsStarred() bool { - if x != nil { - return x.IsStarred - } - return false -} - -func (x *ProtoRepoMeta) GetIsBookmarked() bool { - if x != nil { - return x.IsBookmarked - } - return false -} - -func (x *ProtoRepoMeta) GetRank() int32 { - if x != nil { - return x.Rank - } - return 0 -} - -func (x *ProtoRepoMeta) GetStars() int32 { - if x != nil { - return x.Stars - } - return 0 -} - -func (x *ProtoRepoMeta) GetSize() int32 { - if x != nil { - return x.Size - } - return 0 -} - -func (x *ProtoRepoMeta) GetVendors() []string { - if x != nil { - return x.Vendors - } - return nil -} - -func (x *ProtoRepoMeta) GetPlatforms() []*Platform { - if x != nil { - return x.Platforms - } - return nil -} - -func (x *ProtoRepoMeta) GetLastUpdatedImage() *RepoLastUpdatedImage { - if x != nil { - return x.LastUpdatedImage - } - return nil -} - -type RepoBlobs struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Name string `protobuf:"bytes,1,opt,name=Name,proto3" json:"Name,omitempty"` - Blobs map[string]*BlobInfo `protobuf:"bytes,2,rep,name=Blobs,proto3" json:"Blobs,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` -} - -func (x *RepoBlobs) Reset() { - *x = RepoBlobs{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[5] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *RepoBlobs) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*RepoBlobs) ProtoMessage() {} - -func (x *RepoBlobs) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[5] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use RepoBlobs.ProtoReflect.Descriptor instead. -func (*RepoBlobs) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{5} -} - -func (x *RepoBlobs) GetName() string { - if x != nil { - return x.Name - } - return "" -} - -func (x *RepoBlobs) GetBlobs() map[string]*BlobInfo { - if x != nil { - return x.Blobs - } - return nil -} - -// for example this is a manifest and it has a config, and layers -// or index and has manifests -type BlobInfo struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Size int64 `protobuf:"varint,1,opt,name=Size,proto3" json:"Size,omitempty"` - Vendors []string `protobuf:"bytes,2,rep,name=Vendors,proto3" json:"Vendors,omitempty"` - Platforms []*Platform `protobuf:"bytes,3,rep,name=Platforms,proto3" json:"Platforms,omitempty"` - SubBlobs []string `protobuf:"bytes,4,rep,name=SubBlobs,proto3" json:"SubBlobs,omitempty"` - LastUpdated *timestamppb.Timestamp `protobuf:"bytes,5,opt,name=LastUpdated,proto3,oneof" json:"LastUpdated,omitempty"` -} - -func (x *BlobInfo) Reset() { - *x = BlobInfo{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[6] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *BlobInfo) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*BlobInfo) ProtoMessage() {} - -func (x *BlobInfo) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[6] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use BlobInfo.ProtoReflect.Descriptor instead. -func (*BlobInfo) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{6} -} - -func (x *BlobInfo) GetSize() int64 { - if x != nil { - return x.Size - } - return 0 -} - -func (x *BlobInfo) GetVendors() []string { - if x != nil { - return x.Vendors - } - return nil -} - -func (x *BlobInfo) GetPlatforms() []*Platform { - if x != nil { - return x.Platforms - } - return nil -} - -func (x *BlobInfo) GetSubBlobs() []string { - if x != nil { - return x.SubBlobs - } - return nil -} - -func (x *BlobInfo) GetLastUpdated() *timestamppb.Timestamp { - if x != nil { - return x.LastUpdated - } - return nil -} - -type DescriptorStatistics struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - DownloadCount int32 `protobuf:"varint,1,opt,name=DownloadCount,proto3" json:"DownloadCount,omitempty"` -} - -func (x *DescriptorStatistics) Reset() { - *x = DescriptorStatistics{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[7] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *DescriptorStatistics) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*DescriptorStatistics) ProtoMessage() {} - -func (x *DescriptorStatistics) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[7] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use DescriptorStatistics.ProtoReflect.Descriptor instead. -func (*DescriptorStatistics) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{7} -} - -func (x *DescriptorStatistics) GetDownloadCount() int32 { - if x != nil { - return x.DownloadCount - } - return 0 -} - -type ReferrersInfo struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - List []*ReferrerInfo `protobuf:"bytes,1,rep,name=list,proto3" json:"list,omitempty"` -} - -func (x *ReferrersInfo) Reset() { - *x = ReferrersInfo{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[8] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *ReferrersInfo) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*ReferrersInfo) ProtoMessage() {} - -func (x *ReferrersInfo) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[8] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use ReferrersInfo.ProtoReflect.Descriptor instead. -func (*ReferrersInfo) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{8} -} - -func (x *ReferrersInfo) GetList() []*ReferrerInfo { - if x != nil { - return x.List - } - return nil -} - -type ReferrerInfo struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Digest string `protobuf:"bytes,1,opt,name=Digest,proto3" json:"Digest,omitempty"` - MediaType string `protobuf:"bytes,2,opt,name=MediaType,proto3" json:"MediaType,omitempty"` - ArtifactType string `protobuf:"bytes,3,opt,name=ArtifactType,proto3" json:"ArtifactType,omitempty"` - Size int64 `protobuf:"varint,4,opt,name=Size,proto3" json:"Size,omitempty"` - Annotations map[string]string `protobuf:"bytes,5,rep,name=Annotations,proto3" json:"Annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` -} - -func (x *ReferrerInfo) Reset() { - *x = ReferrerInfo{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[9] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *ReferrerInfo) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*ReferrerInfo) ProtoMessage() {} - -func (x *ReferrerInfo) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[9] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use ReferrerInfo.ProtoReflect.Descriptor instead. -func (*ReferrerInfo) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{9} -} - -func (x *ReferrerInfo) GetDigest() string { - if x != nil { - return x.Digest - } - return "" -} - -func (x *ReferrerInfo) GetMediaType() string { - if x != nil { - return x.MediaType - } - return "" -} - -func (x *ReferrerInfo) GetArtifactType() string { - if x != nil { - return x.ArtifactType - } - return "" -} - -func (x *ReferrerInfo) GetSize() int64 { - if x != nil { - return x.Size - } - return 0 -} - -func (x *ReferrerInfo) GetAnnotations() map[string]string { - if x != nil { - return x.Annotations - } - return nil -} - -type ManifestSignatures struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - Map map[string]*SignaturesInfo `protobuf:"bytes,1,rep,name=map,proto3" json:"map,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` -} - -func (x *ManifestSignatures) Reset() { - *x = ManifestSignatures{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[10] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *ManifestSignatures) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*ManifestSignatures) ProtoMessage() {} - -func (x *ManifestSignatures) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[10] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use ManifestSignatures.ProtoReflect.Descriptor instead. -func (*ManifestSignatures) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{10} -} - -func (x *ManifestSignatures) GetMap() map[string]*SignaturesInfo { - if x != nil { - return x.Map - } - return nil -} - -type SignaturesInfo struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - List []*SignatureInfo `protobuf:"bytes,1,rep,name=list,proto3" json:"list,omitempty"` -} - -func (x *SignaturesInfo) Reset() { - *x = SignaturesInfo{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[11] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *SignaturesInfo) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*SignaturesInfo) ProtoMessage() {} - -func (x *SignaturesInfo) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[11] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use SignaturesInfo.ProtoReflect.Descriptor instead. -func (*SignaturesInfo) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{11} -} - -func (x *SignaturesInfo) GetList() []*SignatureInfo { - if x != nil { - return x.List - } - return nil -} - -type SignatureInfo struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - SignatureManifestDigest string `protobuf:"bytes,1,opt,name=SignatureManifestDigest,proto3" json:"SignatureManifestDigest,omitempty"` - LayersInfo []*LayersInfo `protobuf:"bytes,2,rep,name=LayersInfo,proto3" json:"LayersInfo,omitempty"` -} - -func (x *SignatureInfo) Reset() { - *x = SignatureInfo{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[12] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *SignatureInfo) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*SignatureInfo) ProtoMessage() {} - -func (x *SignatureInfo) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[12] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use SignatureInfo.ProtoReflect.Descriptor instead. -func (*SignatureInfo) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{12} -} - -func (x *SignatureInfo) GetSignatureManifestDigest() string { - if x != nil { - return x.SignatureManifestDigest - } - return "" -} - -func (x *SignatureInfo) GetLayersInfo() []*LayersInfo { - if x != nil { - return x.LayersInfo - } - return nil -} - -type LayersInfo struct { - state protoimpl.MessageState - sizeCache protoimpl.SizeCache - unknownFields protoimpl.UnknownFields - - LayerDigest string `protobuf:"bytes,1,opt,name=LayerDigest,proto3" json:"LayerDigest,omitempty"` - LayerContent []byte `protobuf:"bytes,2,opt,name=LayerContent,proto3" json:"LayerContent,omitempty"` - SignatureKey string `protobuf:"bytes,3,opt,name=SignatureKey,proto3" json:"SignatureKey,omitempty"` - Signer string `protobuf:"bytes,4,opt,name=Signer,proto3" json:"Signer,omitempty"` - Date *timestamppb.Timestamp `protobuf:"bytes,5,opt,name=Date,proto3" json:"Date,omitempty"` -} - -func (x *LayersInfo) Reset() { - *x = LayersInfo{} - if protoimpl.UnsafeEnabled { - mi := &file_imageData_proto_msgTypes[13] - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - ms.StoreMessageInfo(mi) - } -} - -func (x *LayersInfo) String() string { - return protoimpl.X.MessageStringOf(x) -} - -func (*LayersInfo) ProtoMessage() {} - -func (x *LayersInfo) ProtoReflect() protoreflect.Message { - mi := &file_imageData_proto_msgTypes[13] - if protoimpl.UnsafeEnabled && x != nil { - ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) - if ms.LoadMessageInfo() == nil { - ms.StoreMessageInfo(mi) - } - return ms - } - return mi.MessageOf(x) -} - -// Deprecated: Use LayersInfo.ProtoReflect.Descriptor instead. -func (*LayersInfo) Descriptor() ([]byte, []int) { - return file_imageData_proto_rawDescGZIP(), []int{13} -} - -func (x *LayersInfo) GetLayerDigest() string { - if x != nil { - return x.LayerDigest - } - return "" -} - -func (x *LayersInfo) GetLayerContent() []byte { - if x != nil { - return x.LayerContent - } - return nil -} - -func (x *LayersInfo) GetSignatureKey() string { - if x != nil { - return x.SignatureKey - } - return "" -} - -func (x *LayersInfo) GetSigner() string { - if x != nil { - return x.Signer - } - return "" -} - -func (x *LayersInfo) GetDate() *timestamppb.Timestamp { - if x != nil { - return x.Date - } - return nil -} - -var File_imageData_proto protoreflect.FileDescriptor - -var file_imageData_proto_rawDesc = []byte{ - 0x0a, 0x0f, 0x69, 0x6d, 0x61, 0x67, 0x65, 0x44, 0x61, 0x74, 0x61, 0x2e, 0x70, 0x72, 0x6f, 0x74, - 0x6f, 0x12, 0x02, 0x76, 0x31, 0x1a, 0x0f, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, - 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x0c, 0x63, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x2e, 0x70, - 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x0f, 0x74, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x2e, - 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x10, 0x64, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, - 0x72, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0x45, 0x0a, 0x0d, 0x54, 0x61, 0x67, 0x44, 0x65, - 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, - 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, 0x64, - 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, - 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x22, 0xd2, - 0x03, 0x0a, 0x09, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x44, 0x61, 0x74, 0x61, 0x12, 0x2b, 0x0a, 0x09, - 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, - 0x0d, 0x2e, 0x76, 0x31, 0x2e, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x52, 0x09, - 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, - 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, - 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x12, 0x20, 0x0a, 0x0b, 0x41, 0x72, 0x74, 0x69, 0x66, - 0x61, 0x63, 0x54, 0x79, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0b, 0x41, 0x72, - 0x74, 0x69, 0x66, 0x61, 0x63, 0x54, 0x79, 0x70, 0x65, 0x12, 0x2e, 0x0a, 0x09, 0x4d, 0x61, 0x6e, - 0x69, 0x66, 0x65, 0x73, 0x74, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x76, - 0x31, 0x2e, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x44, 0x61, 0x74, 0x61, 0x52, 0x09, - 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x73, 0x12, 0x2d, 0x0a, 0x07, 0x53, 0x75, 0x62, - 0x6a, 0x65, 0x63, 0x74, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, - 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x48, 0x00, 0x52, 0x07, 0x53, 0x75, - 0x62, 0x6a, 0x65, 0x63, 0x74, 0x88, 0x01, 0x01, 0x12, 0x40, 0x0a, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, - 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x06, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1e, 0x2e, - 0x76, 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x44, 0x61, 0x74, 0x61, 0x2e, 0x41, 0x6e, 0x6e, - 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x41, - 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x12, 0x12, 0x0a, 0x04, 0x53, 0x69, - 0x7a, 0x65, 0x18, 0x07, 0x20, 0x01, 0x28, 0x03, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, 0x16, - 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x08, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, - 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x17, 0x0a, 0x04, 0x52, 0x65, 0x70, 0x6f, 0x18, 0x09, - 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x04, 0x52, 0x65, 0x70, 0x6f, 0x88, 0x01, 0x01, 0x12, - 0x15, 0x0a, 0x03, 0x54, 0x61, 0x67, 0x18, 0x0a, 0x20, 0x01, 0x28, 0x09, 0x48, 0x02, 0x52, 0x03, - 0x54, 0x61, 0x67, 0x88, 0x01, 0x01, 0x1a, 0x3e, 0x0a, 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, - 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, - 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, - 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, - 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x53, 0x75, 0x62, 0x6a, 0x65, - 0x63, 0x74, 0x42, 0x07, 0x0a, 0x05, 0x5f, 0x52, 0x65, 0x70, 0x6f, 0x42, 0x06, 0x0a, 0x04, 0x5f, - 0x54, 0x61, 0x67, 0x22, 0xe2, 0x03, 0x0a, 0x0c, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, - 0x44, 0x61, 0x74, 0x61, 0x12, 0x2b, 0x0a, 0x09, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, - 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0d, 0x2e, 0x76, 0x31, 0x2e, 0x56, 0x65, 0x72, - 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x52, 0x09, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, - 0x64, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x02, 0x20, 0x01, 0x28, - 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x21, 0x0a, 0x09, 0x4d, 0x65, 0x64, - 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, 0x52, 0x09, - 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x27, 0x0a, 0x0c, - 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x18, 0x05, 0x20, 0x01, - 0x28, 0x09, 0x48, 0x01, 0x52, 0x0c, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, - 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x26, 0x0a, 0x06, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x18, - 0x06, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, - 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x06, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x12, 0x2d, 0x0a, - 0x07, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x18, 0x07, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, - 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x48, 0x02, - 0x52, 0x07, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x88, 0x01, 0x01, 0x12, 0x43, 0x0a, 0x0b, - 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x08, 0x20, 0x03, 0x28, - 0x0b, 0x32, 0x21, 0x2e, 0x76, 0x31, 0x2e, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x44, - 0x61, 0x74, 0x61, 0x2e, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, - 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, - 0x73, 0x12, 0x12, 0x0a, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x18, 0x09, 0x20, 0x01, 0x28, 0x03, 0x52, - 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, 0x26, 0x0a, 0x06, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x18, - 0x0a, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, 0x43, 0x6f, 0x6e, 0x66, 0x69, - 0x67, 0x44, 0x61, 0x74, 0x61, 0x52, 0x06, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x1a, 0x3e, 0x0a, - 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, - 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, - 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, - 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x42, 0x0c, 0x0a, - 0x0a, 0x5f, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x42, 0x0f, 0x0a, 0x0d, 0x5f, - 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x42, 0x0a, 0x0a, 0x08, - 0x5f, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x22, 0xb1, 0x01, 0x0a, 0x14, 0x52, 0x65, 0x70, - 0x6f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, 0x61, 0x67, - 0x65, 0x12, 0x41, 0x0a, 0x0b, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, - 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, - 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, - 0x6d, 0x70, 0x48, 0x00, 0x52, 0x0b, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, - 0x64, 0x88, 0x01, 0x01, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, - 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, - 0x70, 0x65, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x03, 0x20, 0x01, - 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x10, 0x0a, 0x03, 0x54, 0x61, - 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x54, 0x61, 0x67, 0x42, 0x0e, 0x0a, 0x0c, - 0x5f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x22, 0x8d, 0x07, 0x0a, - 0x0d, 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x12, 0x12, - 0x0a, 0x04, 0x4e, 0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x4e, 0x61, - 0x6d, 0x65, 0x12, 0x2f, 0x0a, 0x04, 0x54, 0x61, 0x67, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0b, - 0x32, 0x1b, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x52, 0x65, 0x70, 0x6f, 0x4d, - 0x65, 0x74, 0x61, 0x2e, 0x54, 0x61, 0x67, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x04, 0x54, - 0x61, 0x67, 0x73, 0x12, 0x41, 0x0a, 0x0a, 0x53, 0x74, 0x61, 0x74, 0x69, 0x73, 0x74, 0x69, 0x63, - 0x73, 0x18, 0x03, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x72, 0x6f, - 0x74, 0x6f, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x2e, 0x53, 0x74, 0x61, 0x74, 0x69, - 0x73, 0x74, 0x69, 0x63, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0a, 0x53, 0x74, 0x61, 0x74, - 0x69, 0x73, 0x74, 0x69, 0x63, 0x73, 0x12, 0x41, 0x0a, 0x0a, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, - 0x75, 0x72, 0x65, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x76, 0x31, 0x2e, - 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x2e, 0x53, 0x69, - 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0a, 0x53, - 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x12, 0x3e, 0x0a, 0x09, 0x52, 0x65, 0x66, - 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, 0x18, 0x05, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x20, 0x2e, 0x76, - 0x31, 0x2e, 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x2e, - 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x09, - 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, 0x12, 0x1c, 0x0a, 0x09, 0x49, 0x73, 0x53, - 0x74, 0x61, 0x72, 0x72, 0x65, 0x64, 0x18, 0x06, 0x20, 0x01, 0x28, 0x08, 0x52, 0x09, 0x49, 0x73, - 0x53, 0x74, 0x61, 0x72, 0x72, 0x65, 0x64, 0x12, 0x22, 0x0a, 0x0c, 0x49, 0x73, 0x42, 0x6f, 0x6f, - 0x6b, 0x6d, 0x61, 0x72, 0x6b, 0x65, 0x64, 0x18, 0x07, 0x20, 0x01, 0x28, 0x08, 0x52, 0x0c, 0x49, - 0x73, 0x42, 0x6f, 0x6f, 0x6b, 0x6d, 0x61, 0x72, 0x6b, 0x65, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x52, - 0x61, 0x6e, 0x6b, 0x18, 0x08, 0x20, 0x01, 0x28, 0x05, 0x52, 0x04, 0x52, 0x61, 0x6e, 0x6b, 0x12, - 0x14, 0x0a, 0x05, 0x53, 0x74, 0x61, 0x72, 0x73, 0x18, 0x09, 0x20, 0x01, 0x28, 0x05, 0x52, 0x05, - 0x53, 0x74, 0x61, 0x72, 0x73, 0x12, 0x12, 0x0a, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x18, 0x0a, 0x20, - 0x01, 0x28, 0x05, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x56, 0x65, 0x6e, - 0x64, 0x6f, 0x72, 0x73, 0x18, 0x0b, 0x20, 0x03, 0x28, 0x09, 0x52, 0x07, 0x56, 0x65, 0x6e, 0x64, - 0x6f, 0x72, 0x73, 0x12, 0x2a, 0x0a, 0x09, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x73, - 0x18, 0x0c, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x6c, 0x61, 0x74, - 0x66, 0x6f, 0x72, 0x6d, 0x52, 0x09, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x73, 0x12, - 0x49, 0x0a, 0x10, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, - 0x61, 0x67, 0x65, 0x18, 0x0d, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x18, 0x2e, 0x76, 0x31, 0x2e, 0x52, - 0x65, 0x70, 0x6f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, - 0x61, 0x67, 0x65, 0x48, 0x00, 0x52, 0x10, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, - 0x65, 0x64, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x88, 0x01, 0x01, 0x1a, 0x4a, 0x0a, 0x09, 0x54, 0x61, - 0x67, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, - 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x27, 0x0a, 0x05, 0x76, 0x61, 0x6c, - 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x11, 0x2e, 0x76, 0x31, 0x2e, 0x54, 0x61, - 0x67, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x05, 0x76, 0x61, 0x6c, - 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x57, 0x0a, 0x0f, 0x53, 0x74, 0x61, 0x74, 0x69, 0x73, - 0x74, 0x69, 0x63, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, - 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x2e, 0x0a, 0x05, 0x76, - 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x18, 0x2e, 0x76, 0x31, 0x2e, - 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x53, 0x74, 0x61, 0x74, 0x69, 0x73, - 0x74, 0x69, 0x63, 0x73, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, - 0x55, 0x0a, 0x0f, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x45, 0x6e, 0x74, - 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, - 0x03, 0x6b, 0x65, 0x79, 0x12, 0x2c, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, - 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x76, 0x31, 0x2e, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, - 0x74, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x52, 0x05, 0x76, 0x61, 0x6c, - 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x4f, 0x0a, 0x0e, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, - 0x65, 0x72, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, - 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x27, 0x0a, 0x05, 0x76, 0x61, - 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x11, 0x2e, 0x76, 0x31, 0x2e, 0x52, - 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x05, 0x76, 0x61, - 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x42, 0x13, 0x0a, 0x11, 0x5f, 0x4c, 0x61, 0x73, 0x74, - 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x22, 0x97, 0x01, 0x0a, - 0x09, 0x52, 0x65, 0x70, 0x6f, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x12, 0x12, 0x0a, 0x04, 0x4e, 0x61, - 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x4e, 0x61, 0x6d, 0x65, 0x12, 0x2e, - 0x0a, 0x05, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x18, 0x2e, - 0x76, 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x2e, 0x42, 0x6c, 0x6f, - 0x62, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x05, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x1a, 0x46, - 0x0a, 0x0a, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, - 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x22, - 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0c, 0x2e, - 0x76, 0x31, 0x2e, 0x42, 0x6c, 0x6f, 0x62, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x05, 0x76, 0x61, 0x6c, - 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0xd3, 0x01, 0x0a, 0x08, 0x42, 0x6c, 0x6f, 0x62, 0x49, - 0x6e, 0x66, 0x6f, 0x12, 0x12, 0x0a, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, - 0x03, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x56, 0x65, 0x6e, 0x64, 0x6f, - 0x72, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x09, 0x52, 0x07, 0x56, 0x65, 0x6e, 0x64, 0x6f, 0x72, - 0x73, 0x12, 0x2a, 0x0a, 0x09, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x73, 0x18, 0x03, - 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, - 0x72, 0x6d, 0x52, 0x09, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x73, 0x12, 0x1a, 0x0a, - 0x08, 0x53, 0x75, 0x62, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x09, 0x52, - 0x08, 0x53, 0x75, 0x62, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x12, 0x41, 0x0a, 0x0b, 0x4c, 0x61, 0x73, - 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, - 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, - 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x48, 0x00, 0x52, 0x0b, 0x4c, 0x61, - 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x88, 0x01, 0x01, 0x42, 0x0e, 0x0a, 0x0c, - 0x5f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x22, 0x3c, 0x0a, 0x14, - 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x53, 0x74, 0x61, 0x74, 0x69, 0x73, - 0x74, 0x69, 0x63, 0x73, 0x12, 0x24, 0x0a, 0x0d, 0x44, 0x6f, 0x77, 0x6e, 0x6c, 0x6f, 0x61, 0x64, - 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, 0x05, 0x52, 0x0d, 0x44, 0x6f, 0x77, - 0x6e, 0x6c, 0x6f, 0x61, 0x64, 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x22, 0x35, 0x0a, 0x0d, 0x52, 0x65, - 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x24, 0x0a, 0x04, 0x6c, - 0x69, 0x73, 0x74, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x76, 0x31, 0x2e, 0x52, - 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x04, 0x6c, 0x69, 0x73, - 0x74, 0x22, 0x81, 0x02, 0x0a, 0x0c, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x49, 0x6e, - 0x66, 0x6f, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x01, 0x20, 0x01, - 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, - 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, - 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x12, 0x22, 0x0a, 0x0c, 0x41, 0x72, 0x74, 0x69, - 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, - 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x12, 0x12, 0x0a, 0x04, - 0x53, 0x69, 0x7a, 0x65, 0x18, 0x04, 0x20, 0x01, 0x28, 0x03, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, - 0x12, 0x43, 0x0a, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, - 0x05, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x66, 0x65, 0x72, - 0x72, 0x65, 0x72, 0x49, 0x6e, 0x66, 0x6f, 0x2e, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, - 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, - 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x1a, 0x3e, 0x0a, 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, - 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, - 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, - 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, - 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0x93, 0x01, 0x0a, 0x12, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, - 0x73, 0x74, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x12, 0x31, 0x0a, 0x03, - 0x6d, 0x61, 0x70, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1f, 0x2e, 0x76, 0x31, 0x2e, 0x4d, - 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, - 0x73, 0x2e, 0x4d, 0x61, 0x70, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x03, 0x6d, 0x61, 0x70, 0x1a, - 0x4a, 0x0a, 0x08, 0x4d, 0x61, 0x70, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, - 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x28, 0x0a, - 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x12, 0x2e, 0x76, - 0x31, 0x2e, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x49, 0x6e, 0x66, 0x6f, - 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0x37, 0x0a, 0x0e, 0x53, - 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x25, 0x0a, - 0x04, 0x6c, 0x69, 0x73, 0x74, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x11, 0x2e, 0x76, 0x31, - 0x2e, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x04, - 0x6c, 0x69, 0x73, 0x74, 0x22, 0x79, 0x0a, 0x0d, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, - 0x65, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x38, 0x0a, 0x17, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, - 0x72, 0x65, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, - 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x17, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, - 0x65, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, - 0x2e, 0x0a, 0x0a, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x18, 0x02, 0x20, - 0x03, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, - 0x6e, 0x66, 0x6f, 0x52, 0x0a, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x22, - 0xbe, 0x01, 0x0a, 0x0a, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x20, - 0x0a, 0x0b, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x01, 0x20, - 0x01, 0x28, 0x09, 0x52, 0x0b, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, - 0x12, 0x22, 0x0a, 0x0c, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x43, 0x6f, 0x6e, 0x74, 0x65, 0x6e, 0x74, - 0x18, 0x02, 0x20, 0x01, 0x28, 0x0c, 0x52, 0x0c, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x43, 0x6f, 0x6e, - 0x74, 0x65, 0x6e, 0x74, 0x12, 0x22, 0x0a, 0x0c, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, - 0x65, 0x4b, 0x65, 0x79, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x53, 0x69, 0x67, 0x6e, - 0x61, 0x74, 0x75, 0x72, 0x65, 0x4b, 0x65, 0x79, 0x12, 0x16, 0x0a, 0x06, 0x53, 0x69, 0x67, 0x6e, - 0x65, 0x72, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x53, 0x69, 0x67, 0x6e, 0x65, 0x72, - 0x12, 0x2e, 0x0a, 0x04, 0x44, 0x61, 0x74, 0x65, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, - 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, - 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, 0x04, 0x44, 0x61, 0x74, 0x65, - 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, -} - -var ( - file_imageData_proto_rawDescOnce sync.Once - file_imageData_proto_rawDescData = file_imageData_proto_rawDesc -) - -func file_imageData_proto_rawDescGZIP() []byte { - file_imageData_proto_rawDescOnce.Do(func() { - file_imageData_proto_rawDescData = protoimpl.X.CompressGZIP(file_imageData_proto_rawDescData) - }) - return file_imageData_proto_rawDescData -} - -var file_imageData_proto_msgTypes = make([]protoimpl.MessageInfo, 23) -var file_imageData_proto_goTypes = []interface{}{ - (*TagDescriptor)(nil), // 0: v1.TagDescriptor - (*ImageData)(nil), // 1: v1.ImageData - (*ManifestData)(nil), // 2: v1.ManifestData - (*RepoLastUpdatedImage)(nil), // 3: v1.RepoLastUpdatedImage - (*ProtoRepoMeta)(nil), // 4: v1.ProtoRepoMeta - (*RepoBlobs)(nil), // 5: v1.RepoBlobs - (*BlobInfo)(nil), // 6: v1.BlobInfo - (*DescriptorStatistics)(nil), // 7: v1.DescriptorStatistics - (*ReferrersInfo)(nil), // 8: v1.ReferrersInfo - (*ReferrerInfo)(nil), // 9: v1.ReferrerInfo - (*ManifestSignatures)(nil), // 10: v1.ManifestSignatures - (*SignaturesInfo)(nil), // 11: v1.SignaturesInfo - (*SignatureInfo)(nil), // 12: v1.SignatureInfo - (*LayersInfo)(nil), // 13: v1.LayersInfo - nil, // 14: v1.ImageData.AnnotationsEntry - nil, // 15: v1.ManifestData.AnnotationsEntry - nil, // 16: v1.ProtoRepoMeta.TagsEntry - nil, // 17: v1.ProtoRepoMeta.StatisticsEntry - nil, // 18: v1.ProtoRepoMeta.SignaturesEntry - nil, // 19: v1.ProtoRepoMeta.ReferrersEntry - nil, // 20: v1.RepoBlobs.BlobsEntry - nil, // 21: v1.ReferrerInfo.AnnotationsEntry - nil, // 22: v1.ManifestSignatures.MapEntry - (*Versioned)(nil), // 23: v1.Versioned - (*Descriptor)(nil), // 24: v1.Descriptor - (*ConfigData)(nil), // 25: v1.ConfigData - (*timestamppb.Timestamp)(nil), // 26: google.protobuf.Timestamp - (*Platform)(nil), // 27: v1.Platform -} -var file_imageData_proto_depIdxs = []int32{ - 23, // 0: v1.ImageData.Versioned:type_name -> v1.Versioned - 2, // 1: v1.ImageData.Manifests:type_name -> v1.ManifestData - 24, // 2: v1.ImageData.Subject:type_name -> v1.Descriptor - 14, // 3: v1.ImageData.Annotations:type_name -> v1.ImageData.AnnotationsEntry - 23, // 4: v1.ManifestData.versioned:type_name -> v1.Versioned - 24, // 5: v1.ManifestData.Layers:type_name -> v1.Descriptor - 24, // 6: v1.ManifestData.Subject:type_name -> v1.Descriptor - 15, // 7: v1.ManifestData.Annotations:type_name -> v1.ManifestData.AnnotationsEntry - 25, // 8: v1.ManifestData.Config:type_name -> v1.ConfigData - 26, // 9: v1.RepoLastUpdatedImage.LastUpdated:type_name -> google.protobuf.Timestamp - 16, // 10: v1.ProtoRepoMeta.Tags:type_name -> v1.ProtoRepoMeta.TagsEntry - 17, // 11: v1.ProtoRepoMeta.Statistics:type_name -> v1.ProtoRepoMeta.StatisticsEntry - 18, // 12: v1.ProtoRepoMeta.Signatures:type_name -> v1.ProtoRepoMeta.SignaturesEntry - 19, // 13: v1.ProtoRepoMeta.Referrers:type_name -> v1.ProtoRepoMeta.ReferrersEntry - 27, // 14: v1.ProtoRepoMeta.Platforms:type_name -> v1.Platform - 3, // 15: v1.ProtoRepoMeta.LastUpdatedImage:type_name -> v1.RepoLastUpdatedImage - 20, // 16: v1.RepoBlobs.Blobs:type_name -> v1.RepoBlobs.BlobsEntry - 27, // 17: v1.BlobInfo.Platforms:type_name -> v1.Platform - 26, // 18: v1.BlobInfo.LastUpdated:type_name -> google.protobuf.Timestamp - 9, // 19: v1.ReferrersInfo.list:type_name -> v1.ReferrerInfo - 21, // 20: v1.ReferrerInfo.Annotations:type_name -> v1.ReferrerInfo.AnnotationsEntry - 22, // 21: v1.ManifestSignatures.map:type_name -> v1.ManifestSignatures.MapEntry - 12, // 22: v1.SignaturesInfo.list:type_name -> v1.SignatureInfo - 13, // 23: v1.SignatureInfo.LayersInfo:type_name -> v1.LayersInfo - 26, // 24: v1.LayersInfo.Date:type_name -> google.protobuf.Timestamp - 0, // 25: v1.ProtoRepoMeta.TagsEntry.value:type_name -> v1.TagDescriptor - 7, // 26: v1.ProtoRepoMeta.StatisticsEntry.value:type_name -> v1.DescriptorStatistics - 10, // 27: v1.ProtoRepoMeta.SignaturesEntry.value:type_name -> v1.ManifestSignatures - 8, // 28: v1.ProtoRepoMeta.ReferrersEntry.value:type_name -> v1.ReferrersInfo - 6, // 29: v1.RepoBlobs.BlobsEntry.value:type_name -> v1.BlobInfo - 11, // 30: v1.ManifestSignatures.MapEntry.value:type_name -> v1.SignaturesInfo - 31, // [31:31] is the sub-list for method output_type - 31, // [31:31] is the sub-list for method input_type - 31, // [31:31] is the sub-list for extension type_name - 31, // [31:31] is the sub-list for extension extendee - 0, // [0:31] is the sub-list for field type_name -} - -func init() { file_imageData_proto_init() } -func file_imageData_proto_init() { - if File_imageData_proto != nil { - return - } - file_versioned_proto_init() - file_config_proto_init() - file_descriptor_proto_init() - if !protoimpl.UnsafeEnabled { - file_imageData_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*TagDescriptor); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ImageData); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ManifestData); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*RepoLastUpdatedImage); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[4].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ProtoRepoMeta); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[5].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*RepoBlobs); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[6].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*BlobInfo); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[7].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*DescriptorStatistics); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[8].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ReferrersInfo); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[9].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ReferrerInfo); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[10].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*ManifestSignatures); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[11].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*SignaturesInfo); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[12].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*SignatureInfo); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - file_imageData_proto_msgTypes[13].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*LayersInfo); i { - case 0: - return &v.state - case 1: - return &v.sizeCache - case 2: - return &v.unknownFields - default: - return nil - } - } - } - file_imageData_proto_msgTypes[1].OneofWrappers = []interface{}{} - file_imageData_proto_msgTypes[2].OneofWrappers = []interface{}{} - file_imageData_proto_msgTypes[3].OneofWrappers = []interface{}{} - file_imageData_proto_msgTypes[4].OneofWrappers = []interface{}{} - file_imageData_proto_msgTypes[6].OneofWrappers = []interface{}{} - type x struct{} - out := protoimpl.TypeBuilder{ - File: protoimpl.DescBuilder{ - GoPackagePath: reflect.TypeOf(x{}).PkgPath(), - RawDescriptor: file_imageData_proto_rawDesc, - NumEnums: 0, - NumMessages: 23, - NumExtensions: 0, - NumServices: 0, - }, - GoTypes: file_imageData_proto_goTypes, - DependencyIndexes: file_imageData_proto_depIdxs, - MessageInfos: file_imageData_proto_msgTypes, - }.Build() - File_imageData_proto = out.File - file_imageData_proto_rawDesc = nil - file_imageData_proto_goTypes = nil - file_imageData_proto_depIdxs = nil -} diff --git a/pkg/meta/proto_go/index.pb.go b/pkg/meta/proto_go/index.pb.go index 105d456566..34dafa637d 100644 --- a/pkg/meta/proto_go/index.pb.go +++ b/pkg/meta/proto_go/index.pb.go @@ -25,12 +25,12 @@ type Index struct { sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - Versioned *Versioned `protobuf:"bytes,1,opt,name=versioned,proto3" json:"versioned,omitempty"` - Mediatype *string `protobuf:"bytes,2,opt,name=mediatype,proto3,oneof" json:"mediatype,omitempty"` - Artifacttype *string `protobuf:"bytes,3,opt,name=artifacttype,proto3,oneof" json:"artifacttype,omitempty"` - Manifests []*Descriptor `protobuf:"bytes,4,rep,name=manifests,proto3" json:"manifests,omitempty"` - Subject *Descriptor `protobuf:"bytes,5,opt,name=subject,proto3,oneof" json:"subject,omitempty"` - Annotations map[string]string `protobuf:"bytes,6,rep,name=annotations,proto3" json:"annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + Versioned *Versioned `protobuf:"bytes,1,opt,name=Versioned,proto3" json:"Versioned,omitempty"` + MediaType *string `protobuf:"bytes,2,opt,name=MediaType,proto3,oneof" json:"MediaType,omitempty"` + ArtifactType *string `protobuf:"bytes,3,opt,name=ArtifactType,proto3,oneof" json:"ArtifactType,omitempty"` + Manifests []*Descriptor `protobuf:"bytes,4,rep,name=Manifests,proto3" json:"Manifests,omitempty"` + Subject *Descriptor `protobuf:"bytes,5,opt,name=Subject,proto3,oneof" json:"Subject,omitempty"` + Annotations map[string]string `protobuf:"bytes,6,rep,name=Annotations,proto3" json:"Annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` } func (x *Index) Reset() { @@ -72,16 +72,16 @@ func (x *Index) GetVersioned() *Versioned { return nil } -func (x *Index) GetMediatype() string { - if x != nil && x.Mediatype != nil { - return *x.Mediatype +func (x *Index) GetMediaType() string { + if x != nil && x.MediaType != nil { + return *x.MediaType } return "" } -func (x *Index) GetArtifacttype() string { - if x != nil && x.Artifacttype != nil { - return *x.Artifacttype +func (x *Index) GetArtifactType() string { + if x != nil && x.ArtifactType != nil { + return *x.ArtifactType } return "" } @@ -114,30 +114,30 @@ var file_index_proto_rawDesc = []byte{ 0x31, 0x1a, 0x10, 0x64, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x0f, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0x86, 0x03, 0x0a, 0x05, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x12, 0x2b, - 0x0a, 0x09, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, + 0x0a, 0x09, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0d, 0x2e, 0x76, 0x31, 0x2e, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, - 0x52, 0x09, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x12, 0x21, 0x0a, 0x09, 0x6d, - 0x65, 0x64, 0x69, 0x61, 0x74, 0x79, 0x70, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, - 0x52, 0x09, 0x6d, 0x65, 0x64, 0x69, 0x61, 0x74, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x27, - 0x0a, 0x0c, 0x61, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x74, 0x79, 0x70, 0x65, 0x18, 0x03, - 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x0c, 0x61, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, - 0x74, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x2c, 0x0a, 0x09, 0x6d, 0x61, 0x6e, 0x69, 0x66, + 0x52, 0x09, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x12, 0x21, 0x0a, 0x09, 0x4d, + 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, + 0x52, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x27, + 0x0a, 0x0c, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x18, 0x03, + 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x0c, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, + 0x54, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x2c, 0x0a, 0x09, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, - 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x09, 0x6d, 0x61, 0x6e, 0x69, - 0x66, 0x65, 0x73, 0x74, 0x73, 0x12, 0x2d, 0x0a, 0x07, 0x73, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, + 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x09, 0x4d, 0x61, 0x6e, 0x69, + 0x66, 0x65, 0x73, 0x74, 0x73, 0x12, 0x2d, 0x0a, 0x07, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, - 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x48, 0x02, 0x52, 0x07, 0x73, 0x75, 0x62, 0x6a, 0x65, 0x63, - 0x74, 0x88, 0x01, 0x01, 0x12, 0x3c, 0x0a, 0x0b, 0x61, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, + 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x48, 0x02, 0x52, 0x07, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, + 0x74, 0x88, 0x01, 0x01, 0x12, 0x3c, 0x0a, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x06, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x2e, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, - 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x61, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, + 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x1a, 0x3e, 0x0a, 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, - 0x38, 0x01, 0x42, 0x0c, 0x0a, 0x0a, 0x5f, 0x6d, 0x65, 0x64, 0x69, 0x61, 0x74, 0x79, 0x70, 0x65, - 0x42, 0x0f, 0x0a, 0x0d, 0x5f, 0x61, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x74, 0x79, 0x70, - 0x65, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x73, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x62, 0x06, 0x70, + 0x38, 0x01, 0x42, 0x0c, 0x0a, 0x0a, 0x5f, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, + 0x42, 0x0f, 0x0a, 0x0d, 0x5f, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, + 0x65, 0x42, 0x0a, 0x0a, 0x08, 0x5f, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, } @@ -161,10 +161,10 @@ var file_index_proto_goTypes = []interface{}{ (*Descriptor)(nil), // 3: v1.Descriptor } var file_index_proto_depIdxs = []int32{ - 2, // 0: v1.Index.versioned:type_name -> v1.Versioned - 3, // 1: v1.Index.manifests:type_name -> v1.Descriptor - 3, // 2: v1.Index.subject:type_name -> v1.Descriptor - 1, // 3: v1.Index.annotations:type_name -> v1.Index.AnnotationsEntry + 2, // 0: v1.Index.Versioned:type_name -> v1.Versioned + 3, // 1: v1.Index.Manifests:type_name -> v1.Descriptor + 3, // 2: v1.Index.Subject:type_name -> v1.Descriptor + 1, // 3: v1.Index.Annotations:type_name -> v1.Index.AnnotationsEntry 4, // [4:4] is the sub-list for method output_type 4, // [4:4] is the sub-list for method input_type 4, // [4:4] is the sub-list for extension type_name diff --git a/pkg/meta/proto_go/manifest.pb.go b/pkg/meta/proto_go/manifest.pb.go index 0032207340..628c5b809b 100644 --- a/pkg/meta/proto_go/manifest.pb.go +++ b/pkg/meta/proto_go/manifest.pb.go @@ -25,13 +25,13 @@ type Manifest struct { sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - Versioned *Versioned `protobuf:"bytes,1,opt,name=versioned,proto3" json:"versioned,omitempty"` - Mediatype *string `protobuf:"bytes,2,opt,name=mediatype,proto3,oneof" json:"mediatype,omitempty"` - Artifacttype *string `protobuf:"bytes,3,opt,name=artifacttype,proto3,oneof" json:"artifacttype,omitempty"` - Config *Descriptor `protobuf:"bytes,4,opt,name=config,proto3" json:"config,omitempty"` - Layers []*Descriptor `protobuf:"bytes,5,rep,name=layers,proto3" json:"layers,omitempty"` - Subject *Descriptor `protobuf:"bytes,6,opt,name=subject,proto3,oneof" json:"subject,omitempty"` - Annotations map[string]string `protobuf:"bytes,7,rep,name=annotations,proto3" json:"annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + Versioned *Versioned `protobuf:"bytes,1,opt,name=Versioned,proto3" json:"Versioned,omitempty"` + MediaType *string `protobuf:"bytes,2,opt,name=MediaType,proto3,oneof" json:"MediaType,omitempty"` + ArtifactType *string `protobuf:"bytes,3,opt,name=ArtifactType,proto3,oneof" json:"ArtifactType,omitempty"` + Config *Descriptor `protobuf:"bytes,4,opt,name=Config,proto3" json:"Config,omitempty"` + Layers []*Descriptor `protobuf:"bytes,5,rep,name=Layers,proto3" json:"Layers,omitempty"` + Subject *Descriptor `protobuf:"bytes,6,opt,name=Subject,proto3,oneof" json:"Subject,omitempty"` + Annotations map[string]string `protobuf:"bytes,7,rep,name=Annotations,proto3" json:"Annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` } func (x *Manifest) Reset() { @@ -73,16 +73,16 @@ func (x *Manifest) GetVersioned() *Versioned { return nil } -func (x *Manifest) GetMediatype() string { - if x != nil && x.Mediatype != nil { - return *x.Mediatype +func (x *Manifest) GetMediaType() string { + if x != nil && x.MediaType != nil { + return *x.MediaType } return "" } -func (x *Manifest) GetArtifacttype() string { - if x != nil && x.Artifacttype != nil { - return *x.Artifacttype +func (x *Manifest) GetArtifactType() string { + if x != nil && x.ArtifactType != nil { + return *x.ArtifactType } return "" } @@ -122,33 +122,33 @@ var file_manifest_proto_rawDesc = []byte{ 0x12, 0x02, 0x76, 0x31, 0x1a, 0x10, 0x64, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x0f, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0xae, 0x03, 0x0a, 0x08, 0x4d, 0x61, 0x6e, 0x69, - 0x66, 0x65, 0x73, 0x74, 0x12, 0x2b, 0x0a, 0x09, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, + 0x66, 0x65, 0x73, 0x74, 0x12, 0x2b, 0x0a, 0x09, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0d, 0x2e, 0x76, 0x31, 0x2e, 0x56, 0x65, 0x72, - 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x52, 0x09, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, - 0x64, 0x12, 0x21, 0x0a, 0x09, 0x6d, 0x65, 0x64, 0x69, 0x61, 0x74, 0x79, 0x70, 0x65, 0x18, 0x02, - 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, 0x52, 0x09, 0x6d, 0x65, 0x64, 0x69, 0x61, 0x74, 0x79, 0x70, - 0x65, 0x88, 0x01, 0x01, 0x12, 0x27, 0x0a, 0x0c, 0x61, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, - 0x74, 0x79, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x0c, 0x61, 0x72, - 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x74, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x26, 0x0a, - 0x06, 0x63, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, - 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x06, 0x63, - 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x12, 0x26, 0x0a, 0x06, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x18, + 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x52, 0x09, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, + 0x64, 0x12, 0x21, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x02, + 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, 0x52, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, + 0x65, 0x88, 0x01, 0x01, 0x12, 0x27, 0x0a, 0x0c, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, + 0x54, 0x79, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x0c, 0x41, 0x72, + 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x88, 0x01, 0x01, 0x12, 0x26, 0x0a, + 0x06, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, + 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x06, 0x43, + 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x12, 0x26, 0x0a, 0x06, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x18, 0x05, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, - 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x06, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x12, 0x2d, 0x0a, - 0x07, 0x73, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, + 0x69, 0x70, 0x74, 0x6f, 0x72, 0x52, 0x06, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x12, 0x2d, 0x0a, + 0x07, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x48, 0x02, - 0x52, 0x07, 0x73, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x88, 0x01, 0x01, 0x12, 0x3f, 0x0a, 0x0b, - 0x61, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x07, 0x20, 0x03, 0x28, + 0x52, 0x07, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x88, 0x01, 0x01, 0x12, 0x3f, 0x0a, 0x0b, + 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x07, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1d, 0x2e, 0x76, 0x31, 0x2e, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x2e, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, - 0x52, 0x0b, 0x61, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x1a, 0x3e, 0x0a, + 0x52, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x1a, 0x3e, 0x0a, 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x42, 0x0c, 0x0a, - 0x0a, 0x5f, 0x6d, 0x65, 0x64, 0x69, 0x61, 0x74, 0x79, 0x70, 0x65, 0x42, 0x0f, 0x0a, 0x0d, 0x5f, - 0x61, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x74, 0x79, 0x70, 0x65, 0x42, 0x0a, 0x0a, 0x08, - 0x5f, 0x73, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, + 0x0a, 0x5f, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x42, 0x0f, 0x0a, 0x0d, 0x5f, + 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x42, 0x0a, 0x0a, 0x08, + 0x5f, 0x53, 0x75, 0x62, 0x6a, 0x65, 0x63, 0x74, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, } var ( @@ -171,11 +171,11 @@ var file_manifest_proto_goTypes = []interface{}{ (*Descriptor)(nil), // 3: v1.Descriptor } var file_manifest_proto_depIdxs = []int32{ - 2, // 0: v1.Manifest.versioned:type_name -> v1.Versioned - 3, // 1: v1.Manifest.config:type_name -> v1.Descriptor - 3, // 2: v1.Manifest.layers:type_name -> v1.Descriptor - 3, // 3: v1.Manifest.subject:type_name -> v1.Descriptor - 1, // 4: v1.Manifest.annotations:type_name -> v1.Manifest.AnnotationsEntry + 2, // 0: v1.Manifest.Versioned:type_name -> v1.Versioned + 3, // 1: v1.Manifest.Config:type_name -> v1.Descriptor + 3, // 2: v1.Manifest.Layers:type_name -> v1.Descriptor + 3, // 3: v1.Manifest.Subject:type_name -> v1.Descriptor + 1, // 4: v1.Manifest.Annotations:type_name -> v1.Manifest.AnnotationsEntry 5, // [5:5] is the sub-list for method output_type 5, // [5:5] is the sub-list for method input_type 5, // [5:5] is the sub-list for extension type_name diff --git a/pkg/meta/proto_go/meta.pb.go b/pkg/meta/proto_go/meta.pb.go new file mode 100644 index 0000000000..fffff3995f --- /dev/null +++ b/pkg/meta/proto_go/meta.pb.go @@ -0,0 +1,1512 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.31.0 +// protoc v3.15.8 +// source: meta.proto + +package proto_go + +import ( + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + timestamppb "google.golang.org/protobuf/types/known/timestamppb" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type TagDescriptor struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + MediaType string `protobuf:"bytes,1,opt,name=MediaType,proto3" json:"MediaType,omitempty"` + Digest string `protobuf:"bytes,2,opt,name=Digest,proto3" json:"Digest,omitempty"` +} + +func (x *TagDescriptor) Reset() { + *x = TagDescriptor{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *TagDescriptor) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*TagDescriptor) ProtoMessage() {} + +func (x *TagDescriptor) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use TagDescriptor.ProtoReflect.Descriptor instead. +func (*TagDescriptor) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{0} +} + +func (x *TagDescriptor) GetMediaType() string { + if x != nil { + return x.MediaType + } + return "" +} + +func (x *TagDescriptor) GetDigest() string { + if x != nil { + return x.Digest + } + return "" +} + +type ImageData struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + MediaType string `protobuf:"bytes,1,opt,name=MediaType,proto3" json:"MediaType,omitempty"` + Manifests []*ManifestData `protobuf:"bytes,2,rep,name=Manifests,proto3" json:"Manifests,omitempty"` + Index *IndexData `protobuf:"bytes,3,opt,name=Index,proto3,oneof" json:"Index,omitempty"` +} + +func (x *ImageData) Reset() { + *x = ImageData{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ImageData) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ImageData) ProtoMessage() {} + +func (x *ImageData) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ImageData.ProtoReflect.Descriptor instead. +func (*ImageData) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{1} +} + +func (x *ImageData) GetMediaType() string { + if x != nil { + return x.MediaType + } + return "" +} + +func (x *ImageData) GetManifests() []*ManifestData { + if x != nil { + return x.Manifests + } + return nil +} + +func (x *ImageData) GetIndex() *IndexData { + if x != nil { + return x.Index + } + return nil +} + +type ManifestData struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Digest string `protobuf:"bytes,1,opt,name=Digest,proto3" json:"Digest,omitempty"` + Size int64 `protobuf:"varint,2,opt,name=Size,proto3" json:"Size,omitempty"` + Manifest *Manifest `protobuf:"bytes,3,opt,name=Manifest,proto3" json:"Manifest,omitempty"` + Config *Image `protobuf:"bytes,4,opt,name=Config,proto3" json:"Config,omitempty"` +} + +func (x *ManifestData) Reset() { + *x = ManifestData{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[2] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ManifestData) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ManifestData) ProtoMessage() {} + +func (x *ManifestData) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[2] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ManifestData.ProtoReflect.Descriptor instead. +func (*ManifestData) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{2} +} + +func (x *ManifestData) GetDigest() string { + if x != nil { + return x.Digest + } + return "" +} + +func (x *ManifestData) GetSize() int64 { + if x != nil { + return x.Size + } + return 0 +} + +func (x *ManifestData) GetManifest() *Manifest { + if x != nil { + return x.Manifest + } + return nil +} + +func (x *ManifestData) GetConfig() *Image { + if x != nil { + return x.Config + } + return nil +} + +type IndexData struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Digest string `protobuf:"bytes,1,opt,name=Digest,proto3" json:"Digest,omitempty"` + Size int64 `protobuf:"varint,2,opt,name=Size,proto3" json:"Size,omitempty"` + Index *Index `protobuf:"bytes,3,opt,name=Index,proto3" json:"Index,omitempty"` +} + +func (x *IndexData) Reset() { + *x = IndexData{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[3] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *IndexData) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*IndexData) ProtoMessage() {} + +func (x *IndexData) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[3] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use IndexData.ProtoReflect.Descriptor instead. +func (*IndexData) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{3} +} + +func (x *IndexData) GetDigest() string { + if x != nil { + return x.Digest + } + return "" +} + +func (x *IndexData) GetSize() int64 { + if x != nil { + return x.Size + } + return 0 +} + +func (x *IndexData) GetIndex() *Index { + if x != nil { + return x.Index + } + return nil +} + +type RepoLastUpdatedImage struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + LastUpdated *timestamppb.Timestamp `protobuf:"bytes,1,opt,name=LastUpdated,proto3,oneof" json:"LastUpdated,omitempty"` + MediaType string `protobuf:"bytes,2,opt,name=MediaType,proto3" json:"MediaType,omitempty"` + Digest string `protobuf:"bytes,3,opt,name=Digest,proto3" json:"Digest,omitempty"` + Tag string `protobuf:"bytes,4,opt,name=Tag,proto3" json:"Tag,omitempty"` +} + +func (x *RepoLastUpdatedImage) Reset() { + *x = RepoLastUpdatedImage{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[4] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *RepoLastUpdatedImage) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*RepoLastUpdatedImage) ProtoMessage() {} + +func (x *RepoLastUpdatedImage) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[4] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use RepoLastUpdatedImage.ProtoReflect.Descriptor instead. +func (*RepoLastUpdatedImage) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{4} +} + +func (x *RepoLastUpdatedImage) GetLastUpdated() *timestamppb.Timestamp { + if x != nil { + return x.LastUpdated + } + return nil +} + +func (x *RepoLastUpdatedImage) GetMediaType() string { + if x != nil { + return x.MediaType + } + return "" +} + +func (x *RepoLastUpdatedImage) GetDigest() string { + if x != nil { + return x.Digest + } + return "" +} + +func (x *RepoLastUpdatedImage) GetTag() string { + if x != nil { + return x.Tag + } + return "" +} + +type RepoMeta struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Name string `protobuf:"bytes,1,opt,name=Name,proto3" json:"Name,omitempty"` + Tags map[string]*TagDescriptor `protobuf:"bytes,2,rep,name=Tags,proto3" json:"Tags,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + Statistics map[string]*DescriptorStatistics `protobuf:"bytes,3,rep,name=Statistics,proto3" json:"Statistics,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + Signatures map[string]*ManifestSignatures `protobuf:"bytes,4,rep,name=Signatures,proto3" json:"Signatures,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + Referrers map[string]*ReferrersInfo `protobuf:"bytes,5,rep,name=Referrers,proto3" json:"Referrers,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` + IsStarred bool `protobuf:"varint,6,opt,name=IsStarred,proto3" json:"IsStarred,omitempty"` + IsBookmarked bool `protobuf:"varint,7,opt,name=IsBookmarked,proto3" json:"IsBookmarked,omitempty"` + Rank int32 `protobuf:"varint,8,opt,name=Rank,proto3" json:"Rank,omitempty"` + Stars int32 `protobuf:"varint,9,opt,name=Stars,proto3" json:"Stars,omitempty"` + Size int32 `protobuf:"varint,10,opt,name=Size,proto3" json:"Size,omitempty"` + Vendors []string `protobuf:"bytes,11,rep,name=Vendors,proto3" json:"Vendors,omitempty"` + Platforms []*Platform `protobuf:"bytes,12,rep,name=Platforms,proto3" json:"Platforms,omitempty"` + LastUpdatedImage *RepoLastUpdatedImage `protobuf:"bytes,13,opt,name=LastUpdatedImage,proto3,oneof" json:"LastUpdatedImage,omitempty"` +} + +func (x *RepoMeta) Reset() { + *x = RepoMeta{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[5] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *RepoMeta) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*RepoMeta) ProtoMessage() {} + +func (x *RepoMeta) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[5] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use RepoMeta.ProtoReflect.Descriptor instead. +func (*RepoMeta) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{5} +} + +func (x *RepoMeta) GetName() string { + if x != nil { + return x.Name + } + return "" +} + +func (x *RepoMeta) GetTags() map[string]*TagDescriptor { + if x != nil { + return x.Tags + } + return nil +} + +func (x *RepoMeta) GetStatistics() map[string]*DescriptorStatistics { + if x != nil { + return x.Statistics + } + return nil +} + +func (x *RepoMeta) GetSignatures() map[string]*ManifestSignatures { + if x != nil { + return x.Signatures + } + return nil +} + +func (x *RepoMeta) GetReferrers() map[string]*ReferrersInfo { + if x != nil { + return x.Referrers + } + return nil +} + +func (x *RepoMeta) GetIsStarred() bool { + if x != nil { + return x.IsStarred + } + return false +} + +func (x *RepoMeta) GetIsBookmarked() bool { + if x != nil { + return x.IsBookmarked + } + return false +} + +func (x *RepoMeta) GetRank() int32 { + if x != nil { + return x.Rank + } + return 0 +} + +func (x *RepoMeta) GetStars() int32 { + if x != nil { + return x.Stars + } + return 0 +} + +func (x *RepoMeta) GetSize() int32 { + if x != nil { + return x.Size + } + return 0 +} + +func (x *RepoMeta) GetVendors() []string { + if x != nil { + return x.Vendors + } + return nil +} + +func (x *RepoMeta) GetPlatforms() []*Platform { + if x != nil { + return x.Platforms + } + return nil +} + +func (x *RepoMeta) GetLastUpdatedImage() *RepoLastUpdatedImage { + if x != nil { + return x.LastUpdatedImage + } + return nil +} + +type RepoBlobs struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Name string `protobuf:"bytes,1,opt,name=Name,proto3" json:"Name,omitempty"` + Blobs map[string]*BlobInfo `protobuf:"bytes,2,rep,name=Blobs,proto3" json:"Blobs,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` +} + +func (x *RepoBlobs) Reset() { + *x = RepoBlobs{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[6] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *RepoBlobs) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*RepoBlobs) ProtoMessage() {} + +func (x *RepoBlobs) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[6] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use RepoBlobs.ProtoReflect.Descriptor instead. +func (*RepoBlobs) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{6} +} + +func (x *RepoBlobs) GetName() string { + if x != nil { + return x.Name + } + return "" +} + +func (x *RepoBlobs) GetBlobs() map[string]*BlobInfo { + if x != nil { + return x.Blobs + } + return nil +} + +// for example this is a manifest and it has a config, and layers +// or index and has manifests +type BlobInfo struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Size int64 `protobuf:"varint,1,opt,name=Size,proto3" json:"Size,omitempty"` + Vendors []string `protobuf:"bytes,2,rep,name=Vendors,proto3" json:"Vendors,omitempty"` + Platforms []*Platform `protobuf:"bytes,3,rep,name=Platforms,proto3" json:"Platforms,omitempty"` + SubBlobs []string `protobuf:"bytes,4,rep,name=SubBlobs,proto3" json:"SubBlobs,omitempty"` + LastUpdated *timestamppb.Timestamp `protobuf:"bytes,5,opt,name=LastUpdated,proto3,oneof" json:"LastUpdated,omitempty"` +} + +func (x *BlobInfo) Reset() { + *x = BlobInfo{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[7] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *BlobInfo) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*BlobInfo) ProtoMessage() {} + +func (x *BlobInfo) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[7] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use BlobInfo.ProtoReflect.Descriptor instead. +func (*BlobInfo) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{7} +} + +func (x *BlobInfo) GetSize() int64 { + if x != nil { + return x.Size + } + return 0 +} + +func (x *BlobInfo) GetVendors() []string { + if x != nil { + return x.Vendors + } + return nil +} + +func (x *BlobInfo) GetPlatforms() []*Platform { + if x != nil { + return x.Platforms + } + return nil +} + +func (x *BlobInfo) GetSubBlobs() []string { + if x != nil { + return x.SubBlobs + } + return nil +} + +func (x *BlobInfo) GetLastUpdated() *timestamppb.Timestamp { + if x != nil { + return x.LastUpdated + } + return nil +} + +type DescriptorStatistics struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + DownloadCount int32 `protobuf:"varint,1,opt,name=DownloadCount,proto3" json:"DownloadCount,omitempty"` +} + +func (x *DescriptorStatistics) Reset() { + *x = DescriptorStatistics{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[8] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *DescriptorStatistics) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*DescriptorStatistics) ProtoMessage() {} + +func (x *DescriptorStatistics) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[8] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use DescriptorStatistics.ProtoReflect.Descriptor instead. +func (*DescriptorStatistics) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{8} +} + +func (x *DescriptorStatistics) GetDownloadCount() int32 { + if x != nil { + return x.DownloadCount + } + return 0 +} + +type ReferrersInfo struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + List []*ReferrerInfo `protobuf:"bytes,1,rep,name=list,proto3" json:"list,omitempty"` +} + +func (x *ReferrersInfo) Reset() { + *x = ReferrersInfo{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[9] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ReferrersInfo) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ReferrersInfo) ProtoMessage() {} + +func (x *ReferrersInfo) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[9] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ReferrersInfo.ProtoReflect.Descriptor instead. +func (*ReferrersInfo) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{9} +} + +func (x *ReferrersInfo) GetList() []*ReferrerInfo { + if x != nil { + return x.List + } + return nil +} + +type ReferrerInfo struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Digest string `protobuf:"bytes,1,opt,name=Digest,proto3" json:"Digest,omitempty"` + Count int64 `protobuf:"varint,2,opt,name=Count,proto3" json:"Count,omitempty"` + MediaType string `protobuf:"bytes,3,opt,name=MediaType,proto3" json:"MediaType,omitempty"` + ArtifactType string `protobuf:"bytes,4,opt,name=ArtifactType,proto3" json:"ArtifactType,omitempty"` + Size int64 `protobuf:"varint,5,opt,name=Size,proto3" json:"Size,omitempty"` + Annotations map[string]string `protobuf:"bytes,6,rep,name=Annotations,proto3" json:"Annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` +} + +func (x *ReferrerInfo) Reset() { + *x = ReferrerInfo{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[10] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ReferrerInfo) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ReferrerInfo) ProtoMessage() {} + +func (x *ReferrerInfo) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[10] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ReferrerInfo.ProtoReflect.Descriptor instead. +func (*ReferrerInfo) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{10} +} + +func (x *ReferrerInfo) GetDigest() string { + if x != nil { + return x.Digest + } + return "" +} + +func (x *ReferrerInfo) GetCount() int64 { + if x != nil { + return x.Count + } + return 0 +} + +func (x *ReferrerInfo) GetMediaType() string { + if x != nil { + return x.MediaType + } + return "" +} + +func (x *ReferrerInfo) GetArtifactType() string { + if x != nil { + return x.ArtifactType + } + return "" +} + +func (x *ReferrerInfo) GetSize() int64 { + if x != nil { + return x.Size + } + return 0 +} + +func (x *ReferrerInfo) GetAnnotations() map[string]string { + if x != nil { + return x.Annotations + } + return nil +} + +type ManifestSignatures struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Map map[string]*SignaturesInfo `protobuf:"bytes,1,rep,name=map,proto3" json:"map,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` +} + +func (x *ManifestSignatures) Reset() { + *x = ManifestSignatures{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[11] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ManifestSignatures) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ManifestSignatures) ProtoMessage() {} + +func (x *ManifestSignatures) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[11] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ManifestSignatures.ProtoReflect.Descriptor instead. +func (*ManifestSignatures) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{11} +} + +func (x *ManifestSignatures) GetMap() map[string]*SignaturesInfo { + if x != nil { + return x.Map + } + return nil +} + +type SignaturesInfo struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + List []*SignatureInfo `protobuf:"bytes,1,rep,name=list,proto3" json:"list,omitempty"` +} + +func (x *SignaturesInfo) Reset() { + *x = SignaturesInfo{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[12] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *SignaturesInfo) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*SignaturesInfo) ProtoMessage() {} + +func (x *SignaturesInfo) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[12] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use SignaturesInfo.ProtoReflect.Descriptor instead. +func (*SignaturesInfo) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{12} +} + +func (x *SignaturesInfo) GetList() []*SignatureInfo { + if x != nil { + return x.List + } + return nil +} + +type SignatureInfo struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + SignatureManifestDigest string `protobuf:"bytes,1,opt,name=SignatureManifestDigest,proto3" json:"SignatureManifestDigest,omitempty"` + LayersInfo []*LayersInfo `protobuf:"bytes,2,rep,name=LayersInfo,proto3" json:"LayersInfo,omitempty"` +} + +func (x *SignatureInfo) Reset() { + *x = SignatureInfo{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[13] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *SignatureInfo) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*SignatureInfo) ProtoMessage() {} + +func (x *SignatureInfo) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[13] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use SignatureInfo.ProtoReflect.Descriptor instead. +func (*SignatureInfo) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{13} +} + +func (x *SignatureInfo) GetSignatureManifestDigest() string { + if x != nil { + return x.SignatureManifestDigest + } + return "" +} + +func (x *SignatureInfo) GetLayersInfo() []*LayersInfo { + if x != nil { + return x.LayersInfo + } + return nil +} + +type LayersInfo struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + LayerDigest string `protobuf:"bytes,1,opt,name=LayerDigest,proto3" json:"LayerDigest,omitempty"` + LayerContent []byte `protobuf:"bytes,2,opt,name=LayerContent,proto3" json:"LayerContent,omitempty"` + SignatureKey string `protobuf:"bytes,3,opt,name=SignatureKey,proto3" json:"SignatureKey,omitempty"` + Signer string `protobuf:"bytes,4,opt,name=Signer,proto3" json:"Signer,omitempty"` + Date *timestamppb.Timestamp `protobuf:"bytes,5,opt,name=Date,proto3" json:"Date,omitempty"` +} + +func (x *LayersInfo) Reset() { + *x = LayersInfo{} + if protoimpl.UnsafeEnabled { + mi := &file_meta_proto_msgTypes[14] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *LayersInfo) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*LayersInfo) ProtoMessage() {} + +func (x *LayersInfo) ProtoReflect() protoreflect.Message { + mi := &file_meta_proto_msgTypes[14] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use LayersInfo.ProtoReflect.Descriptor instead. +func (*LayersInfo) Descriptor() ([]byte, []int) { + return file_meta_proto_rawDescGZIP(), []int{14} +} + +func (x *LayersInfo) GetLayerDigest() string { + if x != nil { + return x.LayerDigest + } + return "" +} + +func (x *LayersInfo) GetLayerContent() []byte { + if x != nil { + return x.LayerContent + } + return nil +} + +func (x *LayersInfo) GetSignatureKey() string { + if x != nil { + return x.SignatureKey + } + return "" +} + +func (x *LayersInfo) GetSigner() string { + if x != nil { + return x.Signer + } + return "" +} + +func (x *LayersInfo) GetDate() *timestamppb.Timestamp { + if x != nil { + return x.Date + } + return nil +} + +var File_meta_proto protoreflect.FileDescriptor + +var file_meta_proto_rawDesc = []byte{ + 0x0a, 0x0a, 0x6d, 0x65, 0x74, 0x61, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x02, 0x76, 0x31, + 0x1a, 0x0c, 0x63, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x0e, + 0x6d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x0b, + 0x69, 0x6e, 0x64, 0x65, 0x78, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x0f, 0x74, 0x69, 0x6d, + 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x10, 0x64, 0x65, + 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0x45, + 0x0a, 0x0d, 0x54, 0x61, 0x67, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, 0x6f, 0x72, 0x12, + 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x01, 0x20, 0x01, + 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x12, 0x16, 0x0a, + 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, + 0x69, 0x67, 0x65, 0x73, 0x74, 0x22, 0x8d, 0x01, 0x0a, 0x09, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x44, + 0x61, 0x74, 0x61, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, + 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, 0x70, + 0x65, 0x12, 0x2e, 0x0a, 0x09, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x73, 0x18, 0x02, + 0x20, 0x03, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x76, 0x31, 0x2e, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, + 0x73, 0x74, 0x44, 0x61, 0x74, 0x61, 0x52, 0x09, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, + 0x73, 0x12, 0x28, 0x0a, 0x05, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0b, + 0x32, 0x0d, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x44, 0x61, 0x74, 0x61, 0x48, + 0x00, 0x52, 0x05, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x88, 0x01, 0x01, 0x42, 0x08, 0x0a, 0x06, 0x5f, + 0x49, 0x6e, 0x64, 0x65, 0x78, 0x22, 0x87, 0x01, 0x0a, 0x0c, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, + 0x73, 0x74, 0x44, 0x61, 0x74, 0x61, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, + 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x12, + 0x0a, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x03, 0x52, 0x04, 0x53, 0x69, + 0x7a, 0x65, 0x12, 0x28, 0x0a, 0x08, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x18, 0x03, + 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, 0x31, 0x2e, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, + 0x73, 0x74, 0x52, 0x08, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x12, 0x21, 0x0a, 0x06, + 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x09, 0x2e, 0x76, + 0x31, 0x2e, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x52, 0x06, 0x43, 0x6f, 0x6e, 0x66, 0x69, 0x67, 0x22, + 0x58, 0x0a, 0x09, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x44, 0x61, 0x74, 0x61, 0x12, 0x16, 0x0a, 0x06, + 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, + 0x67, 0x65, 0x73, 0x74, 0x12, 0x12, 0x0a, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x18, 0x02, 0x20, 0x01, + 0x28, 0x03, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, 0x1f, 0x0a, 0x05, 0x49, 0x6e, 0x64, 0x65, + 0x78, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x09, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6e, 0x64, + 0x65, 0x78, 0x52, 0x05, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x22, 0xb1, 0x01, 0x0a, 0x14, 0x52, 0x65, + 0x70, 0x6f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, 0x61, + 0x67, 0x65, 0x12, 0x41, 0x0a, 0x0b, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, + 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, + 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, + 0x61, 0x6d, 0x70, 0x48, 0x00, 0x52, 0x0b, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, + 0x65, 0x64, 0x88, 0x01, 0x01, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, 0x79, + 0x70, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x54, + 0x79, 0x70, 0x65, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x03, 0x20, + 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x10, 0x0a, 0x03, 0x54, + 0x61, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x54, 0x61, 0x67, 0x42, 0x0e, 0x0a, + 0x0c, 0x5f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x22, 0xf4, 0x06, + 0x0a, 0x08, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x12, 0x12, 0x0a, 0x04, 0x4e, 0x61, + 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x4e, 0x61, 0x6d, 0x65, 0x12, 0x2a, + 0x0a, 0x04, 0x54, 0x61, 0x67, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x76, + 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x2e, 0x54, 0x61, 0x67, 0x73, 0x45, + 0x6e, 0x74, 0x72, 0x79, 0x52, 0x04, 0x54, 0x61, 0x67, 0x73, 0x12, 0x3c, 0x0a, 0x0a, 0x53, 0x74, + 0x61, 0x74, 0x69, 0x73, 0x74, 0x69, 0x63, 0x73, 0x18, 0x03, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1c, + 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x2e, 0x53, 0x74, 0x61, + 0x74, 0x69, 0x73, 0x74, 0x69, 0x63, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0a, 0x53, 0x74, + 0x61, 0x74, 0x69, 0x73, 0x74, 0x69, 0x63, 0x73, 0x12, 0x3c, 0x0a, 0x0a, 0x53, 0x69, 0x67, 0x6e, + 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1c, 0x2e, 0x76, + 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x2e, 0x53, 0x69, 0x67, 0x6e, 0x61, + 0x74, 0x75, 0x72, 0x65, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0a, 0x53, 0x69, 0x67, 0x6e, + 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x12, 0x39, 0x0a, 0x09, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, + 0x65, 0x72, 0x73, 0x18, 0x05, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x76, 0x31, 0x2e, 0x52, + 0x65, 0x70, 0x6f, 0x4d, 0x65, 0x74, 0x61, 0x2e, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, + 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x09, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, + 0x73, 0x12, 0x1c, 0x0a, 0x09, 0x49, 0x73, 0x53, 0x74, 0x61, 0x72, 0x72, 0x65, 0x64, 0x18, 0x06, + 0x20, 0x01, 0x28, 0x08, 0x52, 0x09, 0x49, 0x73, 0x53, 0x74, 0x61, 0x72, 0x72, 0x65, 0x64, 0x12, + 0x22, 0x0a, 0x0c, 0x49, 0x73, 0x42, 0x6f, 0x6f, 0x6b, 0x6d, 0x61, 0x72, 0x6b, 0x65, 0x64, 0x18, + 0x07, 0x20, 0x01, 0x28, 0x08, 0x52, 0x0c, 0x49, 0x73, 0x42, 0x6f, 0x6f, 0x6b, 0x6d, 0x61, 0x72, + 0x6b, 0x65, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x52, 0x61, 0x6e, 0x6b, 0x18, 0x08, 0x20, 0x01, 0x28, + 0x05, 0x52, 0x04, 0x52, 0x61, 0x6e, 0x6b, 0x12, 0x14, 0x0a, 0x05, 0x53, 0x74, 0x61, 0x72, 0x73, + 0x18, 0x09, 0x20, 0x01, 0x28, 0x05, 0x52, 0x05, 0x53, 0x74, 0x61, 0x72, 0x73, 0x12, 0x12, 0x0a, + 0x04, 0x53, 0x69, 0x7a, 0x65, 0x18, 0x0a, 0x20, 0x01, 0x28, 0x05, 0x52, 0x04, 0x53, 0x69, 0x7a, + 0x65, 0x12, 0x18, 0x0a, 0x07, 0x56, 0x65, 0x6e, 0x64, 0x6f, 0x72, 0x73, 0x18, 0x0b, 0x20, 0x03, + 0x28, 0x09, 0x52, 0x07, 0x56, 0x65, 0x6e, 0x64, 0x6f, 0x72, 0x73, 0x12, 0x2a, 0x0a, 0x09, 0x50, + 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x73, 0x18, 0x0c, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0c, + 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x52, 0x09, 0x50, 0x6c, + 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x73, 0x12, 0x49, 0x0a, 0x10, 0x4c, 0x61, 0x73, 0x74, 0x55, + 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x18, 0x0d, 0x20, 0x01, 0x28, + 0x0b, 0x32, 0x18, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x4c, 0x61, 0x73, 0x74, 0x55, + 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x48, 0x00, 0x52, 0x10, 0x4c, + 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, 0x6d, 0x61, 0x67, 0x65, 0x88, + 0x01, 0x01, 0x1a, 0x4a, 0x0a, 0x09, 0x54, 0x61, 0x67, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, + 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, + 0x79, 0x12, 0x27, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, + 0x32, 0x11, 0x2e, 0x76, 0x31, 0x2e, 0x54, 0x61, 0x67, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, + 0x74, 0x6f, 0x72, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x57, + 0x0a, 0x0f, 0x53, 0x74, 0x61, 0x74, 0x69, 0x73, 0x74, 0x69, 0x63, 0x73, 0x45, 0x6e, 0x74, 0x72, + 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, + 0x6b, 0x65, 0x79, 0x12, 0x2e, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, + 0x28, 0x0b, 0x32, 0x18, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, + 0x6f, 0x72, 0x53, 0x74, 0x61, 0x74, 0x69, 0x73, 0x74, 0x69, 0x63, 0x73, 0x52, 0x05, 0x76, 0x61, + 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x55, 0x0a, 0x0f, 0x53, 0x69, 0x67, 0x6e, 0x61, + 0x74, 0x75, 0x72, 0x65, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, + 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x2c, 0x0a, 0x05, + 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x76, 0x31, + 0x2e, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, + 0x72, 0x65, 0x73, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x1a, 0x4f, + 0x0a, 0x0e, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, + 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, + 0x65, 0x79, 0x12, 0x27, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, + 0x0b, 0x32, 0x11, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, + 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x42, + 0x13, 0x0a, 0x11, 0x5f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, 0x49, + 0x6d, 0x61, 0x67, 0x65, 0x22, 0x97, 0x01, 0x0a, 0x09, 0x52, 0x65, 0x70, 0x6f, 0x42, 0x6c, 0x6f, + 0x62, 0x73, 0x12, 0x12, 0x0a, 0x04, 0x4e, 0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, + 0x52, 0x04, 0x4e, 0x61, 0x6d, 0x65, 0x12, 0x2e, 0x0a, 0x05, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x18, + 0x02, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x18, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x42, + 0x6c, 0x6f, 0x62, 0x73, 0x2e, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, + 0x05, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x1a, 0x46, 0x0a, 0x0a, 0x42, 0x6c, 0x6f, 0x62, 0x73, 0x45, + 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, + 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x22, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, + 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, 0x31, 0x2e, 0x42, 0x6c, 0x6f, 0x62, 0x49, + 0x6e, 0x66, 0x6f, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0xd3, + 0x01, 0x0a, 0x08, 0x42, 0x6c, 0x6f, 0x62, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x12, 0x0a, 0x04, 0x53, + 0x69, 0x7a, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x03, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, + 0x18, 0x0a, 0x07, 0x56, 0x65, 0x6e, 0x64, 0x6f, 0x72, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x09, + 0x52, 0x07, 0x56, 0x65, 0x6e, 0x64, 0x6f, 0x72, 0x73, 0x12, 0x2a, 0x0a, 0x09, 0x50, 0x6c, 0x61, + 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x73, 0x18, 0x03, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0c, 0x2e, 0x76, + 0x31, 0x2e, 0x50, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x52, 0x09, 0x50, 0x6c, 0x61, 0x74, + 0x66, 0x6f, 0x72, 0x6d, 0x73, 0x12, 0x1a, 0x0a, 0x08, 0x53, 0x75, 0x62, 0x42, 0x6c, 0x6f, 0x62, + 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x09, 0x52, 0x08, 0x53, 0x75, 0x62, 0x42, 0x6c, 0x6f, 0x62, + 0x73, 0x12, 0x41, 0x0a, 0x0b, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x64, + 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, + 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, + 0x6d, 0x70, 0x48, 0x00, 0x52, 0x0b, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, + 0x64, 0x88, 0x01, 0x01, 0x42, 0x0e, 0x0a, 0x0c, 0x5f, 0x4c, 0x61, 0x73, 0x74, 0x55, 0x70, 0x64, + 0x61, 0x74, 0x65, 0x64, 0x22, 0x3c, 0x0a, 0x14, 0x44, 0x65, 0x73, 0x63, 0x72, 0x69, 0x70, 0x74, + 0x6f, 0x72, 0x53, 0x74, 0x61, 0x74, 0x69, 0x73, 0x74, 0x69, 0x63, 0x73, 0x12, 0x24, 0x0a, 0x0d, + 0x44, 0x6f, 0x77, 0x6e, 0x6c, 0x6f, 0x61, 0x64, 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x18, 0x01, 0x20, + 0x01, 0x28, 0x05, 0x52, 0x0d, 0x44, 0x6f, 0x77, 0x6e, 0x6c, 0x6f, 0x61, 0x64, 0x43, 0x6f, 0x75, + 0x6e, 0x74, 0x22, 0x35, 0x0a, 0x0d, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x73, 0x49, + 0x6e, 0x66, 0x6f, 0x12, 0x24, 0x0a, 0x04, 0x6c, 0x69, 0x73, 0x74, 0x18, 0x01, 0x20, 0x03, 0x28, + 0x0b, 0x32, 0x10, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x49, + 0x6e, 0x66, 0x6f, 0x52, 0x04, 0x6c, 0x69, 0x73, 0x74, 0x22, 0x97, 0x02, 0x0a, 0x0c, 0x52, 0x65, + 0x66, 0x65, 0x72, 0x72, 0x65, 0x72, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x16, 0x0a, 0x06, 0x44, 0x69, + 0x67, 0x65, 0x73, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x44, 0x69, 0x67, 0x65, + 0x73, 0x74, 0x12, 0x14, 0x0a, 0x05, 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x18, 0x02, 0x20, 0x01, 0x28, + 0x03, 0x52, 0x05, 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x12, 0x1c, 0x0a, 0x09, 0x4d, 0x65, 0x64, 0x69, + 0x61, 0x54, 0x79, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x4d, 0x65, 0x64, + 0x69, 0x61, 0x54, 0x79, 0x70, 0x65, 0x12, 0x22, 0x0a, 0x0c, 0x41, 0x72, 0x74, 0x69, 0x66, 0x61, + 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x41, 0x72, + 0x74, 0x69, 0x66, 0x61, 0x63, 0x74, 0x54, 0x79, 0x70, 0x65, 0x12, 0x12, 0x0a, 0x04, 0x53, 0x69, + 0x7a, 0x65, 0x18, 0x05, 0x20, 0x01, 0x28, 0x03, 0x52, 0x04, 0x53, 0x69, 0x7a, 0x65, 0x12, 0x43, + 0x0a, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x06, 0x20, + 0x03, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x66, 0x65, 0x72, 0x72, 0x65, + 0x72, 0x49, 0x6e, 0x66, 0x6f, 0x2e, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, 0x6e, + 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x0b, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, + 0x6f, 0x6e, 0x73, 0x1a, 0x3e, 0x0a, 0x10, 0x41, 0x6e, 0x6e, 0x6f, 0x74, 0x61, 0x74, 0x69, 0x6f, + 0x6e, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, 0x61, 0x6c, + 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, + 0x02, 0x38, 0x01, 0x22, 0x93, 0x01, 0x0a, 0x12, 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, + 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x12, 0x31, 0x0a, 0x03, 0x6d, 0x61, + 0x70, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1f, 0x2e, 0x76, 0x31, 0x2e, 0x4d, 0x61, 0x6e, + 0x69, 0x66, 0x65, 0x73, 0x74, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x2e, + 0x4d, 0x61, 0x70, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x03, 0x6d, 0x61, 0x70, 0x1a, 0x4a, 0x0a, + 0x08, 0x4d, 0x61, 0x70, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, + 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x28, 0x0a, 0x05, 0x76, + 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x12, 0x2e, 0x76, 0x31, 0x2e, + 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x05, + 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0x37, 0x0a, 0x0e, 0x53, 0x69, 0x67, + 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x25, 0x0a, 0x04, 0x6c, + 0x69, 0x73, 0x74, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x11, 0x2e, 0x76, 0x31, 0x2e, 0x53, + 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x04, 0x6c, 0x69, + 0x73, 0x74, 0x22, 0x79, 0x0a, 0x0d, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x49, + 0x6e, 0x66, 0x6f, 0x12, 0x38, 0x0a, 0x17, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, + 0x4d, 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x09, 0x52, 0x17, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x4d, + 0x61, 0x6e, 0x69, 0x66, 0x65, 0x73, 0x74, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x2e, 0x0a, + 0x0a, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x18, 0x02, 0x20, 0x03, 0x28, + 0x0b, 0x32, 0x0e, 0x2e, 0x76, 0x31, 0x2e, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, + 0x6f, 0x52, 0x0a, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x22, 0xbe, 0x01, + 0x0a, 0x0a, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x20, 0x0a, 0x0b, + 0x4c, 0x61, 0x79, 0x65, 0x72, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, + 0x09, 0x52, 0x0b, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x44, 0x69, 0x67, 0x65, 0x73, 0x74, 0x12, 0x22, + 0x0a, 0x0c, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x43, 0x6f, 0x6e, 0x74, 0x65, 0x6e, 0x74, 0x18, 0x02, + 0x20, 0x01, 0x28, 0x0c, 0x52, 0x0c, 0x4c, 0x61, 0x79, 0x65, 0x72, 0x43, 0x6f, 0x6e, 0x74, 0x65, + 0x6e, 0x74, 0x12, 0x22, 0x0a, 0x0c, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, 0x75, 0x72, 0x65, 0x4b, + 0x65, 0x79, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x74, + 0x75, 0x72, 0x65, 0x4b, 0x65, 0x79, 0x12, 0x16, 0x0a, 0x06, 0x53, 0x69, 0x67, 0x6e, 0x65, 0x72, + 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x53, 0x69, 0x67, 0x6e, 0x65, 0x72, 0x12, 0x2e, + 0x0a, 0x04, 0x44, 0x61, 0x74, 0x65, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, + 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, + 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, 0x04, 0x44, 0x61, 0x74, 0x65, 0x62, 0x06, + 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, +} + +var ( + file_meta_proto_rawDescOnce sync.Once + file_meta_proto_rawDescData = file_meta_proto_rawDesc +) + +func file_meta_proto_rawDescGZIP() []byte { + file_meta_proto_rawDescOnce.Do(func() { + file_meta_proto_rawDescData = protoimpl.X.CompressGZIP(file_meta_proto_rawDescData) + }) + return file_meta_proto_rawDescData +} + +var file_meta_proto_msgTypes = make([]protoimpl.MessageInfo, 22) +var file_meta_proto_goTypes = []interface{}{ + (*TagDescriptor)(nil), // 0: v1.TagDescriptor + (*ImageData)(nil), // 1: v1.ImageData + (*ManifestData)(nil), // 2: v1.ManifestData + (*IndexData)(nil), // 3: v1.IndexData + (*RepoLastUpdatedImage)(nil), // 4: v1.RepoLastUpdatedImage + (*RepoMeta)(nil), // 5: v1.RepoMeta + (*RepoBlobs)(nil), // 6: v1.RepoBlobs + (*BlobInfo)(nil), // 7: v1.BlobInfo + (*DescriptorStatistics)(nil), // 8: v1.DescriptorStatistics + (*ReferrersInfo)(nil), // 9: v1.ReferrersInfo + (*ReferrerInfo)(nil), // 10: v1.ReferrerInfo + (*ManifestSignatures)(nil), // 11: v1.ManifestSignatures + (*SignaturesInfo)(nil), // 12: v1.SignaturesInfo + (*SignatureInfo)(nil), // 13: v1.SignatureInfo + (*LayersInfo)(nil), // 14: v1.LayersInfo + nil, // 15: v1.RepoMeta.TagsEntry + nil, // 16: v1.RepoMeta.StatisticsEntry + nil, // 17: v1.RepoMeta.SignaturesEntry + nil, // 18: v1.RepoMeta.ReferrersEntry + nil, // 19: v1.RepoBlobs.BlobsEntry + nil, // 20: v1.ReferrerInfo.AnnotationsEntry + nil, // 21: v1.ManifestSignatures.MapEntry + (*Manifest)(nil), // 22: v1.Manifest + (*Image)(nil), // 23: v1.Image + (*Index)(nil), // 24: v1.Index + (*timestamppb.Timestamp)(nil), // 25: google.protobuf.Timestamp + (*Platform)(nil), // 26: v1.Platform +} +var file_meta_proto_depIdxs = []int32{ + 2, // 0: v1.ImageData.Manifests:type_name -> v1.ManifestData + 3, // 1: v1.ImageData.Index:type_name -> v1.IndexData + 22, // 2: v1.ManifestData.Manifest:type_name -> v1.Manifest + 23, // 3: v1.ManifestData.Config:type_name -> v1.Image + 24, // 4: v1.IndexData.Index:type_name -> v1.Index + 25, // 5: v1.RepoLastUpdatedImage.LastUpdated:type_name -> google.protobuf.Timestamp + 15, // 6: v1.RepoMeta.Tags:type_name -> v1.RepoMeta.TagsEntry + 16, // 7: v1.RepoMeta.Statistics:type_name -> v1.RepoMeta.StatisticsEntry + 17, // 8: v1.RepoMeta.Signatures:type_name -> v1.RepoMeta.SignaturesEntry + 18, // 9: v1.RepoMeta.Referrers:type_name -> v1.RepoMeta.ReferrersEntry + 26, // 10: v1.RepoMeta.Platforms:type_name -> v1.Platform + 4, // 11: v1.RepoMeta.LastUpdatedImage:type_name -> v1.RepoLastUpdatedImage + 19, // 12: v1.RepoBlobs.Blobs:type_name -> v1.RepoBlobs.BlobsEntry + 26, // 13: v1.BlobInfo.Platforms:type_name -> v1.Platform + 25, // 14: v1.BlobInfo.LastUpdated:type_name -> google.protobuf.Timestamp + 10, // 15: v1.ReferrersInfo.list:type_name -> v1.ReferrerInfo + 20, // 16: v1.ReferrerInfo.Annotations:type_name -> v1.ReferrerInfo.AnnotationsEntry + 21, // 17: v1.ManifestSignatures.map:type_name -> v1.ManifestSignatures.MapEntry + 13, // 18: v1.SignaturesInfo.list:type_name -> v1.SignatureInfo + 14, // 19: v1.SignatureInfo.LayersInfo:type_name -> v1.LayersInfo + 25, // 20: v1.LayersInfo.Date:type_name -> google.protobuf.Timestamp + 0, // 21: v1.RepoMeta.TagsEntry.value:type_name -> v1.TagDescriptor + 8, // 22: v1.RepoMeta.StatisticsEntry.value:type_name -> v1.DescriptorStatistics + 11, // 23: v1.RepoMeta.SignaturesEntry.value:type_name -> v1.ManifestSignatures + 9, // 24: v1.RepoMeta.ReferrersEntry.value:type_name -> v1.ReferrersInfo + 7, // 25: v1.RepoBlobs.BlobsEntry.value:type_name -> v1.BlobInfo + 12, // 26: v1.ManifestSignatures.MapEntry.value:type_name -> v1.SignaturesInfo + 27, // [27:27] is the sub-list for method output_type + 27, // [27:27] is the sub-list for method input_type + 27, // [27:27] is the sub-list for extension type_name + 27, // [27:27] is the sub-list for extension extendee + 0, // [0:27] is the sub-list for field type_name +} + +func init() { file_meta_proto_init() } +func file_meta_proto_init() { + if File_meta_proto != nil { + return + } + file_config_proto_init() + file_manifest_proto_init() + file_index_proto_init() + file_descriptor_proto_init() + if !protoimpl.UnsafeEnabled { + file_meta_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*TagDescriptor); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*ImageData); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*ManifestData); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*IndexData); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[4].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*RepoLastUpdatedImage); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[5].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*RepoMeta); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[6].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*RepoBlobs); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[7].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*BlobInfo); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[8].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*DescriptorStatistics); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[9].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*ReferrersInfo); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[10].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*ReferrerInfo); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[11].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*ManifestSignatures); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[12].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*SignaturesInfo); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[13].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*SignatureInfo); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_meta_proto_msgTypes[14].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*LayersInfo); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + file_meta_proto_msgTypes[1].OneofWrappers = []interface{}{} + file_meta_proto_msgTypes[4].OneofWrappers = []interface{}{} + file_meta_proto_msgTypes[5].OneofWrappers = []interface{}{} + file_meta_proto_msgTypes[7].OneofWrappers = []interface{}{} + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_meta_proto_rawDesc, + NumEnums: 0, + NumMessages: 22, + NumExtensions: 0, + NumServices: 0, + }, + GoTypes: file_meta_proto_goTypes, + DependencyIndexes: file_meta_proto_depIdxs, + MessageInfos: file_meta_proto_msgTypes, + }.Build() + File_meta_proto = out.File + file_meta_proto_rawDesc = nil + file_meta_proto_goTypes = nil + file_meta_proto_depIdxs = nil +} diff --git a/pkg/meta/proto_go/versioned.pb.go b/pkg/meta/proto_go/versioned.pb.go index b865ec2fa3..e7adcd681d 100644 --- a/pkg/meta/proto_go/versioned.pb.go +++ b/pkg/meta/proto_go/versioned.pb.go @@ -25,7 +25,7 @@ type Versioned struct { sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - Schemaversion int32 `protobuf:"varint,1,opt,name=schemaversion,proto3" json:"schemaversion,omitempty"` + SchemaVersion int32 `protobuf:"varint,1,opt,name=SchemaVersion,proto3" json:"SchemaVersion,omitempty"` } func (x *Versioned) Reset() { @@ -60,9 +60,9 @@ func (*Versioned) Descriptor() ([]byte, []int) { return file_versioned_proto_rawDescGZIP(), []int{0} } -func (x *Versioned) GetSchemaversion() int32 { +func (x *Versioned) GetSchemaVersion() int32 { if x != nil { - return x.Schemaversion + return x.SchemaVersion } return 0 } @@ -72,9 +72,9 @@ var File_versioned_proto protoreflect.FileDescriptor var file_versioned_proto_rawDesc = []byte{ 0x0a, 0x0f, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x65, 0x64, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x02, 0x76, 0x31, 0x22, 0x31, 0x0a, 0x09, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, - 0x65, 0x64, 0x12, 0x24, 0x0a, 0x0d, 0x73, 0x63, 0x68, 0x65, 0x6d, 0x61, 0x76, 0x65, 0x72, 0x73, - 0x69, 0x6f, 0x6e, 0x18, 0x01, 0x20, 0x01, 0x28, 0x05, 0x52, 0x0d, 0x73, 0x63, 0x68, 0x65, 0x6d, - 0x61, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, + 0x65, 0x64, 0x12, 0x24, 0x0a, 0x0d, 0x53, 0x63, 0x68, 0x65, 0x6d, 0x61, 0x56, 0x65, 0x72, 0x73, + 0x69, 0x6f, 0x6e, 0x18, 0x01, 0x20, 0x01, 0x28, 0x05, 0x52, 0x0d, 0x53, 0x63, 0x68, 0x65, 0x6d, + 0x61, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, } var ( diff --git a/pkg/meta/proto_test.go b/pkg/meta/proto_test.go deleted file mode 100644 index f5cf9e22d7..0000000000 --- a/pkg/meta/proto_test.go +++ /dev/null @@ -1,208 +0,0 @@ -package meta_test - -import ( - "encoding/json" - "fmt" - "net/url" - "sync" - "testing" - - ispec "github.com/opencontainers/image-spec/specs-go/v1" - . "github.com/smartystreets/goconvey/convey" - "google.golang.org/protobuf/proto" - "gopkg.in/resty.v1" - - "zotregistry.io/zot/pkg/api" - "zotregistry.io/zot/pkg/api/config" - "zotregistry.io/zot/pkg/api/constants" - zcommon "zotregistry.io/zot/pkg/common" - extconf "zotregistry.io/zot/pkg/extensions/config" - "zotregistry.io/zot/pkg/log" - "zotregistry.io/zot/pkg/meta/proto_go" - . "zotregistry.io/zot/pkg/test" - . "zotregistry.io/zot/pkg/test/image-utils" -) - -func TestProto(t *testing.T) { - Convey("Basic conversion", t, func() { - Convey("Manifest", func() { - mediaType := ispec.MediaTypeImageManifest - manifest := &proto_go.Manifest{ - Versioned: &proto_go.Versioned{Schemaversion: 2}, - Mediatype: &mediaType, - } - - buf, err := proto.Marshal(manifest) - So(err, ShouldBeNil) - - manifest = &proto_go.Manifest{} - - err = proto.Unmarshal(buf, manifest) - So(err, ShouldBeNil) - So(manifest.GetMediatype(), ShouldEqual, ispec.MediaTypeImageManifest) - }) - - Convey("Index", func() { - mediaType := ispec.MediaTypeImageIndex - index := &proto_go.Index{ - Versioned: &proto_go.Versioned{Schemaversion: 2}, - Mediatype: &mediaType, - } - - out, err := proto.Marshal(index) - So(err, ShouldBeNil) - - index = &proto_go.Index{} - - err = proto.Unmarshal(out, index) - So(err, ShouldBeNil) - So(index.GetMediatype(), ShouldEqual, ispec.MediaTypeImageIndex) - }) - }) -} - -func TestProtoParsing(t *testing.T) { - Convey("proto parse", t, func() { - port := GetFreePort() - baseURL := GetBaseURL(port) - conf := config.New() - conf.HTTP.Port = port - rootDir := t.TempDir() - - conf.Storage.RootDirectory = rootDir - defaultVal := true - conf.Extensions = &extconf.ExtensionConfig{ - Search: &extconf.SearchConfig{ - BaseConfig: extconf.BaseConfig{Enable: &defaultVal}, - }, - } - - storeController := GetDefaultStoreController(rootDir, log.NewLogger("debug", "")) - - err := WriteImageToFileSystem(CreateDefaultImage(), "repo", "tag1", storeController) - So(err, ShouldBeNil) - - err = WriteImageToFileSystem(CreateDefaultImage(), "repo", "tag2", storeController) - So(err, ShouldBeNil) - - img1 := CreateRandomImageWith(). - Annotations(map[string]string{ - "test": "annotation", - }).Build() - - imgRef := CreateRandomImageWith(). - Subject(img1.DescriptorRef()).Build() - - err = WriteImageToFileSystem(img1, "repo", "tag-img", storeController) - So(err, ShouldBeNil) - err = WriteImageToFileSystem(imgRef, "repo", "tag-ref", storeController) - So(err, ShouldBeNil) - - err = WriteImageToFileSystem(CreateDefaultImage(), "repo", "tag2", storeController) - So(err, ShouldBeNil) - - err = WriteMultiArchImageToFileSystem(CreateRandomMultiarch(), "repo-multi", "multi-arch", storeController) - So(err, ShouldBeNil) - - ctlr := api.NewController(conf) - ctrlManager := NewControllerManager(ctlr) - ctrlManager.StartAndWait(port) - defer ctrlManager.StopServer() - - query := ` - { - GlobalSearch(query:"repo:"){ - Repos { - Name LastUpdated Size - Platforms { Os Arch } - Vendors - NewestImage { - RepoName Tag LastUpdated Size IsSigned - Authors - } - } - } - }` - - resp, err := resty.R().Get(baseURL + constants.FullSearchPrefix + "?query=" + url.QueryEscape(query)) - So(resp, ShouldNotBeNil) - So(err, ShouldBeNil) - So(resp.StatusCode(), ShouldEqual, 200) - - responseStructRepos := &zcommon.GlobalSearchResultResp{} - - err = json.Unmarshal(resp.Body(), responseStructRepos) - So(err, ShouldBeNil) - - So(responseStructRepos.Repos, ShouldNotBeEmpty) - }) -} - -func TestGenerate(t *testing.T) { - repoCount := 2000 - manifestImageCount := 10 - multiarchCount := 5 - - storeController := GetDefaultStoreController("/home/laur/dev/images/stress-test/storage", log.NewLogger("debug", "/dev/null")) - - for repoId := 0; repoId < repoCount; repoId++ { - for i := 0; i < manifestImageCount; i++ { - err := WriteImageToFileSystem( - CreateImageWith(). - RandomLayers(2, 5). - RandomConfig(). - Annotations(map[string]string{"test": "annotation"}).Build(), - fmt.Sprintf("repo%d", repoId), - fmt.Sprintf("tag%d", i), - storeController, - ) - if err != nil { - t.FailNow() - } - - t.Logf("Repo: %d, Manifest Tag: %d", repoId, i) - } - - for i := 0; i < multiarchCount; i++ { - err := WriteMultiArchImageToFileSystem( - CreateRandomMultiarch(), - fmt.Sprintf("repo%d", repoId), - fmt.Sprintf("tag-multi%d", i), - storeController, - ) - if err != nil { - t.FailNow() - } - - t.Logf("Repo: %d, Index Tag: %d", repoId, i) - } - } -} - -const ( - homeQuery1 = `{GlobalSearch(query:"", requestedPage: {limit:3 offset:0 sortBy: DOWNLOADS} ) {Page {TotalCount ItemCount} Repos {Name LastUpdated Size Platforms { Os Arch } IsStarred IsBookmarked NewestImage { Tag Vulnerabilities {MaxSeverity Count} Description IsSigned SignatureInfo { Tool IsTrusted Author } Licenses Vendor Labels } DownloadCount}}}` - homeQuery2 = `{GlobalSearch(query:"", requestedPage: {limit:2 offset:0 sortBy: UPDATE_TIME} ) {Page {TotalCount ItemCount} Repos {Name LastUpdated Size Platforms { Os Arch } IsStarred IsBookmarked NewestImage { Tag Vulnerabilities {MaxSeverity Count} Description IsSigned SignatureInfo { Tool IsTrusted Author } Licenses Vendor Labels } DownloadCount}}}` - homeQuery3 = `{GlobalSearch(query:"", requestedPage: {limit:2 offset:0 sortBy: RELEVANCE} ,filter: { IsBookmarked: true}) {Page {TotalCount ItemCount} Repos {Name LastUpdated Size Platforms { Os Arch } IsStarred IsBookmarked NewestImage { Tag Vulnerabilities {MaxSeverity Count} Description IsSigned SignatureInfo { Tool IsTrusted Author } Licenses Vendor Labels } DownloadCount}}}` -) - -func TestProfileHomePage(t *testing.T) { - for i := 0; i < 10; i++ { - wg := &sync.WaitGroup{} - - go RunQuery(homeQuery1, wg, t) - go RunQuery(homeQuery2, wg, t) - go RunQuery(homeQuery3, wg, t) - - wg.Wait() - t.Logf("Execution: %d", i) - } -} - -func RunQuery(query string, wg *sync.WaitGroup, t *testing.T) { - resp, err := resty.R().Get("http://127.0.0.1:5000" + constants.FullSearchPrefix + "?query=" + url.QueryEscape(query)) - if err != nil || resp.StatusCode() != 200 { - panic(fmt.Errorf("StatusCode: %d Err: %w", resp.StatusCode(), err)) - } - - t.Log("\t*") -} diff --git a/pkg/meta/types/types.go b/pkg/meta/types/types.go index b26c165ccd..ad5e0bd3a5 100644 --- a/pkg/meta/types/types.go +++ b/pkg/meta/types/types.go @@ -8,16 +8,6 @@ import ( ispec "github.com/opencontainers/image-spec/specs-go/v1" ) -// DetailedRepoMeta is a auxiliary structure used for sorting RepoMeta arrays by information -// that's not directly available in the RepoMetadata structure (ex. that needs to be calculated -// by iterating the manifests, etc.) -type DetailedRepoMeta struct { - RepoMetadata - Rank int - Downloads int - UpdateTime time.Time -} - // Used to model changes to an object after a call to the DB. type ToggleState int @@ -28,20 +18,19 @@ const ( ) type ( - FilterFunc func(repoMeta RepoMetadata, manifestMeta ManifestMetadata) bool - FilterProtoFunc func(repoMeta RepoMetadata2, imageData ImageData2) bool - FilterRepoFunc func(repoMeta RepoMetadata) bool - FilterProtoRepoFunc func(repoMeta RepoMetadata2) bool - FilterRepoNameFunc func(repo string) int - FilterFullRepoFunc func(repoMeta FullRepoMetadata) bool - FilterRepoTagFunc func(repo, tag string) bool + // Currently imageData applied for indexes is applied for each manifest individually so imageData.manifests + // contains just 1 manifest. + FilterFunc func(repoMeta RepoMeta, imageData ImageData) bool + FilterRepoNameFunc func(repo string) bool + FilterFullRepoFunc func(repoMeta RepoMeta) bool + FilterRepoTagFunc func(repo, tag string) bool ) -func AcceptAllRepoNames(repo string) int { - return 1 +func AcceptAllRepoNames(repo string) bool { + return true } -func AcceptAllRepoMeta(repometa FullRepoMetadata) bool { +func AcceptAllRepoMeta(repoMeta RepoMeta) bool { return true } @@ -49,165 +38,112 @@ func AcceptAllRepoTag(repo, tag string) bool { return true } -func AcceptAllImageData(repoMeta RepoMetadata2, imageData ImageData2) bool { +func AcceptAllImageData(repoMeta RepoMeta, imageData ImageData) bool { return true } +func GetLatestImageDigests(repoMetaList []RepoMeta) []string { + digests := make([]string, 0, len(repoMetaList)) + + for i := range repoMetaList { + if repoMetaList[i].LastUpdatedImage != nil { + digests = append(digests, repoMetaList[i].LastUpdatedImage.Digest) + } + } + + return digests +} + type ( ImageDigest = string ) type MetaDB interface { //nolint:interfacebloat UserDB - // IncrementRepoStars adds 1 to the star count of an image - IncrementRepoStars(repo string) error - // IncrementRepoStars subtracts 1 from the star count of an image - DecrementRepoStars(repo string) error + SetImageData(digest godigest.Digest, imageData ImageData) error - // GetRepoStars returns the total number of stars a repo has - GetRepoStars(repo string) (int, error) + // SetRepoReference sets the given image data to the repo metadata. + SetRepoReference(repo string, reference string, imageData ImageData) error - // SetRepoReference sets the reference of a manifest in the tag list of a repo - SetRepoReference(repo string, reference string, manifestDigest godigest.Digest, mediaType string) error + // SearchRepos searches for repos given a search string + SearchRepos(ctx context.Context, searchText string) ([]RepoMeta, error) - /* - RemoveRepoReference removes the tag from RepoMetadata if the reference is a tag, + // SearchTags searches for images(repo:tag) given a search string + SearchTags(ctx context.Context, searchText string) ([]ImageMeta, error) - it also removes its corresponding digest from Statistics, Signatures and Referrers if there are no tags - pointing to it. - If the reference is a digest then it will remove the digest from Statistics, Signatures and Referrers only - if there are no tags pointing to the digest, otherwise it's noop - */ - RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest) error + // FilterTags filters for images given a filter function + FilterTags(ctx context.Context, filterRepoTag FilterRepoTagFunc, filterFunc FilterFunc, + ) ([]ImageMeta, error) - // DeleteRepoTag delets the tag from the tag list of a repo - DeleteRepoTag(repo string, tag string) error + // FilterRepos filters for repos given a filter function + FilterRepos(ctx context.Context, rankName FilterRepoNameFunc, filterFunc FilterFullRepoFunc, + ) ([]RepoMeta, error) - // GetRepoMeta returns RepoMetadata of a repo from the database - GetRepoMeta(repo string) (RepoMetadata, error) + // GetRepoMeta return the full information about a repo + GetRepoMeta(ctx context.Context, repo string) (RepoMeta, error) - // GetUserRepometa return RepoMetadata of a repo from the database along side specific information about the - // user - GetUserRepoMeta(ctx context.Context, repo string) (RepoMetadata, error) + // GetImageMeta return the full information about an image + GetImageMeta(ctx context.Context, repo string, tag string) (ImageMeta, error) - // GetRepoMeta returns RepoMetadata of a repo from the database - SetRepoMeta(repo string, repoMeta RepoMetadata) error + // GetImageData returns the raw information about an image + GetImageData(digest godigest.Digest) (ImageData, error) // GetMultipleRepoMeta returns information about all repositories as map[string]RepoMetadata filtered by the filter // function - GetMultipleRepoMeta(ctx context.Context, filter func(repoMeta RepoMetadata) bool) ( - []RepoMetadata, error) - - // SetManifestData sets ManifestData for a given manifest in the database - SetManifestData(manifestDigest godigest.Digest, md ManifestData) error + GetMultipleRepoMeta(ctx context.Context, filter func(repoMeta RepoMeta) bool) ( + []RepoMeta, error) - // GetManifestData return the manifest and its related config - GetManifestData(manifestDigest godigest.Digest) (ManifestData, error) - - // GetManifestMeta returns ManifestMetadata for a given manifest from the database - GetManifestMeta(repo string, manifestDigest godigest.Digest) (ManifestMetadata, error) + // AddManifestSignature adds signature metadata to a given manifest in the database + AddManifestSignature(repo string, signedManifestDigest godigest.Digest, sm SignatureMetadata) error - // GetManifestMeta sets ManifestMetadata for a given manifest in the database - SetManifestMeta(repo string, manifestDigest godigest.Digest, mm ManifestMetadata) error + // DeleteSignature deletes signature metadata to a given manifest from the database + DeleteSignature(repo string, signedManifestDigest godigest.Digest, sigMeta SignatureMetadata) error - // SetIndexData sets indexData for a given index in the database - SetIndexData(digest godigest.Digest, indexData IndexData) error + // UpdateSignaturesValidity checks and updates signatures validity of a given manifest + UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error - // GetIndexData returns indexData for a given Index from the database - GetIndexData(indexDigest godigest.Digest) (IndexData, error) + // IncrementRepoStars adds 1 to the star count of an image + IncrementRepoStars(repo string) error - // SetReferrer adds a referrer to the referrers list of a manifest inside a repo - SetReferrer(repo string, referredDigest godigest.Digest, referrer ReferrerInfo) error + // DecrementRepoStars subtracts 1 from the star count of an image + DecrementRepoStars(repo string) error - // SetReferrer delets a referrer to the referrers list of a manifest inside a repo - DeleteReferrer(repo string, referredDigest godigest.Digest, referrerDigest godigest.Digest) error + // SetRepoMeta returns RepoMetadata of a repo from the database + SetRepoMeta(repo string, repoMeta RepoMeta) error - // GetReferrersInfo returnes a list of for all referrers of the given digest that match one of the + // GetReferrersInfo returns a list of for all referrers of the given digest that match one of the // artifact types. - GetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string) ( - []ReferrerInfo, error) + GetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string) ([]ReferrerInfo, error) - // IncrementManifestDownloads adds 1 to the download count of a manifest + // IncrementImageDownloads adds 1 to the download count of an image IncrementImageDownloads(repo string, reference string) error - // AddManifestSignature adds signature metadata to a given manifest in the database - AddManifestSignature(repo string, signedManifestDigest godigest.Digest, sm SignatureMetadata) error - - // DeleteSignature delets signature metadata to a given manifest from the database - DeleteSignature(repo string, signedManifestDigest godigest.Digest, sm SignatureMetadata) error - - // UpdateSignaturesValidity checks and updates signatures validity of a given manifest - UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error + // FilterImageData returns the image data for the given digests + FilterImageData(ctx context.Context, digests []string) (map[string]ImageData, error) - // SearchRepos searches for repos given a search string - SearchRepos(ctx context.Context, searchText string) ( - []RepoMetadata, map[string]ManifestMetadata, map[string]IndexData, error) + /* + RemoveRepoReference removes the tag from RepoMetadata if the reference is a tag, - // SearchTags searches for images(repo:tag) given a search string - SearchTags(ctx context.Context, searchText string) ( - []RepoMetadata, map[string]ManifestMetadata, map[string]IndexData, error) + it also removes its corresponding digest from Statistics, Signatures and Referrers if there are no tags + pointing to it. + If the reference is a digest then it will remove the digest from Statistics, Signatures and Referrers only + if there are no tags pointing to the digest, otherwise it's noop + */ + RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest) error - // FilterRepos filters for repos given a filter function - FilterRepos(ctx context.Context, filter FilterRepoFunc) ( - []RepoMetadata, map[string]ManifestMetadata, map[string]IndexData, error) + // ResetRepoReferences resets all layout specific data (tags, signatures, referrers, etc.) but keep user specific + // data such as star count and other statistics + ResetRepoReferences(repo string) error - // FilterTags filters for images given a filter function - FilterTags(ctx context.Context, filterFunc FilterFunc) ( - []RepoMetadata, map[string]ManifestMetadata, map[string]IndexData, error) + // ResetDB will delete all data in the DB + ResetDB() error PatchDB() error ImageTrustStore() ImageTrustStore SetImageTrustStore(imgTrustStore ImageTrustStore) - - // -------------------------------------------------------------------------- - - ProtoSetImageData(digest godigest.Digest, imageData ImageData2) error - - ProtoSetRepoReference(repo string, reference string, imageData ImageData2) error - - ProtoSearchRepos(ctx context.Context, searchText string) ([]FullRepoMetadata, error) - - ProtoSearchTags(ctx context.Context, searchText string) ([]FullImageData, error) - - ProtoFilterTags(ctx context.Context, filterRepoTag FilterRepoTagFunc, filterFunc FilterProtoFunc) ([]FullImageData, error) - - ProtoFilterRepos(ctx context.Context, rankName FilterRepoNameFunc, filterFunc FilterFullRepoFunc) ([]FullRepoMetadata, error) - - ProtoGetRepoMeta(repo string) (RepoMetadata2, error) - - ProtoGetFullRepoMeta(ctx context.Context, repo string) (FullRepoMetadata, error) - - ProtoGetImageData(digest godigest.Digest) (ImageData2, error) - - ProtoGetMultipleRepoMeta(ctx context.Context, filter func(repoMeta RepoMetadata2) bool) ( - []RepoMetadata2, error) - - ProtoAddManifestSignature(repo string, signedManifestDigest godigest.Digest, sm SignatureMetadata) error - - ProtoDeleteSignature(repo string, signedManifestDigest godigest.Digest, sigMeta SignatureMetadata) error - - ProtoIncrementRepoStars(repo string) error - - ProtoDecrementRepoStars(repo string) error - - ProtoGetRepoStars(repo string) (int, error) - - ProtoDeleteRepoTag(repo string, tag string) error - - ProtoGetUserRepoMeta(ctx context.Context, repo string) (RepoMetadata2, error) - - ProtoSetRepoMeta(repo string, repoMeta RepoMetadata2) error - - ProtoGetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string) ([]ReferrerInfo, error) - - ProtoIncrementImageDownloads(repo string, reference string) error - - ProtoUpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error - - ProtoFilterImageData(ctx context.Context, digests []string) (map[string]ImageData2, error) } type UserDB interface { //nolint:interfacebloat @@ -249,32 +185,29 @@ type UserDB interface { //nolint:interfacebloat type ImageTrustStore interface { VerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, blob []byte, - repo string, - ) (string, time.Time, bool, error) - - ProtoVerifySignature( - signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, imageData ImageData2, + signatureType string, rawSignature []byte, sigKey string, manifestDigest godigest.Digest, imageData ImageData, repo string, ) (string, time.Time, bool, error) } -type ImageData2 struct { - MediaType string - Digest godigest.Digest - Size int64 - Index *ispec.Index - Manifests []ManifestData2 +// ImageData can store all data related to a image, multiarch or simple. Used for writing imaged to MetaDB. +type ImageData struct { + MediaType string // MediaType refers to the image descriptor, a manifest or a index (if multiarch) + Digest godigest.Digest // Digest refers to the image descriptor, a manifest or a index (if multiarch) + Size int64 // Size refers to the image descriptor, a manifest or a index (if multiarch) + Index *ispec.Index // If the image is multiarch the Index will be non-nil + Manifests []ManifestData // All manifests under the image, 1 for simple images and many for multiarch } -type ManifestData2 struct { - Size int64 - Digest godigest.Digest - ispec.Manifest - ConfigContent ispec.Image +// ManifestData represents all data related to an image manifests (found from the image contents itself). +type ManifestData struct { + Size int64 + Digest godigest.Digest + Manifest ispec.Manifest + Config ispec.Image } -type RepoMetadata2 struct { +type RepoMeta struct { Name string Tags map[string]Descriptor @@ -282,55 +215,38 @@ type RepoMetadata2 struct { Signatures map[string]ManifestSignatures Referrers map[string][]ReferrerInfo + LastUpdatedImage *LastUpdatedImage + Platforms []ispec.Platform + Vendors []string + Size int64 + IsStarred bool IsBookmarked bool Rank int - Stars int -} - -// used in repo search -type RepoStatistics struct { - Platforms []ispec.Platform - Vendors []string - Size string - LastUpdated time.Time -} - -type FullRepoMetadata struct { - Name string - Tags map[string]Descriptor - Size int64 - Platforms []ispec.Platform - Vendors []string - IsStarred bool - IsBookmarked bool StarCount int DownloadCount int - Rank int - - LastUpdatedImage *LastUpdatedImage - Statistics map[string]DescriptorStatistics - Signatures map[string]ManifestSignatures - Referrers map[string][]ReferrerInfo } -type FullImageData struct { - Repo string - Tag string - MediaType string - Digest godigest.Digest - Size int64 - Index *ispec.Index - Manifests []FullManifestData +// ImageMeta is a condensed structure of all information needed about an image when searching MetaDB. +type ImageMeta struct { + Repo string + Tag string + MediaType string + Digest godigest.Digest + Size int64 + Index *ispec.Index + Manifests []ManifestMeta + IsStarred bool + IsBookmarked bool Referrers []ReferrerInfo Statistics DescriptorStatistics Signatures ManifestSignatures } -type FullManifestData struct { - ManifestData2 +type ManifestMeta struct { + ManifestData Referrers []ReferrerInfo Statistics DescriptorStatistics @@ -343,22 +259,6 @@ type LastUpdatedImage struct { LastUpdated *time.Time } -type ManifestMetadata struct { - ManifestBlob []byte - ConfigBlob []byte - DownloadCount int - Signatures ManifestSignatures -} - -type IndexData struct { - IndexBlob []byte -} - -type ManifestData struct { - ManifestBlob []byte - ConfigBlob []byte -} - type ReferrerInfo struct { Digest string MediaType string @@ -379,21 +279,6 @@ type DescriptorStatistics struct { type ManifestSignatures map[string][]SignatureInfo -type RepoMetadata struct { - Name string - Tags map[string]Descriptor - - Statistics map[string]DescriptorStatistics - Signatures map[string]ManifestSignatures - Referrers map[string][]ReferrerInfo - - IsStarred bool - IsBookmarked bool - Rank int - - Stars int -} - type LayerInfo struct { LayerDigest string LayerContent []byte diff --git a/pkg/meta/version/version_test.go b/pkg/meta/version/version_test.go index ec919ea988..8aebee9fc2 100644 --- a/pkg/meta/version/version_test.go +++ b/pkg/meta/version/version_test.go @@ -124,14 +124,14 @@ func TestVersioningDynamoDB(t *testing.T) { Convey("Tests", t, func() { params := mdynamodb.DBDriverParameters{ - Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), - Region: "us-east-2", - RepoMetaTablename: "RepoMetadataTable" + uuid.String(), - ManifestDataTablename: "ManifestDataTable" + uuid.String(), - IndexDataTablename: "IndexDataTable" + uuid.String(), - UserDataTablename: "UserDataTable" + uuid.String(), - APIKeyTablename: "ApiKeyTable" + uuid.String(), - VersionTablename: "Version" + uuid.String(), + Endpoint: os.Getenv("DYNAMODBMOCK_ENDPOINT"), + Region: "us-east-2", + RepoMetaTablename: "RepoMetadataTable" + uuid.String(), + RepoBlobsInfoTablename: "RepoBlobsInfoTablename" + uuid.String(), + ImageDataTablename: "ImageDataTablename" + uuid.String(), + UserDataTablename: "UserDataTable" + uuid.String(), + APIKeyTablename: "ApiKeyTable" + uuid.String(), + VersionTablename: "Version" + uuid.String(), } dynamoClient, err := mdynamodb.GetDynamoClient(params) @@ -142,8 +142,7 @@ func TestVersioningDynamoDB(t *testing.T) { dynamoWrapper, err := mdynamodb.New(dynamoClient, params, log) So(err, ShouldBeNil) - So(dynamoWrapper.ResetManifestDataTable(), ShouldBeNil) - So(dynamoWrapper.ResetRepoMetaTable(), ShouldBeNil) + So(dynamoWrapper.ResetTable(dynamoWrapper.RepoMetaTablename), ShouldBeNil) Convey("dbVersion is empty", func() { err := setDynamoDBVersion(dynamoWrapper.Client, params.VersionTablename, "") @@ -201,7 +200,7 @@ func setDynamoDBVersion(client *dynamodb.Client, versTable, vers string) error { ":Version": mdAttributeValue, }, Key: map[string]types.AttributeValue{ - "VersionKey": &types.AttributeValueMemberS{ + "Key": &types.AttributeValueMemberS{ Value: version.DBVersionKey, }, }, diff --git a/pkg/storage/storage.go b/pkg/storage/storage.go index 56da20b74e..7d3cf4cd75 100644 --- a/pkg/storage/storage.go +++ b/pkg/storage/storage.go @@ -3,7 +3,6 @@ package storage import ( "encoding/json" "fmt" - "regexp" "strings" "github.com/docker/distribution/registry/storage/driver/factory" @@ -228,10 +227,7 @@ func CheckIsImageSignature(repoName string, manifestBlob []byte, reference strin return true, NotationType, manifestContent.Subject.Digest, nil } - // check cosign - cosignTagRule := regexp.MustCompile(`sha256\-.+\.sig`) - - if tag := reference; cosignTagRule.MatchString(reference) { + if tag := reference; zcommon.IsCosignTag(reference) { prefixLen := len("sha256-") digestLen := 64 signedImageManifestDigestEncoded := tag[prefixLen : prefixLen+digestLen] diff --git a/pkg/test/common/utils.go b/pkg/test/common/utils.go index 43c6102ae5..d487cfbb0b 100644 --- a/pkg/test/common/utils.go +++ b/pkg/test/common/utils.go @@ -177,3 +177,35 @@ func CustomRedirectPolicy(noOfRedirect int) resty.RedirectPolicy { return nil }) } + +func AccumulateField[R any, T any](list []T, accFunc func(T) R) []R { + result := make([]R, 0, len(list)) + + for i := range list { + result = append(result, accFunc(list[i])) + } + + return result +} + +func ContainSameElements[T comparable](list1, list2 []T) bool { + if len(list1) != len(list2) { + return false + } + + count1 := map[T]int{} + count2 := map[T]int{} + + for i := range list1 { + count1[list1[i]]++ + count2[list2[i]]++ + } + + for key := range count1 { + if count1[key] != count2[key] { + return false + } + } + + return true +} diff --git a/pkg/test/deprecated/deprecated.go b/pkg/test/deprecated/deprecated.go index d6bca08739..4c6ed0ade3 100644 --- a/pkg/test/deprecated/deprecated.go +++ b/pkg/test/deprecated/deprecated.go @@ -387,7 +387,7 @@ func GetRandomMultiarchImage(reference string) (image.MultiarchImage, error) { index.SchemaVersion = 2 return image.MultiarchImage{ - Index: index, Images: images, Reference: reference, + Index: index, Images: images, }, err } diff --git a/pkg/test/image-utils/images.go b/pkg/test/image-utils/images.go index 38587f0735..2d1e5436a8 100644 --- a/pkg/test/image-utils/images.go +++ b/pkg/test/image-utils/images.go @@ -11,6 +11,7 @@ import ( "github.com/opencontainers/image-spec/specs-go" ispec "github.com/opencontainers/image-spec/specs-go/v1" + mTypes "zotregistry.io/zot/pkg/meta/types" storageConstants "zotregistry.io/zot/pkg/storage/constants" ) @@ -48,6 +49,9 @@ type ConfigBuilder interface { // an OCI artifact. // (see: https://github.com/opencontainers/image-spec/blob/main/manifest.md#guidelines-for-artifact-usage) ArtifactConfig(artifactType string) ManifestBuilder + // PlatformConfig is used when we're interesting in specifying only the platform of a manifest. + // Other fields of the config are random. + PlatformConfig(architecture, os string) ManifestBuilder // DefaultConfig sets the default config, platform linux/amd64. DefaultConfig() ManifestBuilder // CustomConfigBlob will set a custom blob as the image config without other checks. @@ -92,13 +96,17 @@ type Image struct { } func (img *Image) Digest() godigest.Digest { + if img.ManifestDescriptor.Digest != "" { + return img.ManifestDescriptor.Digest + } + + // when we'll migrate all code to the new format of creating images we can replace this with + // the value from manifestDescriptor blob, err := json.Marshal(img.Manifest) if err != nil { panic("unreachable: ispec.Manifest should always be marshable") } - // when we'll migrate all code to the new format of creating images we can replace this with - // the value from manifestDescriptor return godigest.FromBytes(blob) } @@ -106,6 +114,16 @@ func (img *Image) DigestStr() string { return img.Digest().String() } +func (img *Image) Size() int { + size := img.ConfigDescriptor.Size + img.ManifestDescriptor.Size + + for _, layer := range img.Manifest.Layers { + size += layer.Size + } + + return int(size) +} + func (img Image) Descriptor() ispec.Descriptor { return ispec.Descriptor{ MediaType: img.ManifestDescriptor.MediaType, @@ -122,6 +140,22 @@ func (img Image) DescriptorRef() *ispec.Descriptor { } } +func (img Image) AsImageData() mTypes.ImageData { + return mTypes.ImageData{ + MediaType: img.Manifest.MediaType, + Digest: img.ManifestDescriptor.Digest, + Size: img.ManifestDescriptor.Size, + Manifests: []mTypes.ManifestData{ + { + Size: img.ManifestDescriptor.Size, + Digest: img.ManifestDescriptor.Digest, + Manifest: img.Manifest, + Config: img.Config, + }, + }, + } +} + type Layer struct { Blob []byte MediaType string @@ -281,6 +315,16 @@ func (ib *BaseImageBuilder) DefaultConfig() ManifestBuilder { return ib.ImageConfig(GetDefaultConfig()) } +func (ib *BaseImageBuilder) PlatformConfig(arch, os string) ManifestBuilder { + conf := GetDefaultConfig() + + conf.Created = RandomDateRef(time.UTC) + conf.Author = getRandomAuthor() + conf.Platform = ispec.Platform{Architecture: arch, OS: os} + + return ib.ImageConfig(conf) +} + func (ib *BaseImageBuilder) EmptyConfig() ManifestBuilder { ib.configDescriptor = ispec.DescriptorEmptyJSON diff --git a/pkg/test/image-utils/images_test.go b/pkg/test/image-utils/images_test.go index 90b9692c0d..769497a9ad 100644 --- a/pkg/test/image-utils/images_test.go +++ b/pkg/test/image-utils/images_test.go @@ -166,16 +166,6 @@ func TestPredefinedImages(t *testing.T) { img = CreateRandomVulnerableImageWith().ArtifactType("art.type").Build() So(img.Manifest.ArtifactType, ShouldEqual, "art.type") }) - - Convey("Predefined Multiarch-Images", t, func() { - multiArch := CreateRandomMultiarch() - So(len(multiArch.Images), ShouldEqual, 3) - So(multiArch.Reference, ShouldResemble, multiArch.Digest().String()) - - multiArch = CreateVulnerableMultiarch() - So(len(multiArch.Images), ShouldEqual, 3) - So(multiArch.Reference, ShouldResemble, multiArch.Digest().String()) - }) } func TestImageMethods(t *testing.T) { diff --git a/pkg/test/image-utils/multiarch.go b/pkg/test/image-utils/multiarch.go index c8da8d67ed..b4ae59c492 100644 --- a/pkg/test/image-utils/multiarch.go +++ b/pkg/test/image-utils/multiarch.go @@ -11,9 +11,8 @@ import ( ) type MultiarchImage struct { - Index ispec.Index - Images []Image - Reference string + Index ispec.Index + Images []Image IndexDescriptor ispec.Descriptor } @@ -31,17 +30,27 @@ func (mi *MultiarchImage) DigestStr() string { return mi.Digest().String() } -func (mi *MultiarchImage) IndexData() mTypes.IndexData { - indexBlob, err := json.Marshal(mi.Index) - if err != nil { - panic("unreachable: ispec.Index should always be marshable") +func (mi MultiarchImage) AsImageData() mTypes.ImageData { + index := mi.Index + + manifests := make([]mTypes.ManifestData, 0, len(index.Manifests)) + + for _, image := range mi.Images { + manifests = append(manifests, image.AsImageData().Manifests...) } - return mTypes.IndexData{IndexBlob: indexBlob} + return mTypes.ImageData{ + MediaType: ispec.MediaTypeImageIndex, + Digest: mi.IndexDescriptor.Digest, + Size: mi.IndexDescriptor.Size, + Index: &index, + Manifests: manifests, + } } type ImagesBuilder interface { Images(images []Image) MultiarchBuilder + RandomImages(count int) MultiarchBuilder } type MultiarchBuilder interface { @@ -88,6 +97,18 @@ func (mb *BaseMultiarchBuilder) Images(images []Image) MultiarchBuilder { return mb } +func (mb *BaseMultiarchBuilder) RandomImages(count int) MultiarchBuilder { + images := make([]Image, count) + + for i := range images { + images[i] = CreateRandomImage() + } + + mb.images = images + + return mb +} + func (mb *BaseMultiarchBuilder) Subject(subject *ispec.Descriptor) MultiarchBuilder { mb.subject = subject @@ -135,12 +156,9 @@ func (mb *BaseMultiarchBuilder) Build() MultiarchImage { indexDigest := godigest.FromBytes(indexBlob) - ref := indexDigest.String() - return MultiarchImage{ - Index: index, - Images: mb.images, - Reference: ref, + Index: index, + Images: mb.images, IndexDescriptor: ispec.Descriptor{ MediaType: ispec.MediaTypeImageIndex, diff --git a/pkg/test/image-utils/upload.go b/pkg/test/image-utils/upload.go index 2f5f581928..bda283f046 100644 --- a/pkg/test/image-utils/upload.go +++ b/pkg/test/image-utils/upload.go @@ -223,9 +223,15 @@ func UploadMultiarchImage(multiImage MultiarchImage, baseURL string, repo, ref s } // put manifest - indexBlob, err := json.Marshal(multiImage.Index) - if err = inject.Error(err); err != nil { - return err + indexBlob := multiImage.IndexDescriptor.Data + + if len(indexBlob) == 0 { + var err error + + indexBlob, err = json.Marshal(multiImage.Index) + if err = inject.Error(err); err != nil { + return err + } } resp, err := resty.R(). diff --git a/pkg/test/image-utils/utils.go b/pkg/test/image-utils/utils.go index 0b1289bed1..6a4d567e04 100644 --- a/pkg/test/image-utils/utils.go +++ b/pkg/test/image-utils/utils.go @@ -194,3 +194,12 @@ func GetIndexBlobWithManifests(manifestDigests []godigest.Digest) ([]byte, error return json.Marshal(indexContent) } + +func PlatformConfig(arch, os string) ispec.Image { + return ispec.Image{ + Platform: ispec.Platform{ + Architecture: arch, + OS: os, + }, + } +} diff --git a/pkg/test/mocks/repo_db_mock.go b/pkg/test/mocks/repo_db_mock.go index 44c7ccd366..1a95eb5194 100644 --- a/pkg/test/mocks/repo_db_mock.go +++ b/pkg/test/mocks/repo_db_mock.go @@ -5,79 +5,10 @@ import ( godigest "github.com/opencontainers/go-digest" - "zotregistry.io/zot/pkg/meta/proto_go" mTypes "zotregistry.io/zot/pkg/meta/types" ) type MetaDBMock struct { - SetRepoDescriptionFn func(repo, description string) error - - IncrementRepoStarsFn func(repo string) error - - DecrementRepoStarsFn func(repo string) error - - GetRepoStarsFn func(repo string) (int, error) - - SetRepoLogoFn func(repo string, logoPath string) error - - SetRepoReferenceFn func(repo string, Reference string, manifestDigest godigest.Digest, mediaType string) error - - RemoveRepoReferenceFn func(repo, reference string, manifestDigest godigest.Digest) error - - DeleteRepoTagFn func(repo string, tag string) error - - GetRepoMetaFn func(repo string) (mTypes.RepoMetadata, error) - - GetUserRepoMetaFn func(ctx context.Context, repo string) (mTypes.RepoMetadata, error) - - SetRepoMetaFn func(repo string, repoMeta mTypes.RepoMetadata) error - - GetMultipleRepoMetaFn func(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata) bool) ( - []mTypes.RepoMetadata, error) - - GetManifestDataFn func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) - - SetManifestDataFn func(manifestDigest godigest.Digest, mm mTypes.ManifestData) error - - GetManifestMetaFn func(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) - - SetManifestMetaFn func(repo string, manifestDigest godigest.Digest, mm mTypes.ManifestMetadata) error - - SetIndexDataFn func(digest godigest.Digest, indexData mTypes.IndexData) error - - GetIndexDataFn func(indexDigest godigest.Digest) (mTypes.IndexData, error) - - SetReferrerFn func(repo string, referredDigest godigest.Digest, referrer mTypes.ReferrerInfo) error - - DeleteReferrerFn func(repo string, referredDigest godigest.Digest, referrerDigest godigest.Digest) error - - GetReferrersFn func(repo string, referredDigest godigest.Digest) ([]mTypes.Descriptor, error) - - GetReferrersInfoFn func(repo string, referredDigest godigest.Digest, artifactTypes []string) ( - []mTypes.ReferrerInfo, error) - - IncrementImageDownloadsFn func(repo string, reference string) error - - UpdateSignaturesValidityFn func(repo string, manifestDigest godigest.Digest) error - - AddManifestSignatureFn func(repo string, signedManifestDigest godigest.Digest, sm mTypes.SignatureMetadata) error - - DeleteSignatureFn func(repo string, signedManifestDigest godigest.Digest, sm mTypes.SignatureMetadata) error - - SearchReposFn func(ctx context.Context, txt string) ( - []mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error) - - SearchTagsFn func(ctx context.Context, txt string) ( - []mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, - error) - - FilterReposFn func(ctx context.Context, filter mTypes.FilterRepoFunc) ( - []mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) - - FilterTagsFn func(ctx context.Context, filterFunc mTypes.FilterFunc) ( - []mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) - GetStarredReposFn func(ctx context.Context) ([]string, error) GetBookmarkedReposFn func(ctx context.Context) ([]string, error) @@ -114,258 +45,84 @@ type MetaDBMock struct { SetImageTrustStoreFn func(mTypes.ImageTrustStore) - SetProtoImageDataFn func(digest godigest.Digest, imageData mTypes.ImageData2) error - - ProtoSetReferrerFn func(repo string, referredDigest godigest.Digest, referrer *proto_go.ReferrerInfo) error - ProtoSetRepoReferenceFn func(repo string, reference string, manifestDigest godigest.Digest, mediaType string) error - ProtoSearchReposFn func(ctx context.Context, searchText string) ([]mTypes.RepoMetadata2, map[string]mTypes.ImageData2, error) - ProtoSearchTagsFn func(ctx context.Context, searchText string) ([]mTypes.FullImageData, error) - ProtoFilterTagFn func(ctx context.Context, filterFunc mTypes.FilterProtoFunc) ([]mTypes.RepoMetadata2, map[string]mTypes.ImageData2, error) - ProtoGetRepoMetaFn func(repo string) (mTypes.RepoMetadata2, error) - ProtoGetImageDataFn func(digest godigest.Digest) (mTypes.ImageData2, error) - ProtoGetMultipleRepoMetaFn func(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata2) bool) ([]mTypes.RepoMetadata2, error) - ProtoFilterReposFn func(ctx context.Context, filter mTypes.FilterProtoRepoFunc) ([]mTypes.RepoMetadata2, map[string]mTypes.ImageData2, error) - ProtoIncrementRepoStarsFn func(repo string) error - ProtoDecrementRepoStarsFn func(repo string) error - ProtoGetRepoStarsFn func(repo string) (int, error) - ProtoDeleteRepoTagFn func(repo string, tag string) error - ProtoGetUserRepoMetaFn func(ctx context.Context, repo string) (mTypes.RepoMetadata2, error) - ProtoSetRepoMetaFn func(repo string, repoMeta mTypes.RepoMetadata2) error - ProtoDeleteReferrerFn func(repo string, referredDigest godigest.Digest, referrerDigest godigest.Digest) error - ProtoGetReferrersInfoFn func(repo string, referredDigest godigest.Digest, artifactTypes []string) ([]mTypes.ReferrerInfo, error) - ProtoIncrementImageDownloadsFn func(repo string, reference string) error - ProtoUpdateSignaturesValidityFn func(repo string, manifestDigest godigest.Digest) error - ProtoAddManifestSignatureFn func(repo string, signedManifestDigest godigest.Digest, sygMeta mTypes.SignatureMetadata) error - ProtoDeleteSignatureFn func(repo string, signedManifestDigest godigest.Digest, sigMeta mTypes.SignatureMetadata) error -} + SetRepoReferenceFn func(repo string, reference string, imageData mTypes.ImageData) error -func (sdm MetaDBMock) ImageTrustStore() mTypes.ImageTrustStore { - if sdm.ImageTrustStoreFn != nil { - return sdm.ImageTrustStoreFn() - } + SearchReposFn func(ctx context.Context, searchText string, + ) ([]mTypes.RepoMeta, error) - return nil -} + SearchTagsFn func(ctx context.Context, searchText string) ([]mTypes.ImageMeta, error) -func (sdm MetaDBMock) SetImageTrustStore(imgTrustStore mTypes.ImageTrustStore) { - if sdm.SetImageTrustStoreFn != nil { - sdm.SetImageTrustStoreFn(imgTrustStore) - } -} + FilterTagFn func(ctx context.Context, filterFunc mTypes.FilterFunc, + ) ([]mTypes.RepoMeta, map[string]mTypes.ImageData, error) -func (sdm MetaDBMock) SetRepoDescription(repo, description string) error { - if sdm.SetRepoDescriptionFn != nil { - return sdm.SetRepoDescriptionFn(repo, description) - } + GetImageDataFn func(digest godigest.Digest) (mTypes.ImageData, error) - return nil -} + GetMultipleRepoMetaFn func(ctx context.Context, filter func(repoMeta mTypes.RepoMeta) bool, + ) ([]mTypes.RepoMeta, error) -func (sdm MetaDBMock) IncrementRepoStars(repo string) error { - if sdm.IncrementRepoStarsFn != nil { - return sdm.IncrementRepoStarsFn(repo) - } - - return nil -} - -func (sdm MetaDBMock) DecrementRepoStars(repo string) error { - if sdm.DecrementRepoStarsFn != nil { - return sdm.DecrementRepoStarsFn(repo) - } + FilterReposFn func(ctx context.Context, rankName mTypes.FilterRepoNameFunc, + filterFunc mTypes.FilterFullRepoFunc) ([]mTypes.RepoMeta, error) - return nil -} - -func (sdm MetaDBMock) GetRepoStars(repo string) (int, error) { - if sdm.GetRepoStarsFn != nil { - return sdm.GetRepoStarsFn(repo) - } - - return 0, nil -} - -func (sdm MetaDBMock) SetRepoReference(repo string, reference string, manifestDigest godigest.Digest, - mediaType string, -) error { - if sdm.SetRepoReferenceFn != nil { - return sdm.SetRepoReferenceFn(repo, reference, manifestDigest, mediaType) - } - - return nil -} - -func (sdm MetaDBMock) RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest) error { - if sdm.RemoveRepoReferenceFn != nil { - return sdm.RemoveRepoReferenceFn(repo, reference, manifestDigest) - } - - return nil -} - -func (sdm MetaDBMock) DeleteRepoTag(repo string, tag string) error { - if sdm.DeleteRepoTagFn != nil { - return sdm.DeleteRepoTagFn(repo, tag) - } - - return nil -} - -func (sdm MetaDBMock) GetRepoMeta(repo string) (mTypes.RepoMetadata, error) { - if sdm.GetRepoMetaFn != nil { - return sdm.GetRepoMetaFn(repo) - } - - return mTypes.RepoMetadata{}, nil -} - -func (sdm MetaDBMock) GetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMetadata, error) { - if sdm.GetUserRepoMetaFn != nil { - return sdm.GetUserRepoMetaFn(ctx, repo) - } - - return mTypes.RepoMetadata{}, nil -} - -func (sdm MetaDBMock) SetRepoMeta(repo string, repoMeta mTypes.RepoMetadata) error { - if sdm.SetRepoMetaFn != nil { - return sdm.SetRepoMetaFn(repo, repoMeta) - } + IncrementRepoStarsFn func(repo string) error - return nil -} + DecrementRepoStarsFn func(repo string) error -func (sdm MetaDBMock) GetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata) bool, -) ([]mTypes.RepoMetadata, error) { - if sdm.GetMultipleRepoMetaFn != nil { - return sdm.GetMultipleRepoMetaFn(ctx, filter) - } + GetUserRepoMetaFn func(ctx context.Context, repo string) (mTypes.RepoMeta, error) - return []mTypes.RepoMetadata{}, nil -} + SetRepoMetaFn func(repo string, repoMeta mTypes.RepoMeta) error -func (sdm MetaDBMock) GetManifestData(manifestDigest godigest.Digest) (mTypes.ManifestData, error) { - if sdm.GetManifestDataFn != nil { - return sdm.GetManifestDataFn(manifestDigest) - } + DeleteReferrerFn func(repo string, referredDigest godigest.Digest, referrerDigest godigest.Digest) error - return mTypes.ManifestData{}, nil -} + GetReferrersInfoFn func(repo string, referredDigest godigest.Digest, artifactTypes []string, + ) ([]mTypes.ReferrerInfo, error) -func (sdm MetaDBMock) SetManifestData(manifestDigest godigest.Digest, md mTypes.ManifestData) error { - if sdm.SetManifestDataFn != nil { - return sdm.SetManifestDataFn(manifestDigest, md) - } + IncrementImageDownloadsFn func(repo string, reference string) error - return nil -} + UpdateSignaturesValidityFn func(repo string, manifestDigest godigest.Digest) error -func (sdm MetaDBMock) GetManifestMeta(repo string, manifestDigest godigest.Digest) (mTypes.ManifestMetadata, error) { - if sdm.GetManifestMetaFn != nil { - return sdm.GetManifestMetaFn(repo, manifestDigest) - } + AddManifestSignatureFn func(repo string, signedManifestDigest godigest.Digest, sygMeta mTypes.SignatureMetadata, + ) error - return mTypes.ManifestMetadata{}, nil -} + DeleteSignatureFn func(repo string, signedManifestDigest godigest.Digest, sigMeta mTypes.SignatureMetadata) error -func (sdm MetaDBMock) SetManifestMeta(repo string, manifestDigest godigest.Digest, mm mTypes.ManifestMetadata) error { - if sdm.SetManifestMetaFn != nil { - return sdm.SetManifestMetaFn(repo, manifestDigest, mm) - } + SetImageDataFn func(digest godigest.Digest, imageData mTypes.ImageData) error - return nil -} + FilterTagsFn func(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, + filterFunc mTypes.FilterFunc) ([]mTypes.ImageMeta, error) -func (sdm MetaDBMock) IncrementImageDownloads(repo string, reference string) error { - if sdm.IncrementImageDownloadsFn != nil { - return sdm.IncrementImageDownloadsFn(repo, reference) - } + GetRepoMetaFn func(ctx context.Context, repo string) (mTypes.RepoMeta, error) - return nil -} + FilterImageDataFn func(ctx context.Context, digests []string) (map[string]mTypes.ImageData, error) -func (sdm MetaDBMock) UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { - if sdm.UpdateSignaturesValidityFn != nil { - return sdm.UpdateSignaturesValidityFn(repo, manifestDigest) - } + RemoveRepoReferenceFn func(repo, reference string, manifestDigest godigest.Digest) error - return nil -} + GetImageMetaFn func(ctx context.Context, repo string, tag string) (mTypes.ImageMeta, error) -func (sdm MetaDBMock) AddManifestSignature(repo string, signedManifestDigest godigest.Digest, - sm mTypes.SignatureMetadata, -) error { - if sdm.AddManifestSignatureFn != nil { - return sdm.AddManifestSignatureFn(repo, signedManifestDigest, sm) - } + ResetRepoReferencesFn func(repo string) error - return nil + ResetDBFn func() error } -func (sdm MetaDBMock) DeleteSignature(repo string, signedManifestDigest godigest.Digest, - sm mTypes.SignatureMetadata, -) error { - if sdm.DeleteSignatureFn != nil { - return sdm.DeleteSignatureFn(repo, signedManifestDigest, sm) +func (sdm MetaDBMock) ResetDB() error { + if sdm.ResetDBFn != nil { + return sdm.ResetDBFn() } return nil } -func (sdm MetaDBMock) SearchRepos(ctx context.Context, searchText string, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - if sdm.SearchReposFn != nil { - return sdm.SearchReposFn(ctx, searchText) - } - - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{}, nil -} - -func (sdm MetaDBMock) SearchTags(ctx context.Context, searchText string, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - if sdm.SearchTagsFn != nil { - return sdm.SearchTagsFn(ctx, searchText) - } - - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{}, nil -} - -func (sdm MetaDBMock) FilterRepos(ctx context.Context, filter mTypes.FilterRepoFunc, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - if sdm.FilterReposFn != nil { - return sdm.FilterReposFn(ctx, filter) - } - - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{}, nil -} - -func (sdm MetaDBMock) FilterTags(ctx context.Context, filterFunc mTypes.FilterFunc, -) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, map[string]mTypes.IndexData, error) { - if sdm.FilterTagsFn != nil { - return sdm.FilterTagsFn(ctx, filterFunc) - } - - return []mTypes.RepoMetadata{}, map[string]mTypes.ManifestMetadata{}, - map[string]mTypes.IndexData{}, nil -} - -func (sdm MetaDBMock) SetIndexData(digest godigest.Digest, indexData mTypes.IndexData) error { - if sdm.SetIndexDataFn != nil { - return sdm.SetIndexDataFn(digest, indexData) +func (sdm MetaDBMock) ImageTrustStore() mTypes.ImageTrustStore { + if sdm.ImageTrustStoreFn != nil { + return sdm.ImageTrustStoreFn() } return nil } -func (sdm MetaDBMock) GetIndexData(indexDigest godigest.Digest) (mTypes.IndexData, error) { - if sdm.GetIndexDataFn != nil { - return sdm.GetIndexDataFn(indexDigest) +func (sdm MetaDBMock) SetImageTrustStore(imgTrustStore mTypes.ImageTrustStore) { + if sdm.SetImageTrustStoreFn != nil { + sdm.SetImageTrustStoreFn(imgTrustStore) } - - return mTypes.IndexData{}, nil } func (sdm MetaDBMock) PatchDB() error { @@ -376,34 +133,6 @@ func (sdm MetaDBMock) PatchDB() error { return nil } -func (sdm MetaDBMock) SetReferrer(repo string, referredDigest godigest.Digest, referrer mTypes.ReferrerInfo) error { - if sdm.SetReferrerFn != nil { - return sdm.SetReferrerFn(repo, referredDigest, referrer) - } - - return nil -} - -func (sdm MetaDBMock) DeleteReferrer(repo string, referredDigest godigest.Digest, - referrerDigest godigest.Digest, -) error { - if sdm.DeleteReferrerFn != nil { - return sdm.DeleteReferrerFn(repo, referredDigest, referrerDigest) - } - - return nil -} - -func (sdm MetaDBMock) GetReferrersInfo(repo string, referredDigest godigest.Digest, - artifactTypes []string, -) ([]mTypes.ReferrerInfo, error) { - if sdm.GetReferrersInfoFn != nil { - return sdm.GetReferrersInfoFn(repo, referredDigest, artifactTypes) - } - - return []mTypes.ReferrerInfo{}, nil -} - func (sdm MetaDBMock) GetStarredRepos(ctx context.Context) ([]string, error) { if sdm.GetStarredReposFn != nil { return sdm.GetStarredReposFn(ctx) @@ -524,107 +253,191 @@ func (sdm MetaDBMock) DeleteUserAPIKey(ctx context.Context, id string) error { return nil } -// TODO -func (sdm MetaDBMock) SetProtoImageData(digest godigest.Digest, imageData mTypes.ImageData2) error { - if sdm.SetProtoImageDataFn != nil { - return sdm.SetProtoImageDataFn(digest, imageData) +func (sdm MetaDBMock) SetImageData(digest godigest.Digest, imageData mTypes.ImageData) error { + if sdm.SetImageDataFn != nil { + return sdm.SetImageDataFn(digest, imageData) } return nil } -// TODO -func (sdm MetaDBMock) ProtoSetImageData(digest godigest.Digest, imageData mTypes.ImageData2) error { +func (sdm MetaDBMock) SetRepoReference(repo string, reference string, imageData mTypes.ImageData) error { + if sdm.SetRepoReferenceFn != nil { + return sdm.SetRepoReferenceFn(repo, reference, imageData) + } return nil } -// TODO -func (bdw MetaDBMock) ProtoSetRepoReference(repo string, reference string, imageData mTypes.ImageData2) error { - return nil -} +func (sdm MetaDBMock) SearchRepos(ctx context.Context, searchText string) ([]mTypes.RepoMeta, error) { + if sdm.SearchReposFn != nil { + return sdm.SearchReposFn(ctx, searchText) + } -// TODO -func (bdw MetaDBMock) ProtoSearchRepos(ctx context.Context, searchText string) ([]mTypes.FullRepoMetadata, error) { - return []mTypes.FullRepoMetadata{}, nil + return []mTypes.RepoMeta{}, nil } -// TODO -func (bdw MetaDBMock) ProtoSearchTags(ctx context.Context, searchText string) ([]mTypes.FullImageData, error) { - return []mTypes.FullImageData{}, nil -} +func (sdm MetaDBMock) SearchTags(ctx context.Context, searchText string) ([]mTypes.ImageMeta, error) { + if sdm.SearchTagsFn != nil { + return sdm.SearchTagsFn(ctx, searchText) + } -func (bdw MetaDBMock) ProtoFilterTags(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, filterFunc mTypes.FilterProtoFunc, -) ([]mTypes.FullImageData, error) { - return []mTypes.FullImageData{}, nil + return []mTypes.ImageMeta{}, nil } -func (bdw MetaDBMock) ProtoGetRepoMeta(repo string) (mTypes.RepoMetadata2, error) { - return mTypes.RepoMetadata2{}, nil +func (sdm MetaDBMock) FilterTags(ctx context.Context, filterRepoTag mTypes.FilterRepoTagFunc, + filterFunc mTypes.FilterFunc, +) ([]mTypes.ImageMeta, error) { + if sdm.FilterTagsFn != nil { + return sdm.FilterTagsFn(ctx, filterRepoTag, filterFunc) + } + + return []mTypes.ImageMeta{}, nil } -func (bdw MetaDBMock) ProtoGetFullRepoMeta(ctx context.Context, repo string) (mTypes.FullRepoMetadata, error) { - return mTypes.FullRepoMetadata{}, nil +func (sdm MetaDBMock) GetRepoMeta(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + if sdm.GetRepoMetaFn != nil { + return sdm.GetRepoMetaFn(ctx, repo) + } + + return mTypes.RepoMeta{}, nil } -func (bdw MetaDBMock) ProtoGetImageData(digest godigest.Digest) (mTypes.ImageData2, error) { - return mTypes.ImageData2{}, nil +func (sdm MetaDBMock) GetImageData(digest godigest.Digest) (mTypes.ImageData, error) { + if sdm.GetImageDataFn != nil { + return sdm.GetImageDataFn(digest) + } + + return mTypes.ImageData{}, nil } -func (bdw MetaDBMock) ProtoGetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMetadata2) bool) ([]mTypes.RepoMetadata2, error) { - return []mTypes.RepoMetadata2{}, nil +func (sdm MetaDBMock) GetMultipleRepoMeta(ctx context.Context, filter func(repoMeta mTypes.RepoMeta) bool, +) ([]mTypes.RepoMeta, error) { + if sdm.GetMultipleRepoMetaFn != nil { + return sdm.GetMultipleRepoMetaFn(ctx, filter) + } + + return []mTypes.RepoMeta{}, nil } -func (bdw MetaDBMock) ProtoFilterRepos(ctx context.Context, rankName mTypes.FilterRepoNameFunc, filterFunc mTypes.FilterFullRepoFunc, -) ([]mTypes.FullRepoMetadata, error) { - return []mTypes.FullRepoMetadata{}, nil +func (sdm MetaDBMock) FilterRepos(ctx context.Context, rankName mTypes.FilterRepoNameFunc, + filterFunc mTypes.FilterFullRepoFunc, +) ([]mTypes.RepoMeta, error) { + if sdm.FilterReposFn != nil { + return sdm.FilterReposFn(ctx, rankName, filterFunc) + } + + return []mTypes.RepoMeta{}, nil } -func (bdw MetaDBMock) ProtoIncrementRepoStars(repo string) error { +func (sdm MetaDBMock) IncrementRepoStars(repo string) error { + if sdm.IncrementRepoStarsFn != nil { + return sdm.IncrementRepoStarsFn(repo) + } + return nil } -func (bdw MetaDBMock) ProtoDecrementRepoStars(repo string) error { +func (sdm MetaDBMock) DecrementRepoStars(repo string) error { + if sdm.DecrementRepoStarsFn != nil { + return sdm.DecrementRepoStarsFn(repo) + } + return nil } -func (bdw MetaDBMock) ProtoGetRepoStars(repo string) (int, error) { - return 0, nil +func (sdm MetaDBMock) GetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMeta, error) { + if sdm.GetRepoMetaFn != nil { + return sdm.GetRepoMetaFn(ctx, repo) + } + + return mTypes.RepoMeta{}, nil } -func (bdw MetaDBMock) ProtoDeleteRepoTag(repo string, tag string) error { +func (sdm MetaDBMock) SetRepoMeta(repo string, repoMeta mTypes.RepoMeta) error { + if sdm.SetRepoMetaFn != nil { + return sdm.SetRepoMetaFn(repo, repoMeta) + } + return nil } -func (bdw MetaDBMock) ProtoGetUserRepoMeta(ctx context.Context, repo string) (mTypes.RepoMetadata2, error) { - return mTypes.RepoMetadata2{}, nil +func (sdm MetaDBMock) GetReferrersInfo(repo string, referredDigest godigest.Digest, + artifactTypes []string, +) ([]mTypes.ReferrerInfo, error) { + if sdm.GetReferrersInfoFn != nil { + return sdm.GetReferrersInfoFn(repo, referredDigest, artifactTypes) + } + + return []mTypes.ReferrerInfo{}, nil } -func (bdw MetaDBMock) ProtoSetRepoMeta(repo string, repoMeta mTypes.RepoMetadata2) error { +func (sdm MetaDBMock) IncrementImageDownloads(repo string, reference string) error { + if sdm.IncrementImageDownloadsFn != nil { + return sdm.IncrementImageDownloadsFn(repo, reference) + } + return nil } -func (bdw MetaDBMock) ProtoGetReferrersInfo(repo string, referredDigest godigest.Digest, artifactTypes []string) ([]mTypes.ReferrerInfo, error) { - return []mTypes.ReferrerInfo{}, nil -} +func (sdm MetaDBMock) UpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { + if sdm.UpdateSignaturesValidityFn != nil { + return sdm.UpdateSignaturesValidityFn(repo, manifestDigest) + } -func (bdw MetaDBMock) ProtoIncrementImageDownloads(repo string, reference string) error { return nil } -func (bdw MetaDBMock) ProtoUpdateSignaturesValidity(repo string, manifestDigest godigest.Digest) error { +func (sdm MetaDBMock) AddManifestSignature(repo string, signedManifestDigest godigest.Digest, + sygMeta mTypes.SignatureMetadata, +) error { + if sdm.AddManifestSignatureFn != nil { + return sdm.AddManifestSignatureFn(repo, signedManifestDigest, sygMeta) + } + return nil } -func (bdw MetaDBMock) ProtoAddManifestSignature(repo string, signedManifestDigest godigest.Digest, sygMeta mTypes.SignatureMetadata) error { +func (sdm MetaDBMock) DeleteSignature(repo string, signedManifestDigest godigest.Digest, + sigMeta mTypes.SignatureMetadata, +) error { + if sdm.DeleteSignatureFn != nil { + return sdm.DeleteSignatureFn(repo, signedManifestDigest, sigMeta) + } + return nil } -func (bdw MetaDBMock) ProtoDeleteSignature(repo string, signedManifestDigest godigest.Digest, sigMeta mTypes.SignatureMetadata) error { +func (sdm MetaDBMock) FilterImageData(ctx context.Context, digests []string, +) (map[string]mTypes.ImageData, error) { + if sdm.FilterImageDataFn != nil { + return sdm.FilterImageDataFn(ctx, digests) + } + + return map[string]mTypes.ImageData{}, nil +} + +func (sdm MetaDBMock) RemoveRepoReference(repo, reference string, manifestDigest godigest.Digest) error { + if sdm.RemoveRepoReferenceFn != nil { + return sdm.RemoveRepoReferenceFn(repo, reference, manifestDigest) + } + return nil } -func (bdw MetaDBMock) ProtoFilterImageData(ctx context.Context, digests []string, -) (map[string]mTypes.ImageData2, error) { - return map[string]mTypes.ImageData2{}, nil +func (sdm MetaDBMock) GetImageMeta(ctx context.Context, repo string, tag string, +) (mTypes.ImageMeta, error) { + if sdm.GetImageMetaFn != nil { + return sdm.GetImageMetaFn(ctx, repo, tag) + } + + return mTypes.ImageMeta{}, nil +} + +func (sdm MetaDBMock) ResetRepoReferences(repo string) error { + if sdm.ResetRepoReferencesFn != nil { + return sdm.ResetRepoReferencesFn(repo) + } + + return nil } diff --git a/pkg/test/oci-utils/repo.go b/pkg/test/oci-utils/repo.go index dc14a3c59a..5b6b490015 100644 --- a/pkg/test/oci-utils/repo.go +++ b/pkg/test/oci-utils/repo.go @@ -1,110 +1,142 @@ package ociutils import ( - ispec "github.com/opencontainers/image-spec/specs-go/v1" + "context" + "fmt" + zerr "zotregistry.io/zot/errors" mTypes "zotregistry.io/zot/pkg/meta/types" + reqCtx "zotregistry.io/zot/pkg/requestcontext" imageUtil "zotregistry.io/zot/pkg/test/image-utils" ) type RepoImage struct { imageUtil.Image - Tag string + Reference string Statistics mTypes.DescriptorStatistics } type RepoMultiArchImage struct { imageUtil.MultiarchImage ImageStatistics map[string]mTypes.DescriptorStatistics - Tag string + Reference string } type Repo struct { Name string Images []RepoImage MultiArchImages []RepoMultiArchImage + Signatures map[string]mTypes.ManifestSignatures + Stars int IsBookmarked bool IsStarred bool } -func GetMetadataForRepos(repos ...Repo) ([]mTypes.RepoMetadata, map[string]mTypes.ManifestMetadata, - map[string]mTypes.IndexData, -) { - var ( - reposMetadata = []mTypes.RepoMetadata{} - manifestMetadataMap = map[string]mTypes.ManifestMetadata{} - indexDataMap = map[string]mTypes.IndexData{} - ) +func InitializeTestMetaDB(ctx context.Context, metaDB mTypes.MetaDB, repos ...Repo) (context.Context, error) { + uac := reqCtx.NewUserAccessControl() + uac.SetUsername("test") + uacContext := context.WithValue(ctx, reqCtx.GetContextKey(), *uac) + + err := validateRepos(repos) + if err != nil { + return uacContext, err + } for _, repo := range repos { - repoMeta := mTypes.RepoMetadata{ - Name: repo.Name, - Tags: map[string]mTypes.Descriptor{}, - Signatures: map[string]mTypes.ManifestSignatures{}, - Statistics: map[string]mTypes.DescriptorStatistics{}, - IsStarred: repo.IsStarred, - IsBookmarked: repo.IsBookmarked, - } + statistics := map[string]mTypes.DescriptorStatistics{"": {}} for _, image := range repo.Images { - addImageMetaToMetaDB(image, repoMeta, manifestMetadataMap) + err := metaDB.SetRepoReference(repo.Name, image.Reference, image.AsImageData()) + if err != nil { + return uacContext, err + } + + statistics[image.DigestStr()] = image.Statistics } for _, multiArch := range repo.MultiArchImages { - if multiArch.ImageStatistics == nil { - multiArch.ImageStatistics = map[string]mTypes.DescriptorStatistics{} + for _, image := range multiArch.Images { + err := metaDB.SetRepoReference(repo.Name, image.DigestStr(), image.AsImageData()) + if err != nil { + return uacContext, err + } + + statistics[image.DigestStr()] = multiArch.ImageStatistics[image.DigestStr()] } - repoMeta.Tags[multiArch.Tag] = mTypes.Descriptor{ - MediaType: ispec.MediaTypeImageIndex, - Digest: multiArch.DigestStr(), + err := metaDB.SetRepoReference(repo.Name, multiArch.Reference, multiArch.AsImageData()) + if err != nil { + return uacContext, err } - repoMeta.Statistics[multiArch.DigestStr()] = multiArch.ImageStatistics[multiArch.DigestStr()] + statistics[multiArch.DigestStr()] = multiArch.ImageStatistics[multiArch.DigestStr()] + } - for _, image := range multiArch.Images { - addImageMetaToMetaDB(RepoImage{ - Image: image, - Statistics: multiArch.ImageStatistics[image.DigestStr()], - }, repoMeta, manifestMetadataMap) - } + // Update repo metadata + repoMeta, err := metaDB.GetRepoMeta(ctx, repo.Name) + if err != nil { + return uacContext, err + } + + repoMeta.StarCount = repo.Stars + repoMeta.IsStarred = repo.IsStarred + repoMeta.IsBookmarked = repo.IsBookmarked + + // updateStatistics + for key, value := range statistics { + repoMeta.Statistics[key] = value + } + + // update signatures? + for key, value := range repo.Signatures { + repoMeta.Signatures[key] = value + } - indexDataMap[multiArch.IndexDescriptor.Digest.String()] = mTypes.IndexData{ - IndexBlob: multiArch.IndexDescriptor.Data, + err = metaDB.SetRepoMeta(repo.Name, repoMeta) + if err != nil { + return uacContext, err + } + + // User data is set after we create the repo + if repo.IsBookmarked { + _, err := metaDB.ToggleBookmarkRepo(uacContext, repo.Name) + if err != nil { + return uacContext, err } } - reposMetadata = append(reposMetadata, repoMeta) + if repo.IsStarred { + _, err := metaDB.ToggleStarRepo(uacContext, repo.Name) + if err != nil { + return uacContext, err + } + } } - return reposMetadata, manifestMetadataMap, indexDataMap + return uacContext, nil } -func addImageMetaToMetaDB(image RepoImage, repoMeta mTypes.RepoMetadata, - manifestMetadataMap map[string]mTypes.ManifestMetadata, -) { - if image.Tag != "" { - repoMeta.Tags[image.Tag] = mTypes.Descriptor{ - MediaType: ispec.MediaTypeImageManifest, - Digest: image.DigestStr(), - } - } - // here we can do many more checks about the images like check for referrers, signatures but it's not needed yet - // I need just the tags for now and the fake signature. - - // This is done just to mark a manifest as signed in the resulted RepoMeta - if image.Manifest.ArtifactType == imageUtil.TestFakeSignatureArtType && image.Manifest.Subject != nil { - signedManifestDig := image.Manifest.Subject.Digest.String() - repoMeta.Signatures[signedManifestDig] = mTypes.ManifestSignatures{ - "fakeSignature": []mTypes.SignatureInfo{{SignatureManifestDigest: image.ManifestDescriptor.Digest.String()}}, +func validateRepos(repos []Repo) error { + repoNames := map[string]struct{}{} + + for _, repo := range repos { + if _, found := repoNames[repo.Name]; found { + return fmt.Errorf("%w '%s'", zerr.ErrMultipleReposSameName, repo.Name) } + + repoNames[repo.Name] = struct{}{} } - repoMeta.Statistics[image.DigestStr()] = image.Statistics + return nil +} - manifestMetadataMap[image.DigestStr()] = mTypes.ManifestMetadata{ - ManifestBlob: image.ManifestDescriptor.Data, - ConfigBlob: image.ConfigDescriptor.Data, - DownloadCount: image.Statistics.DownloadCount, +func GetFakeSignatureInfo(signatureDigest string) map[string][]mTypes.SignatureInfo { + return map[string][]mTypes.SignatureInfo{ + "fake-signature": { + { + SignatureManifestDigest: signatureDigest, + LayersInfo: []mTypes.LayerInfo{}, + }, + }, } } diff --git a/test/blackbox/cloud_only.bats b/test/blackbox/cloud_only.bats index 5d970ffcce..c4351092d4 100644 --- a/test/blackbox/cloud_only.bats +++ b/test/blackbox/cloud_only.bats @@ -41,8 +41,8 @@ function setup() { "region": "us-east-2", "cacheTablename": "BlobTable", "repoMetaTablename": "RepoMetadataTable", - "manifestDataTablename": "ManifestDataTable", - "indexDataTablename": "IndexDataTable", + "imageDataTablename": "ImageDataTable", + "repoBlobsInfoTablename": "RepoBlobsInfoTable", "userDataTablename": "UserDataTable", "apiKeyTablename":"ApiKeyTable", "versionTablename": "Version"