mirror of
https://github.com/project-zot/zot.git
synced 2025-01-20 22:52:51 -05:00
bcdd9988f5
1. Only scan CVEs for images returned by graphql calls Since pagination was refactored to account for image indexes, we had started to run the CVE scanner before pagination was applied, resulting in decreased ZOT performance if CVE information was requested 2. Increase in medory-cache of cve results to 1m, from 10k digests. 3. Update CVE model to use CVSS severity values in our code. Previously we relied upon the strings returned by trivy directly, and the sorting they implemented. Since CVE severities are standardized, we don't need to pass around an adapter object just for pagination and sorting purposes anymore. This also improves our testing since we don't mock the sorting functions anymore. 4. Fix a flaky CLI test not waiting for the zot service to start. 5. Add the search build label on search/cve tests which were missing it. 6. The boltdb update method was used in a few places where view was supposed to be called. 7. Add logs for start and finish of parsing MetaDB. 8. Avoid unmarshalling twice to obtain annotations for multiarch images. Signed-off-by: Andrei Aaron <aaaron@luxoft.com>
481 lines
14 KiB
Go
481 lines
14 KiB
Go
package meta
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
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"
|
|
"zotregistry.io/zot/pkg/log"
|
|
mTypes "zotregistry.io/zot/pkg/meta/types"
|
|
"zotregistry.io/zot/pkg/storage"
|
|
storageTypes "zotregistry.io/zot/pkg/storage/types"
|
|
)
|
|
|
|
// 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 {
|
|
log.Info().Msg("Started parsing storage and updating MetaDB")
|
|
|
|
allRepos, err := getAllRepos(storeController)
|
|
if err != nil {
|
|
rootDir := storeController.DefaultStore.RootDir()
|
|
log.Error().Err(err).Str("rootDir", rootDir).
|
|
Msg("load-local-layout: failed to get all repo names present under rootDir")
|
|
|
|
return err
|
|
}
|
|
|
|
for _, repo := range allRepos {
|
|
err := ParseRepo(repo, metaDB, storeController, log)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Msg("load-local-layout: failed to sync repo")
|
|
|
|
return err
|
|
}
|
|
}
|
|
|
|
log.Info().Msg("Done parsing storage and updating MetaDB")
|
|
|
|
return nil
|
|
}
|
|
|
|
// ParseRepo reads the contents of a repo and syncs all images and signatures found.
|
|
func ParseRepo(repo string, metaDB mTypes.MetaDB, storeController storage.StoreController, log log.Logger) error {
|
|
imageStore := storeController.GetImageStore(repo)
|
|
|
|
var lockLatency time.Time
|
|
|
|
imageStore.RLock(&lockLatency)
|
|
defer imageStore.RUnlock(&lockLatency)
|
|
|
|
indexBlob, err := imageStore.GetIndexContent(repo)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Msg("load-repo: failed to read index.json for repo")
|
|
|
|
return err
|
|
}
|
|
|
|
var indexContent ispec.Index
|
|
|
|
err = json.Unmarshal(indexBlob, &indexContent)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Msg("load-repo: failed to unmarshal index.json for repo")
|
|
|
|
return err
|
|
}
|
|
|
|
err = resetRepoMeta(repo, metaDB, log)
|
|
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]
|
|
|
|
descriptorBlob, err := getCachedBlob(repo, descriptor, metaDB, imageStore, log)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("load-repo: error checking manifestMeta in MetaDB")
|
|
|
|
return err
|
|
}
|
|
|
|
isSignature, signatureType, signedManifestDigest, err := storage.CheckIsImageSignature(repo,
|
|
descriptorBlob, tag)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("tag", tag).
|
|
Msg("load-repo: failed checking if image is signature for specified image")
|
|
|
|
return err
|
|
}
|
|
|
|
if isSignature {
|
|
layers, err := GetSignatureLayersInfo(repo, tag, descriptor.Digest.String(), signatureType,
|
|
descriptorBlob, imageStore, log)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = metaDB.AddManifestSignature(repo, signedManifestDigest,
|
|
mTypes.SignatureMetadata{
|
|
SignatureType: signatureType,
|
|
SignatureDigest: descriptor.Digest.String(),
|
|
LayersInfo: layers,
|
|
})
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("tag", tag).
|
|
Str("manifestDigest", signedManifestDigest.String()).
|
|
Msg("load-repo: failed set signature meta for signed image")
|
|
|
|
return err
|
|
}
|
|
|
|
err = metaDB.UpdateSignaturesValidity(repo, signedManifestDigest)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("reference", tag).Str("digest", signedManifestDigest.String()).Msg(
|
|
"load-repo: failed verify signatures validity for signed image")
|
|
|
|
return err
|
|
}
|
|
|
|
continue
|
|
}
|
|
|
|
reference := tag
|
|
|
|
if tag == "" {
|
|
reference = descriptor.Digest.String()
|
|
}
|
|
|
|
err = SetImageMetaFromInput(repo, reference, descriptor.MediaType, descriptor.Digest, descriptorBlob,
|
|
imageStore, metaDB, log)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("tag", tag).
|
|
Msg("load-repo: failed to set metadata for image")
|
|
|
|
return err
|
|
}
|
|
}
|
|
|
|
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.GetRepoMeta(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.SetRepoMeta(repo, mTypes.RepoMetadata{
|
|
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 {
|
|
return nil, err
|
|
}
|
|
|
|
if storeController.SubStore != nil {
|
|
for _, store := range storeController.SubStore {
|
|
substoreRepos, err := store.GetRepositories()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
allRepos = append(allRepos, substoreRepos...)
|
|
}
|
|
}
|
|
|
|
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) {
|
|
switch signatureType {
|
|
case zcommon.CosignSignature:
|
|
return getCosignSignatureLayersInfo(repo, tag, manifestDigest, manifestBlob, imageStore, log)
|
|
case zcommon.NotationSignature:
|
|
return getNotationSignatureLayersInfo(repo, manifestDigest, manifestBlob, imageStore, log)
|
|
default:
|
|
return []mTypes.LayerInfo{}, nil
|
|
}
|
|
}
|
|
|
|
func getCosignSignatureLayersInfo(
|
|
repo, tag, manifestDigest string, manifestBlob []byte, imageStore storageTypes.ImageStore, log log.Logger,
|
|
) ([]mTypes.LayerInfo, error) {
|
|
layers := []mTypes.LayerInfo{}
|
|
|
|
var manifestContent ispec.Manifest
|
|
if err := json.Unmarshal(manifestBlob, &manifestContent); err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("reference", tag).Str("digest", manifestDigest).Msg(
|
|
"load-repo: unable to marshal blob index")
|
|
|
|
return layers, err
|
|
}
|
|
|
|
var lockLatency time.Time
|
|
|
|
imageStore.RLock(&lockLatency)
|
|
defer imageStore.RUnlock(&lockLatency)
|
|
|
|
for _, layer := range manifestContent.Layers {
|
|
layerContent, err := imageStore.GetBlobContent(repo, layer.Digest)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("reference", tag).Str("layerDigest", layer.Digest.String()).Msg(
|
|
"load-repo: unable to get cosign signature layer content")
|
|
|
|
return layers, err
|
|
}
|
|
|
|
layerSigKey, ok := layer.Annotations[zcommon.CosignSigKey]
|
|
if !ok {
|
|
log.Error().Err(err).Str("repository", repo).Str("reference", tag).Str("layerDigest", layer.Digest.String()).Msg(
|
|
"load-repo: unable to get specific annotation of cosign signature")
|
|
}
|
|
|
|
layers = append(layers, mTypes.LayerInfo{
|
|
LayerDigest: layer.Digest.String(),
|
|
LayerContent: layerContent,
|
|
SignatureKey: layerSigKey,
|
|
})
|
|
}
|
|
|
|
return layers, nil
|
|
}
|
|
|
|
func getNotationSignatureLayersInfo(
|
|
repo, manifestDigest string, manifestBlob []byte, imageStore storageTypes.ImageStore, log log.Logger,
|
|
) ([]mTypes.LayerInfo, error) {
|
|
layers := []mTypes.LayerInfo{}
|
|
|
|
var manifestContent ispec.Manifest
|
|
if err := json.Unmarshal(manifestBlob, &manifestContent); err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("reference", manifestDigest).Msg(
|
|
"load-repo: unable to marshal blob index")
|
|
|
|
return layers, err
|
|
}
|
|
|
|
if len(manifestContent.Layers) != 1 {
|
|
log.Error().Err(zerr.ErrBadManifest).Str("repository", repo).Str("reference", manifestDigest).
|
|
Msg("load-repo: notation signature manifest requires exactly one layer but it does not")
|
|
|
|
return layers, zerr.ErrBadManifest
|
|
}
|
|
|
|
layer := manifestContent.Layers[0].Digest
|
|
|
|
var lockLatency time.Time
|
|
|
|
imageStore.RLock(&lockLatency)
|
|
defer imageStore.RUnlock(&lockLatency)
|
|
|
|
layerContent, err := imageStore.GetBlobContent(repo, layer)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("repository", repo).Str("reference", manifestDigest).Str("layerDigest", layer.String()).Msg(
|
|
"load-repo: unable to get notation signature blob content")
|
|
|
|
return layers, err
|
|
}
|
|
|
|
layerSigKey := manifestContent.Layers[0].MediaType
|
|
|
|
layers = append(layers, mTypes.LayerInfo{
|
|
LayerDigest: layer.String(),
|
|
LayerContent: layerContent,
|
|
SignatureKey: layerSigKey,
|
|
})
|
|
|
|
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
|
|
}
|
|
|
|
func GetReferredInfo(descriptorBlob []byte, referrerDigest, mediaType string,
|
|
) (godigest.Digest, mTypes.ReferrerInfo, bool, error) {
|
|
var (
|
|
referrerInfo mTypes.ReferrerInfo
|
|
referrerSubject *ispec.Descriptor
|
|
)
|
|
|
|
switch mediaType {
|
|
case ispec.MediaTypeImageManifest:
|
|
var manifestContent ispec.Manifest
|
|
|
|
err := json.Unmarshal(descriptorBlob, &manifestContent)
|
|
if err != nil {
|
|
return "", referrerInfo, false,
|
|
fmt.Errorf("metadb: can't unmarshal manifest for digest %s: %w", referrerDigest, err)
|
|
}
|
|
|
|
referrerSubject = manifestContent.Subject
|
|
|
|
referrerInfo = mTypes.ReferrerInfo{
|
|
Digest: referrerDigest,
|
|
MediaType: mediaType,
|
|
ArtifactType: zcommon.GetManifestArtifactType(manifestContent),
|
|
Size: len(descriptorBlob),
|
|
Annotations: manifestContent.Annotations,
|
|
}
|
|
case ispec.MediaTypeImageIndex:
|
|
var indexContent ispec.Index
|
|
|
|
err := json.Unmarshal(descriptorBlob, &indexContent)
|
|
if err != nil {
|
|
return "", referrerInfo, false,
|
|
fmt.Errorf("metadb: can't unmarshal manifest for digest %s: %w", referrerDigest, err)
|
|
}
|
|
|
|
referrerSubject = indexContent.Subject
|
|
|
|
referrerInfo = mTypes.ReferrerInfo{
|
|
Digest: referrerDigest,
|
|
MediaType: mediaType,
|
|
ArtifactType: zcommon.GetIndexArtifactType(indexContent),
|
|
Size: len(descriptorBlob),
|
|
Annotations: indexContent.Annotations,
|
|
}
|
|
}
|
|
|
|
if referrerSubject == nil || referrerSubject.Digest.String() == "" {
|
|
return "", mTypes.ReferrerInfo{}, false, nil
|
|
}
|
|
|
|
return referrerSubject.Digest, referrerInfo, true, nil
|
|
}
|