mirror of
https://github.com/project-zot/zot.git
synced 2025-01-27 23:01:43 -05:00
440 lines
12 KiB
Go
440 lines
12 KiB
Go
|
package signatures_test
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"encoding/json"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"os"
|
||
|
"path"
|
||
|
"testing"
|
||
|
"time"
|
||
|
|
||
|
"github.com/notaryproject/notation-go"
|
||
|
"github.com/notaryproject/notation-go/verifier/trustpolicy"
|
||
|
ispec "github.com/opencontainers/image-spec/specs-go/v1"
|
||
|
"github.com/sigstore/cosign/v2/cmd/cosign/cli/generate"
|
||
|
"github.com/sigstore/cosign/v2/cmd/cosign/cli/options"
|
||
|
"github.com/sigstore/cosign/v2/cmd/cosign/cli/sign"
|
||
|
. "github.com/smartystreets/goconvey/convey"
|
||
|
|
||
|
zerr "zotregistry.io/zot/errors"
|
||
|
"zotregistry.io/zot/pkg/api"
|
||
|
"zotregistry.io/zot/pkg/api/config"
|
||
|
"zotregistry.io/zot/pkg/meta/signatures"
|
||
|
"zotregistry.io/zot/pkg/test"
|
||
|
)
|
||
|
|
||
|
var errExpiryError = errors.New("expiry err")
|
||
|
|
||
|
func TestInitCosignAndNotationDirs(t *testing.T) {
|
||
|
Convey("InitCosignDir error", t, func() {
|
||
|
dir := t.TempDir()
|
||
|
err := os.Chmod(dir, 0o000)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = signatures.InitCosignAndNotationDirs(dir)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
|
||
|
err = os.Chmod(dir, 0o500)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = signatures.InitCosignAndNotationDirs(dir)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
|
||
|
cosignDir, err := signatures.GetCosignDirPath()
|
||
|
So(cosignDir, ShouldBeEmpty)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(err, ShouldEqual, zerr.ErrSignConfigDirNotSet)
|
||
|
})
|
||
|
|
||
|
Convey("InitNotationDir error", t, func() {
|
||
|
dir := t.TempDir()
|
||
|
err := os.Chmod(dir, 0o000)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = signatures.InitCosignAndNotationDirs(dir)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
|
||
|
err = os.Chmod(dir, 0o500)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = signatures.InitCosignAndNotationDirs(dir)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
|
||
|
err = signatures.InitNotationDir(dir)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
|
||
|
notationDir, err := signatures.GetNotationDirPath()
|
||
|
So(notationDir, ShouldBeEmpty)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(err, ShouldEqual, zerr.ErrSignConfigDirNotSet)
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func TestVerifySignatures(t *testing.T) {
|
||
|
Convey("wrong manifest content", t, func() {
|
||
|
manifestContent := []byte("wrong json")
|
||
|
|
||
|
_, _, _, err := signatures.VerifySignature("", []byte(""), "", "", manifestContent, "repo")
|
||
|
So(err, ShouldNotBeNil)
|
||
|
})
|
||
|
|
||
|
Convey("empty manifest digest", t, func() {
|
||
|
image, err := test.GetRandomImage("image")
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
manifestContent, err := json.Marshal(image.Manifest)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_, _, _, err = signatures.VerifySignature("", []byte(""), "", "", manifestContent, "repo")
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(err, ShouldEqual, zerr.ErrBadManifestDigest)
|
||
|
})
|
||
|
|
||
|
Convey("wrong signature type", t, func() {
|
||
|
image, err := test.GetRandomImage("image")
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
manifestContent, err := json.Marshal(image.Manifest)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
manifestDigest, err := image.Digest()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_, _, _, err = signatures.VerifySignature("wrongType", []byte(""), "", manifestDigest, manifestContent, "repo")
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(err, ShouldEqual, zerr.ErrInvalidSignatureType)
|
||
|
})
|
||
|
|
||
|
Convey("verify cosign signature", t, func() {
|
||
|
repo := "repo"
|
||
|
tag := "test"
|
||
|
image, err := test.GetRandomImage(tag)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
manifestContent, err := json.Marshal(image.Manifest)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
manifestDigest, err := image.Digest()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
Convey("cosignDir is not set", func() {
|
||
|
_, _, _, err = signatures.VerifySignature("cosign", []byte(""), "", manifestDigest, manifestContent, repo)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(err, ShouldEqual, zerr.ErrSignConfigDirNotSet)
|
||
|
})
|
||
|
|
||
|
Convey("cosignDir does not have read permissions", func() {
|
||
|
dir := t.TempDir()
|
||
|
|
||
|
err := signatures.InitCosignDir(dir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
cosignDir, err := signatures.GetCosignDirPath()
|
||
|
So(err, ShouldBeNil)
|
||
|
err = os.Chmod(cosignDir, 0o300)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_, _, _, err = signatures.VerifySignature("cosign", []byte(""), "", manifestDigest, manifestContent, repo)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
})
|
||
|
|
||
|
Convey("no valid public key", func() {
|
||
|
dir := t.TempDir()
|
||
|
|
||
|
err := signatures.InitCosignDir(dir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
cosignDir, err := signatures.GetCosignDirPath()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = test.WriteFileWithPermission(path.Join(cosignDir, "file"), []byte("not a public key"), 0o600, false)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_, _, isTrusted, err := signatures.VerifySignature("cosign", []byte(""), "", manifestDigest, manifestContent, repo)
|
||
|
So(err, ShouldBeNil)
|
||
|
So(isTrusted, ShouldBeFalse)
|
||
|
})
|
||
|
|
||
|
Convey("signature is trusted", func() {
|
||
|
rootDir := t.TempDir()
|
||
|
|
||
|
port := test.GetFreePort()
|
||
|
baseURL := test.GetBaseURL(port)
|
||
|
conf := config.New()
|
||
|
conf.HTTP.Port = port
|
||
|
conf.Storage.GC = false
|
||
|
ctlr := api.NewController(conf)
|
||
|
ctlr.Config.Storage.RootDirectory = rootDir
|
||
|
|
||
|
cm := test.NewControllerManager(ctlr)
|
||
|
cm.StartAndWait(conf.HTTP.Port)
|
||
|
defer cm.StopServer()
|
||
|
|
||
|
err := test.UploadImage(image, baseURL, repo)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = signatures.InitCosignDir(rootDir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
cosignDir, err := signatures.GetCosignDirPath()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
cwd, err := os.Getwd()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_ = os.Chdir(cosignDir)
|
||
|
|
||
|
// generate a keypair
|
||
|
os.Setenv("COSIGN_PASSWORD", "")
|
||
|
err = generate.GenerateKeyPairCmd(context.TODO(), "", "cosign", nil)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_ = os.Chdir(cwd)
|
||
|
|
||
|
// sign the image
|
||
|
err = sign.SignCmd(&options.RootOptions{Verbose: true, Timeout: 1 * time.Minute},
|
||
|
options.KeyOpts{KeyRef: path.Join(cosignDir, "cosign.key"), PassFunc: generate.GetPass},
|
||
|
options.SignOptions{
|
||
|
Registry: options.RegistryOptions{AllowInsecure: true},
|
||
|
AnnotationOptions: options.AnnotationOptions{Annotations: []string{fmt.Sprintf("tag=%s", tag)}},
|
||
|
Upload: true,
|
||
|
},
|
||
|
[]string{fmt.Sprintf("localhost:%s/%s@%s", port, repo, manifestDigest.String())})
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = os.Remove(path.Join(cosignDir, "cosign.key"))
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
indexContent, err := ctlr.StoreController.DefaultStore.GetIndexContent(repo)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
var index ispec.Index
|
||
|
err = json.Unmarshal(indexContent, &index)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
var rawSignature []byte
|
||
|
var sigKey string
|
||
|
|
||
|
for _, manifest := range index.Manifests {
|
||
|
if manifest.Digest != manifestDigest {
|
||
|
blobContent, err := ctlr.StoreController.DefaultStore.GetBlobContent(repo, manifest.Digest)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
var cosignSig ispec.Manifest
|
||
|
|
||
|
err = json.Unmarshal(blobContent, &cosignSig)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
sigKey = cosignSig.Layers[0].Annotations[signatures.CosignSigKey]
|
||
|
|
||
|
rawSignature, err = ctlr.StoreController.DefaultStore.GetBlobContent(repo, cosignSig.Layers[0].Digest)
|
||
|
So(err, ShouldBeNil)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// signature is trusted
|
||
|
author, _, isTrusted, err := signatures.VerifySignature("cosign", rawSignature, sigKey, manifestDigest,
|
||
|
manifestContent, repo)
|
||
|
So(err, ShouldBeNil)
|
||
|
So(isTrusted, ShouldBeTrue)
|
||
|
So(author, ShouldNotBeEmpty)
|
||
|
})
|
||
|
})
|
||
|
|
||
|
Convey("verify notation signature", t, func() {
|
||
|
repo := "repo"
|
||
|
tag := "test"
|
||
|
image, err := test.GetRandomImage(tag)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
manifestContent, err := json.Marshal(image.Manifest)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
manifestDigest, err := image.Digest()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
Convey("notationDir is not set", func() {
|
||
|
_, _, _, err = signatures.VerifySignature("notation", []byte("signature"), "", manifestDigest, manifestContent, repo)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(err, ShouldEqual, zerr.ErrSignConfigDirNotSet)
|
||
|
})
|
||
|
|
||
|
Convey("no signature provided", func() {
|
||
|
dir := t.TempDir()
|
||
|
|
||
|
err := signatures.InitNotationDir(dir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_, _, isTrusted, err := signatures.VerifySignature("notation", []byte(""), "", manifestDigest, manifestContent, repo)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(isTrusted, ShouldBeFalse)
|
||
|
})
|
||
|
|
||
|
Convey("trustpolicy.json does not exist", func() {
|
||
|
dir := t.TempDir()
|
||
|
|
||
|
err := signatures.InitNotationDir(dir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_, _, _, err = signatures.VerifySignature("notation", []byte("signature"), "", manifestDigest, manifestContent, repo)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
})
|
||
|
|
||
|
Convey("trustpolicy.json has invalid content", func() {
|
||
|
dir := t.TempDir()
|
||
|
|
||
|
err := signatures.InitNotationDir(dir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
notationDir, err := signatures.GetNotationDirPath()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = test.WriteFileWithPermission(path.Join(notationDir, "trustpolicy.json"), []byte("invalid content"),
|
||
|
0o600, false)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
_, _, _, err = signatures.VerifySignature("notation", []byte("signature"), "", manifestDigest, manifestContent,
|
||
|
repo)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
})
|
||
|
|
||
|
Convey("signature is trusted", func() {
|
||
|
rootDir := t.TempDir()
|
||
|
|
||
|
port := test.GetFreePort()
|
||
|
baseURL := test.GetBaseURL(port)
|
||
|
conf := config.New()
|
||
|
conf.HTTP.Port = port
|
||
|
conf.Storage.GC = false
|
||
|
ctlr := api.NewController(conf)
|
||
|
ctlr.Config.Storage.RootDirectory = rootDir
|
||
|
|
||
|
cm := test.NewControllerManager(ctlr)
|
||
|
cm.StartAndWait(conf.HTTP.Port)
|
||
|
defer cm.StopServer()
|
||
|
|
||
|
err := test.UploadImage(image, baseURL, repo)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = signatures.InitNotationDir(rootDir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
notationDir, err := signatures.GetNotationDirPath()
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
test.NotationPathLock.Lock()
|
||
|
defer test.NotationPathLock.Unlock()
|
||
|
|
||
|
test.LoadNotationPath(notationDir)
|
||
|
|
||
|
// generate a keypair
|
||
|
err = test.GenerateNotationCerts(notationDir, "notation-sign-test")
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
// sign the image
|
||
|
image := fmt.Sprintf("localhost:%s/%s", port, fmt.Sprintf("%s:%s", repo, tag))
|
||
|
|
||
|
err = test.SignWithNotation("notation-sign-test", image, notationDir)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = test.CopyFiles(path.Join(notationDir, "notation", "truststore"), path.Join(notationDir, "truststore"))
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
err = os.RemoveAll(path.Join(notationDir, "notation"))
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
trustPolicy := `
|
||
|
{
|
||
|
"version": "1.0",
|
||
|
"trustPolicies": [
|
||
|
{
|
||
|
"name": "notation-sign-test",
|
||
|
"registryScopes": [ "*" ],
|
||
|
"signatureVerification": {
|
||
|
"level" : "strict"
|
||
|
},
|
||
|
"trustStores": ["ca:notation-sign-test"],
|
||
|
"trustedIdentities": [
|
||
|
"*"
|
||
|
]
|
||
|
}
|
||
|
]
|
||
|
}`
|
||
|
|
||
|
err = test.WriteFileWithPermission(path.Join(notationDir, "trustpolicy.json"), []byte(trustPolicy), 0o600, false)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
indexContent, err := ctlr.StoreController.DefaultStore.GetIndexContent(repo)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
var index ispec.Index
|
||
|
err = json.Unmarshal(indexContent, &index)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
var rawSignature []byte
|
||
|
var sigKey string
|
||
|
|
||
|
for _, manifest := range index.Manifests {
|
||
|
if manifest.Digest != manifestDigest {
|
||
|
blobContent, err := ctlr.StoreController.DefaultStore.GetBlobContent(repo, manifest.Digest)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
var notationSig ispec.Manifest
|
||
|
|
||
|
err = json.Unmarshal(blobContent, ¬ationSig)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
sigKey = notationSig.Layers[0].MediaType
|
||
|
|
||
|
rawSignature, err = ctlr.StoreController.DefaultStore.GetBlobContent(repo, notationSig.Layers[0].Digest)
|
||
|
So(err, ShouldBeNil)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// signature is trusted
|
||
|
author, _, isTrusted, err := signatures.VerifySignature("notation", rawSignature, sigKey, manifestDigest,
|
||
|
manifestContent, repo)
|
||
|
So(err, ShouldBeNil)
|
||
|
So(isTrusted, ShouldBeTrue)
|
||
|
So(author, ShouldNotBeEmpty)
|
||
|
|
||
|
err = os.Truncate(path.Join(notationDir, "truststore/x509/ca/notation-sign-test/notation-sign-test.crt"), 0)
|
||
|
So(err, ShouldBeNil)
|
||
|
|
||
|
// signature is not trusted
|
||
|
author, _, isTrusted, err = signatures.VerifySignature("notation", rawSignature, sigKey, manifestDigest,
|
||
|
manifestContent, repo)
|
||
|
So(err, ShouldNotBeNil)
|
||
|
So(isTrusted, ShouldBeFalse)
|
||
|
So(author, ShouldNotBeEmpty)
|
||
|
})
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func TestCheckExpiryErr(t *testing.T) {
|
||
|
Convey("no expiry err", t, func() {
|
||
|
isExpiryErr := signatures.CheckExpiryErr([]*notation.ValidationResult{{Error: nil, Type: "wrongtype"}}, time.Now(),
|
||
|
nil)
|
||
|
So(isExpiryErr, ShouldBeFalse)
|
||
|
|
||
|
isExpiryErr = signatures.CheckExpiryErr([]*notation.ValidationResult{{
|
||
|
Error: nil, Type: trustpolicy.TypeAuthenticTimestamp,
|
||
|
}}, time.Now(), errExpiryError)
|
||
|
So(isExpiryErr, ShouldBeFalse)
|
||
|
})
|
||
|
|
||
|
Convey("expiry err", t, func() {
|
||
|
isExpiryErr := signatures.CheckExpiryErr([]*notation.ValidationResult{
|
||
|
{Error: errExpiryError, Type: trustpolicy.TypeExpiry},
|
||
|
}, time.Now(), errExpiryError)
|
||
|
So(isExpiryErr, ShouldBeTrue)
|
||
|
|
||
|
isExpiryErr = signatures.CheckExpiryErr([]*notation.ValidationResult{
|
||
|
{Error: errExpiryError, Type: trustpolicy.TypeAuthenticTimestamp},
|
||
|
}, time.Now().AddDate(0, 0, -1), errExpiryError)
|
||
|
So(isExpiryErr, ShouldBeTrue)
|
||
|
})
|
||
|
}
|