2023-03-10 13:37:29 -05:00
|
|
|
package common_test
|
|
|
|
|
|
|
|
import (
|
2023-04-24 13:13:15 -05:00
|
|
|
"errors"
|
2023-03-10 13:37:29 -05:00
|
|
|
"testing"
|
2023-04-03 12:53:09 -05:00
|
|
|
"time"
|
2023-03-10 13:37:29 -05:00
|
|
|
|
2023-04-24 13:13:15 -05:00
|
|
|
"github.com/opencontainers/go-digest"
|
|
|
|
ispec "github.com/opencontainers/image-spec/specs-go/v1"
|
2023-03-10 13:37:29 -05:00
|
|
|
. "github.com/smartystreets/goconvey/convey"
|
|
|
|
|
2023-03-28 12:20:09 -05:00
|
|
|
"zotregistry.io/zot/pkg/meta/common"
|
2023-07-18 12:27:26 -05:00
|
|
|
mTypes "zotregistry.io/zot/pkg/meta/types"
|
2023-04-24 13:13:15 -05:00
|
|
|
"zotregistry.io/zot/pkg/test/mocks"
|
2023-03-10 13:37:29 -05:00
|
|
|
)
|
|
|
|
|
2023-04-24 13:13:15 -05:00
|
|
|
var ErrTestError = errors.New("test error")
|
|
|
|
|
2023-03-10 13:37:29 -05:00
|
|
|
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)
|
|
|
|
|
|
|
|
res = common.MatchesArtifactTypes("type", []string{"someOtherType"})
|
|
|
|
So(res, ShouldBeFalse)
|
|
|
|
})
|
2023-04-03 12:53:09 -05:00
|
|
|
|
|
|
|
Convey("CheckImageLastUpdated", t, func() {
|
|
|
|
Convey("No image checked, it doesn't have time", func() {
|
|
|
|
repoLastUpdated := time.Time{}
|
|
|
|
isSigned := false
|
|
|
|
noImageChecked := true
|
2023-07-18 12:27:26 -05:00
|
|
|
manifestFilterData := mTypes.FilterData{
|
2023-04-03 12:53:09 -05:00
|
|
|
DownloadCount: 10,
|
|
|
|
LastUpdated: time.Time{},
|
|
|
|
IsSigned: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
repoLastUpdated, noImageChecked, isSigned = common.CheckImageLastUpdated(repoLastUpdated, isSigned, noImageChecked,
|
|
|
|
manifestFilterData)
|
|
|
|
So(repoLastUpdated, ShouldResemble, manifestFilterData.LastUpdated)
|
|
|
|
So(isSigned, ShouldEqual, manifestFilterData.IsSigned)
|
|
|
|
So(noImageChecked, ShouldEqual, false)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("First image checked, it has time", func() {
|
|
|
|
repoLastUpdated := time.Time{}
|
|
|
|
isSigned := false
|
|
|
|
noImageChecked := true
|
2023-07-18 12:27:26 -05:00
|
|
|
manifestFilterData := mTypes.FilterData{
|
2023-04-03 12:53:09 -05:00
|
|
|
DownloadCount: 10,
|
|
|
|
LastUpdated: time.Date(2000, 1, 1, 1, 1, 1, 1, time.UTC),
|
|
|
|
IsSigned: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
repoLastUpdated, noImageChecked, isSigned = common.CheckImageLastUpdated(repoLastUpdated, isSigned, noImageChecked,
|
|
|
|
manifestFilterData)
|
|
|
|
So(repoLastUpdated, ShouldResemble, manifestFilterData.LastUpdated)
|
|
|
|
So(isSigned, ShouldEqual, manifestFilterData.IsSigned)
|
|
|
|
So(noImageChecked, ShouldEqual, false)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("Not first image checked, current image is newer", func() {
|
|
|
|
repoLastUpdated := time.Date(2000, 1, 1, 1, 1, 1, 1, time.UTC)
|
|
|
|
isSigned := true
|
|
|
|
noImageChecked := false
|
2023-07-18 12:27:26 -05:00
|
|
|
manifestFilterData := mTypes.FilterData{
|
2023-04-03 12:53:09 -05:00
|
|
|
DownloadCount: 10,
|
|
|
|
LastUpdated: time.Date(2023, 1, 1, 1, 1, 1, 1, time.UTC),
|
|
|
|
IsSigned: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
repoLastUpdated, noImageChecked, isSigned = common.CheckImageLastUpdated(repoLastUpdated, isSigned,
|
|
|
|
noImageChecked, manifestFilterData)
|
|
|
|
So(repoLastUpdated, ShouldResemble, manifestFilterData.LastUpdated)
|
|
|
|
So(isSigned, ShouldEqual, manifestFilterData.IsSigned)
|
|
|
|
So(noImageChecked, ShouldEqual, false)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("Not first image checked, current image is older", func() {
|
|
|
|
repoLastUpdated := time.Date(2024, 1, 1, 1, 1, 1, 1, time.UTC)
|
|
|
|
isSigned := false
|
|
|
|
noImageChecked := false
|
2023-07-18 12:27:26 -05:00
|
|
|
manifestFilterData := mTypes.FilterData{
|
2023-04-03 12:53:09 -05:00
|
|
|
DownloadCount: 10,
|
|
|
|
LastUpdated: time.Date(2022, 1, 1, 1, 1, 1, 1, time.UTC),
|
|
|
|
IsSigned: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
updatedRepoLastUpdated, noImageChecked, isSigned := common.CheckImageLastUpdated(repoLastUpdated, isSigned,
|
|
|
|
noImageChecked,
|
|
|
|
manifestFilterData)
|
|
|
|
So(updatedRepoLastUpdated, ShouldResemble, repoLastUpdated)
|
|
|
|
So(isSigned, ShouldEqual, false)
|
|
|
|
So(noImageChecked, ShouldEqual, false)
|
|
|
|
})
|
|
|
|
})
|
2023-04-24 13:13:15 -05:00
|
|
|
|
|
|
|
Convey("SignatureAlreadyExists", t, func() {
|
|
|
|
res := common.SignatureAlreadyExists(
|
2023-07-18 12:27:26 -05:00
|
|
|
[]mTypes.SignatureInfo{{SignatureManifestDigest: "digest"}},
|
|
|
|
mTypes.SignatureMetadata{SignatureDigest: "digest"},
|
2023-04-24 13:13:15 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
So(res, ShouldEqual, true)
|
|
|
|
|
|
|
|
res = common.SignatureAlreadyExists(
|
2023-07-18 12:27:26 -05:00
|
|
|
[]mTypes.SignatureInfo{{SignatureManifestDigest: "digest"}},
|
|
|
|
mTypes.SignatureMetadata{SignatureDigest: "digest2"},
|
2023-04-24 13:13:15 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
So(res, ShouldEqual, false)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("FilterDataByRepo", t, func() {
|
2023-07-31 14:16:09 -05:00
|
|
|
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)
|
|
|
|
})
|
2023-04-24 13:13:15 -05:00
|
|
|
Convey("Errors", func() {
|
|
|
|
// Unmarshal index data error
|
|
|
|
_, _, err := common.FilterDataByRepo(
|
2023-07-18 12:27:26 -05:00
|
|
|
[]mTypes.RepoMetadata{{
|
|
|
|
Tags: map[string]mTypes.Descriptor{
|
2023-04-24 13:13:15 -05:00
|
|
|
"tag": {
|
|
|
|
Digest: "indexDigest",
|
|
|
|
MediaType: ispec.MediaTypeImageIndex,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}},
|
2023-07-18 12:27:26 -05:00
|
|
|
map[string]mTypes.ManifestMetadata{},
|
|
|
|
map[string]mTypes.IndexData{
|
2023-04-24 13:13:15 -05:00
|
|
|
"indexDigest": {
|
|
|
|
IndexBlob: []byte("bad blob"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("FetchDataForRepos", t, func() {
|
|
|
|
Convey("Errors", func() {
|
|
|
|
// Unmarshal index data error
|
|
|
|
_, _, err := common.FetchDataForRepos(
|
2023-07-18 12:27:26 -05:00
|
|
|
mocks.MetaDBMock{
|
|
|
|
GetIndexDataFn: func(indexDigest digest.Digest) (mTypes.IndexData, error) {
|
|
|
|
return mTypes.IndexData{
|
2023-04-24 13:13:15 -05:00
|
|
|
IndexBlob: []byte("bad blob"),
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
2023-07-18 12:27:26 -05:00
|
|
|
[]mTypes.RepoMetadata{{
|
|
|
|
Tags: map[string]mTypes.Descriptor{
|
2023-04-24 13:13:15 -05:00
|
|
|
"tag": {
|
|
|
|
Digest: "indexDigest",
|
|
|
|
MediaType: ispec.MediaTypeImageIndex,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}},
|
|
|
|
)
|
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFetchDataForRepos(t *testing.T) {
|
|
|
|
Convey("GetReferredSubject", t, func() {
|
2023-07-18 12:27:26 -05:00
|
|
|
mockMetaDB := mocks.MetaDBMock{}
|
2023-04-24 13:13:15 -05:00
|
|
|
|
|
|
|
Convey("GetManifestData errors", func() {
|
2023-07-18 12:27:26 -05:00
|
|
|
mockMetaDB.GetManifestDataFn = func(manifestDigest digest.Digest) (mTypes.ManifestData, error) {
|
|
|
|
return mTypes.ManifestData{}, ErrTestError
|
2023-04-24 13:13:15 -05:00
|
|
|
}
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
_, _, err := common.FetchDataForRepos(mockMetaDB, []mTypes.RepoMetadata{
|
2023-04-24 13:13:15 -05:00
|
|
|
{
|
2023-07-18 12:27:26 -05:00
|
|
|
Tags: map[string]mTypes.Descriptor{
|
2023-04-24 13:13:15 -05:00
|
|
|
"tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageManifest},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("GetIndexData errors", func() {
|
2023-07-18 12:27:26 -05:00
|
|
|
mockMetaDB.GetIndexDataFn = func(indexDigest digest.Digest) (mTypes.IndexData, error) {
|
|
|
|
return mTypes.IndexData{}, ErrTestError
|
2023-04-24 13:13:15 -05:00
|
|
|
}
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
_, _, err := common.FetchDataForRepos(mockMetaDB, []mTypes.RepoMetadata{
|
2023-04-24 13:13:15 -05:00
|
|
|
{
|
2023-07-18 12:27:26 -05:00
|
|
|
Tags: map[string]mTypes.Descriptor{
|
2023-04-24 13:13:15 -05:00
|
|
|
"tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageIndex},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
|
|
|
|
Convey("GetIndexData ok, GetManifestData errors", func() {
|
2023-07-18 12:27:26 -05:00
|
|
|
mockMetaDB.GetIndexDataFn = func(indexDigest digest.Digest) (mTypes.IndexData, error) {
|
|
|
|
return mTypes.IndexData{
|
2023-04-24 13:13:15 -05:00
|
|
|
IndexBlob: []byte(`{
|
|
|
|
"manifests": [
|
|
|
|
{"digest": "dig1"}
|
|
|
|
]
|
|
|
|
}`),
|
|
|
|
}, nil
|
|
|
|
}
|
2023-07-18 12:27:26 -05:00
|
|
|
mockMetaDB.GetManifestDataFn = func(manifestDigest digest.Digest) (mTypes.ManifestData, error) {
|
|
|
|
return mTypes.ManifestData{}, ErrTestError
|
2023-04-24 13:13:15 -05:00
|
|
|
}
|
|
|
|
|
2023-07-18 12:27:26 -05:00
|
|
|
_, _, err := common.FetchDataForRepos(mockMetaDB, []mTypes.RepoMetadata{
|
2023-04-24 13:13:15 -05:00
|
|
|
{
|
2023-07-18 12:27:26 -05:00
|
|
|
Tags: map[string]mTypes.Descriptor{
|
2023-04-24 13:13:15 -05:00
|
|
|
"tag1": {Digest: "dig1", MediaType: ispec.MediaTypeImageIndex},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
})
|
2023-03-10 13:37:29 -05:00
|
|
|
}
|