2023-07-06 11:36:26 +03:00
|
|
|
package trivy_test
|
|
|
|
|
|
|
|
import (
|
2023-08-18 11:46:11 +03:00
|
|
|
"path/filepath"
|
2023-07-06 11:36:26 +03:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
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/api"
|
|
|
|
"zotregistry.io/zot/pkg/api/config"
|
|
|
|
extconf "zotregistry.io/zot/pkg/extensions/config"
|
|
|
|
"zotregistry.io/zot/pkg/extensions/monitoring"
|
|
|
|
"zotregistry.io/zot/pkg/extensions/search/cve/trivy"
|
|
|
|
"zotregistry.io/zot/pkg/log"
|
2023-07-18 20:27:26 +03:00
|
|
|
"zotregistry.io/zot/pkg/meta"
|
|
|
|
"zotregistry.io/zot/pkg/meta/boltdb"
|
|
|
|
mTypes "zotregistry.io/zot/pkg/meta/types"
|
2023-07-06 11:36:26 +03:00
|
|
|
"zotregistry.io/zot/pkg/storage"
|
|
|
|
"zotregistry.io/zot/pkg/storage/local"
|
|
|
|
"zotregistry.io/zot/pkg/test"
|
|
|
|
"zotregistry.io/zot/pkg/test/mocks"
|
|
|
|
)
|
|
|
|
|
2023-08-18 11:46:11 +03:00
|
|
|
func TestScanBigTestFile(t *testing.T) {
|
|
|
|
Convey("Scan zot-test", t, func() {
|
|
|
|
projRootDir, err := test.GetProjectRootDir()
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
testImage := filepath.Join(projRootDir, "test/data/zot-test")
|
|
|
|
|
|
|
|
tempDir := t.TempDir()
|
|
|
|
port := test.GetFreePort()
|
|
|
|
conf := config.New()
|
|
|
|
conf.HTTP.Port = port
|
|
|
|
defaultVal := true
|
|
|
|
conf.Storage.RootDirectory = tempDir
|
|
|
|
conf.Extensions = &extconf.ExtensionConfig{
|
|
|
|
Search: &extconf.SearchConfig{
|
|
|
|
BaseConfig: extconf.BaseConfig{Enable: &defaultVal},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
ctlr := api.NewController(conf)
|
|
|
|
So(ctlr, ShouldNotBeNil)
|
|
|
|
|
|
|
|
err = test.CopyFiles(testImage, filepath.Join(tempDir, "zot-test"))
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
cm := test.NewControllerManager(ctlr)
|
|
|
|
cm.StartAndWait(port)
|
|
|
|
defer cm.StopServer()
|
|
|
|
// scan
|
|
|
|
scanner := trivy.NewScanner(ctlr.StoreController, ctlr.MetaDB, "ghcr.io/project-zot/trivy-db", "", ctlr.Log)
|
|
|
|
|
|
|
|
err = scanner.UpdateDB()
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
cveMap, err := scanner.ScanImage("zot-test:0.0.1")
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
So(cveMap, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-07-06 11:36:26 +03:00
|
|
|
func TestScanningByDigest(t *testing.T) {
|
|
|
|
Convey("Scan the individual manifests inside an index", t, func() {
|
|
|
|
// start server
|
|
|
|
tempDir := t.TempDir()
|
|
|
|
port := test.GetFreePort()
|
|
|
|
baseURL := test.GetBaseURL(port)
|
|
|
|
conf := config.New()
|
|
|
|
conf.HTTP.Port = port
|
|
|
|
defaultVal := true
|
|
|
|
conf.Storage.RootDirectory = tempDir
|
|
|
|
conf.Extensions = &extconf.ExtensionConfig{
|
|
|
|
Search: &extconf.SearchConfig{
|
|
|
|
BaseConfig: extconf.BaseConfig{Enable: &defaultVal},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
ctlr := api.NewController(conf)
|
|
|
|
So(ctlr, ShouldNotBeNil)
|
|
|
|
|
|
|
|
cm := test.NewControllerManager(ctlr)
|
|
|
|
cm.StartAndWait(port)
|
|
|
|
defer cm.StopServer()
|
|
|
|
// push index with 2 manifests: one with vulns and one without
|
2023-08-18 11:46:11 +03:00
|
|
|
vulnImage := test.CreateDefaultVulnerableImage()
|
2023-07-06 11:36:26 +03:00
|
|
|
|
2023-07-26 13:08:04 +03:00
|
|
|
simpleImage := test.CreateRandomImage()
|
2023-07-06 11:36:26 +03:00
|
|
|
|
2023-07-28 17:53:46 +03:00
|
|
|
multiArch := test.GetMultiarchImageForImages([]test.Image{simpleImage, vulnImage}) //nolint:staticcheck
|
2023-07-06 11:36:26 +03:00
|
|
|
|
2023-07-28 17:53:46 +03:00
|
|
|
err := test.UploadMultiarchImage(multiArch, baseURL, "multi-arch", "multi-arch-tag")
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
// scan
|
2023-07-18 20:27:26 +03:00
|
|
|
scanner := trivy.NewScanner(ctlr.StoreController, ctlr.MetaDB, "ghcr.io/project-zot/trivy-db", "", ctlr.Log)
|
2023-07-06 11:36:26 +03:00
|
|
|
|
|
|
|
err = scanner.UpdateDB()
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-26 13:08:04 +03:00
|
|
|
cveMap, err := scanner.ScanImage("multi-arch@" + vulnImage.DigestStr())
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability1ID)
|
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability2ID)
|
2023-07-16 18:27:59 +03:00
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability3ID)
|
2023-07-06 11:36:26 +03:00
|
|
|
|
2023-07-26 13:08:04 +03:00
|
|
|
cveMap, err = scanner.ScanImage("multi-arch@" + simpleImage.DigestStr())
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
So(cveMap, ShouldBeEmpty)
|
|
|
|
|
2023-07-26 13:08:04 +03:00
|
|
|
cveMap, err = scanner.ScanImage("multi-arch@" + multiArch.DigestStr())
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability1ID)
|
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability2ID)
|
2023-07-16 18:27:59 +03:00
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability3ID)
|
2023-07-06 11:36:26 +03:00
|
|
|
|
|
|
|
cveMap, err = scanner.ScanImage("multi-arch:multi-arch-tag")
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability1ID)
|
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability2ID)
|
2023-07-16 18:27:59 +03:00
|
|
|
So(cveMap, ShouldContainKey, test.Vulnerability3ID)
|
2023-07-06 11:36:26 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestScannerErrors(t *testing.T) {
|
|
|
|
digest := godigest.FromString("dig")
|
|
|
|
|
|
|
|
Convey("Errors", t, func() {
|
|
|
|
storeController := storage.StoreController{}
|
|
|
|
storeController.DefaultStore = mocks.MockedImageStore{}
|
|
|
|
|
2023-07-18 20:27:26 +03:00
|
|
|
metaDB := mocks.MetaDBMock{}
|
2023-07-06 11:36:26 +03:00
|
|
|
log := log.NewLogger("debug", "")
|
|
|
|
|
|
|
|
Convey("IsImageFormatSanable", func() {
|
2023-07-18 20:27:26 +03:00
|
|
|
metaDB.GetManifestDataFn = func(manifestDigest godigest.Digest) (mTypes.ManifestData, error) {
|
|
|
|
return mTypes.ManifestData{}, zerr.ErrManifestDataNotFound
|
2023-07-06 11:36:26 +03:00
|
|
|
}
|
2023-07-18 20:27:26 +03:00
|
|
|
metaDB.GetIndexDataFn = func(indexDigest godigest.Digest) (mTypes.IndexData, error) {
|
|
|
|
return mTypes.IndexData{}, zerr.ErrManifestDataNotFound
|
2023-07-06 11:36:26 +03:00
|
|
|
}
|
2023-07-18 20:27:26 +03:00
|
|
|
scanner := trivy.NewScanner(storeController, metaDB, "", "", log)
|
2023-07-06 11:36:26 +03:00
|
|
|
|
|
|
|
_, err := scanner.ScanImage("repo@" + digest.String())
|
|
|
|
So(err, ShouldNotBeNil)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestVulnerableLayer(t *testing.T) {
|
|
|
|
Convey("Vulnerable layer", t, func() {
|
2023-07-26 13:08:04 +03:00
|
|
|
vulnerableLayer, err := test.GetLayerWithVulnerability()
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
created, err := time.Parse(time.RFC3339, "2023-03-29T18:19:24Z")
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
|
|
|
config := ispec.Image{
|
|
|
|
Created: &created,
|
|
|
|
Platform: ispec.Platform{
|
|
|
|
Architecture: "amd64",
|
|
|
|
OS: "linux",
|
|
|
|
},
|
|
|
|
Config: ispec.ImageConfig{
|
|
|
|
Env: []string{"PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin"},
|
|
|
|
Cmd: []string{"/bin/sh"},
|
|
|
|
},
|
|
|
|
RootFS: ispec.RootFS{
|
|
|
|
Type: "layers",
|
|
|
|
DiffIDs: []godigest.Digest{"sha256:f1417ff83b319fbdae6dd9cd6d8c9c88002dcd75ecf6ec201c8c6894681cf2b5"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2023-07-26 13:08:04 +03:00
|
|
|
img := test.CreateImageWith().
|
|
|
|
LayerBlobs([][]byte{vulnerableLayer}).
|
|
|
|
ImageConfig(config).
|
|
|
|
Build()
|
2023-07-06 11:36:26 +03:00
|
|
|
|
|
|
|
tempDir := t.TempDir()
|
|
|
|
|
|
|
|
log := log.NewLogger("debug", "")
|
2023-09-01 20:54:39 +03:00
|
|
|
imageStore := local.NewImageStore(tempDir, false, false, 0, 0, false, false,
|
2023-07-06 11:36:26 +03:00
|
|
|
log, monitoring.NewMetricsServer(false, log), nil, nil)
|
|
|
|
|
|
|
|
storeController := storage.StoreController{
|
|
|
|
DefaultStore: imageStore,
|
|
|
|
}
|
|
|
|
|
2023-07-28 17:53:46 +03:00
|
|
|
err = test.WriteImageToFileSystem(img, "repo", img.DigestStr(), storeController)
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-18 20:27:26 +03:00
|
|
|
params := boltdb.DBParameters{
|
2023-07-06 11:36:26 +03:00
|
|
|
RootDir: tempDir,
|
|
|
|
}
|
2023-07-18 20:27:26 +03:00
|
|
|
boltDriver, err := boltdb.GetBoltDriver(params)
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-18 20:27:26 +03:00
|
|
|
metaDB, err := boltdb.New(boltDriver, log)
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-18 20:27:26 +03:00
|
|
|
err = meta.ParseStorage(metaDB, storeController, log)
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-18 20:27:26 +03:00
|
|
|
scanner := trivy.NewScanner(storeController, metaDB, "ghcr.io/project-zot/trivy-db", "", log)
|
2023-07-06 11:36:26 +03:00
|
|
|
|
|
|
|
err = scanner.UpdateDB()
|
|
|
|
So(err, ShouldBeNil)
|
|
|
|
|
2023-07-26 13:08:04 +03:00
|
|
|
cveMap, err := scanner.ScanImage("repo@" + img.DigestStr())
|
2023-07-06 11:36:26 +03:00
|
|
|
So(err, ShouldBeNil)
|
2023-07-16 18:27:59 +03:00
|
|
|
t.Logf("cveMap: %v", cveMap)
|
|
|
|
// As of July 15 2023 there are 3 CVEs: CVE-2023-1255, CVE-2023-2650, CVE-2023-2975
|
|
|
|
// There may be more discovered in the future
|
|
|
|
So(len(cveMap), ShouldBeGreaterThanOrEqualTo, 3)
|
2023-07-06 11:36:26 +03:00
|
|
|
})
|
|
|
|
}
|