0
Fork 0
mirror of https://github.com/project-zot/zot.git synced 2024-12-30 22:34:13 -05:00

fix(repodb): GQL request for ExpandedRepoInfo errors when artifacts with tags are present (#1265)

If we push an artifact and give it a tag, repodb would crash because of the null pointer dereferencing

Now when iterating over the tags of a repo and stumbling upon a unsupported media type, it's being ignored

Signed-off-by: Laurentiu Niculae <niculae.laurentiu1@gmail.com>
This commit is contained in:
LaurentiuNiculae 2023-03-15 19:34:48 +02:00 committed by GitHub
parent 7656b6f011
commit 150ee88945
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 89 additions and 6 deletions

View file

@ -78,6 +78,7 @@ var (
ErrLimitIsNegative = errors.New("pageturner: limit has negative value") ErrLimitIsNegative = errors.New("pageturner: limit has negative value")
ErrOffsetIsNegative = errors.New("pageturner: offset has negative value") ErrOffsetIsNegative = errors.New("pageturner: offset has negative value")
ErrSortCriteriaNotSupported = errors.New("pageturner: the sort criteria is not supported") ErrSortCriteriaNotSupported = errors.New("pageturner: the sort criteria is not supported")
ErrMediaTypeNotSupported = errors.New("repodb: media type is not supported")
ErrTimeout = errors.New("operation timeout") ErrTimeout = errors.New("operation timeout")
ErrNotImplemented = errors.New("not implemented") ErrNotImplemented = errors.New("not implemented")
) )

View file

@ -1018,7 +1018,7 @@ func TestGetReferrersGQL(t *testing.T) {
So(err, ShouldBeNil) So(err, ShouldBeNil)
artifactManifestDigest := godigest.FromBytes(artifactManifestBlob) artifactManifestDigest := godigest.FromBytes(artifactManifestBlob)
err = UploadArtifactManifest(artifact, baseURL, repo) err = UploadArtifactManifest(artifact, nil, baseURL, repo)
So(err, ShouldBeNil) So(err, ShouldBeNil)
gqlQuery := ` gqlQuery := `
@ -1401,6 +1401,72 @@ func TestExpandedRepoInfo(t *testing.T) {
So(err, ShouldBeNil) So(err, ShouldBeNil)
}) })
Convey("Test expanded repo info with tagged referrers", t, func() {
rootDir := t.TempDir()
port := GetFreePort()
baseURL := GetBaseURL(port)
conf := config.New()
conf.HTTP.Port = port
conf.Storage.RootDirectory = rootDir
conf.Storage.GC = false
defaultVal := true
conf.Extensions = &extconf.ExtensionConfig{
Search: &extconf.SearchConfig{BaseConfig: extconf.BaseConfig{Enable: &defaultVal}},
}
conf.Extensions.Search.CVE = nil
ctlr := api.NewController(conf)
ctlrManager := NewControllerManager(ctlr)
ctlrManager.StartAndWait(port)
defer ctlrManager.StopServer()
image, err := GetRandomImage("test")
So(err, ShouldBeNil)
manifestDigest, err := image.Digest()
So(err, ShouldBeNil)
err = UploadImage(image, baseURL, "repo")
So(err, ShouldBeNil)
referrer, err := GetRandomArtifact(&ispec.Descriptor{
Digest: manifestDigest,
MediaType: ispec.MediaTypeImageManifest,
})
So(err, ShouldBeNil)
tag := "test-ref-tag"
err = UploadArtifactManifest(&referrer.Manifest, &tag, baseURL, "repo")
So(err, ShouldBeNil)
// ------- Make the call to GQL and see that it doesn't crash and that the referrer isn't in the list of tags
responseStruct := &ExpandedRepoInfoResp{}
query := `
{
ExpandedRepoInfo(repo:"repo"){
Images {
RepoName
Tag
Manifests {
Digest
Layers {Size Digest}
}
}
}
}`
resp, err := resty.R().Get(baseURL + graphqlQueryPrefix + "?query=" + url.QueryEscape(query))
So(resp, ShouldNotBeNil)
So(err, ShouldBeNil)
So(resp.StatusCode(), ShouldEqual, 200)
err = json.Unmarshal(resp.Body(), responseStruct)
So(err, ShouldBeNil)
So(len(responseStruct.ExpandedRepoInfo.RepoInfo.ImageSummaries), ShouldEqual, 1)
repoInfo := responseStruct.ExpandedRepoInfo.RepoInfo
So(repoInfo.ImageSummaries[0].Tag, ShouldEqual, "test")
})
Convey("Test image tags order", t, func() { Convey("Test image tags order", t, func() {
port := GetFreePort() port := GetFreePort()
baseURL := GetBaseURL(port) baseURL := GetBaseURL(port)

View file

@ -13,6 +13,7 @@ import (
ispec "github.com/opencontainers/image-spec/specs-go/v1" ispec "github.com/opencontainers/image-spec/specs-go/v1"
"github.com/vektah/gqlparser/v2/gqlerror" "github.com/vektah/gqlparser/v2/gqlerror"
zerr "zotregistry.io/zot/errors"
"zotregistry.io/zot/pkg/extensions/search/common" "zotregistry.io/zot/pkg/extensions/search/common"
cveinfo "zotregistry.io/zot/pkg/extensions/search/cve" cveinfo "zotregistry.io/zot/pkg/extensions/search/cve"
cvemodel "zotregistry.io/zot/pkg/extensions/search/cve/model" cvemodel "zotregistry.io/zot/pkg/extensions/search/cve/model"
@ -164,7 +165,7 @@ func Descriptor2ImageSummary(ctx context.Context, descriptor repodb.Descriptor,
return ImageIndex2ImageSummary(ctx, repo, tag, godigest.Digest(descriptor.Digest), skipCVE, return ImageIndex2ImageSummary(ctx, repo, tag, godigest.Digest(descriptor.Digest), skipCVE,
repoMeta, indexDataMap[descriptor.Digest], manifestMetaMap, cveInfo) repoMeta, indexDataMap[descriptor.Digest], manifestMetaMap, cveInfo)
default: default:
return &gql_generated.ImageSummary{}, map[string]int64{}, nil return &gql_generated.ImageSummary{}, map[string]int64{}, zerr.ErrMediaTypeNotSupported
} }
} }

View file

@ -1091,6 +1091,8 @@ func (bdw *DBWrapper) SearchRepos(ctx context.Context, searchText string, filter
indexDataMap[indexDigest] = indexData indexDataMap[indexDigest] = indexData
default: default:
bdw.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType) bdw.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType)
continue
} }
} }
@ -1416,6 +1418,8 @@ func (bdw *DBWrapper) FilterTags(ctx context.Context, filter repodb.FilterFunc,
indexDataMap[indexDigest] = indexData indexDataMap[indexDigest] = indexData
default: default:
bdw.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType) bdw.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType)
continue
} }
} }
@ -1603,6 +1607,8 @@ func (bdw *DBWrapper) SearchTags(ctx context.Context, searchText string, filter
indexDataMap[indexDigest] = indexData indexDataMap[indexDigest] = indexData
default: default:
bdw.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType) bdw.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType)
continue
} }
} }

View file

@ -972,6 +972,8 @@ func (dwr *DBWrapper) SearchRepos(ctx context.Context, searchText string, filter
indexDataMap[indexDigest] = indexData indexDataMap[indexDigest] = indexData
default: default:
dwr.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType) dwr.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType)
continue
} }
} }
@ -1245,6 +1247,8 @@ func (dwr *DBWrapper) FilterTags(ctx context.Context, filter repodb.FilterFunc,
indexDataMap[indexDigest] = indexData indexDataMap[indexDigest] = indexData
default: default:
dwr.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType) dwr.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType)
continue
} }
} }
@ -1409,6 +1413,8 @@ func (dwr *DBWrapper) SearchTags(ctx context.Context, searchText string, filter
indexDataMap[indexDigest] = indexData indexDataMap[indexDigest] = indexData
default: default:
dwr.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType) dwr.Log.Error().Msgf("Unsupported type: %s", descriptor.MediaType)
continue
} }
} }

View file

@ -958,19 +958,22 @@ func DeleteImage(repo, reference, baseURL string) (int, error) {
} }
// UploadArtifactManifest is used in tests where we don't need to upload the blobs of the artifact. // UploadArtifactManifest is used in tests where we don't need to upload the blobs of the artifact.
func UploadArtifactManifest(artifactManifest *ispec.Artifact, baseURL, repo string) error { func UploadArtifactManifest(artifactManifest *ispec.Artifact, ref *string, baseURL, repo string) error {
// put manifest // put manifest
artifactManifestBlob, err := json.Marshal(artifactManifest) artifactManifestBlob, err := json.Marshal(artifactManifest)
if err != nil { if err != nil {
return err return err
} }
reference := godigest.FromBytes(artifactManifestBlob).String()
artifactManifestDigest := godigest.FromBytes(artifactManifestBlob) if ref != nil {
reference = *ref
}
_, err = resty.R(). _, err = resty.R().
SetHeader("Content-type", ispec.MediaTypeArtifactManifest). SetHeader("Content-type", ispec.MediaTypeArtifactManifest).
SetBody(artifactManifestBlob). SetBody(artifactManifestBlob).
Put(baseURL + "/v2/" + repo + "/manifests/" + artifactManifestDigest.String()) Put(baseURL + "/v2/" + repo + "/manifests/" + reference)
return err return err
} }

View file

@ -241,7 +241,7 @@ func TestUploadArtifact(t *testing.T) {
artifact := ispec.Artifact{} artifact := ispec.Artifact{}
err := test.UploadArtifactManifest(&artifact, baseURL, "test") err := test.UploadArtifactManifest(&artifact, nil, baseURL, "test")
So(err, ShouldNotBeNil) So(err, ShouldNotBeNil)
}) })
} }