diff --git a/jsonld/test.go b/jsonld/test.go index 996d40e95b..a4215eba14 100644 --- a/jsonld/test.go +++ b/jsonld/test.go @@ -150,7 +150,7 @@ func (t testContextManager) Configure(config Config) error { } // NewTestJSONLDManager creates a new test context manager which contains extra test contexts -func NewTestJSONLDManager(t *testing.T) JSONLD { +func NewTestJSONLDManager(t testing.TB) JSONLD { t.Helper() contextConfig := DefaultContextConfig() diff --git a/vcr/holder/wallet.go b/vcr/holder/wallet.go index d464077e1e..e383235c5d 100644 --- a/vcr/holder/wallet.go +++ b/vcr/holder/wallet.go @@ -89,7 +89,7 @@ func (h wallet) BuildPresentation(ctx context.Context, credentials []vc.Verifiab if validateVC { for _, cred := range credentials { - err := h.verifier.Validate(cred, &options.ProofOptions.Created) + err := h.verifier.VerifySignature(cred, &options.ProofOptions.Created) if err != nil { return nil, core.InvalidInputError("invalid credential (id=%s): %w", cred.ID, err) } diff --git a/vcr/holder/wallet_test.go b/vcr/holder/wallet_test.go index 10c0521370..0401ec82b9 100644 --- a/vcr/holder/wallet_test.go +++ b/vcr/holder/wallet_test.go @@ -224,7 +224,7 @@ func TestWallet_BuildPresentation(t *testing.T) { keyResolver := resolver.NewMockKeyResolver(ctrl) mockVerifier := verifier.NewMockVerifier(ctrl) - mockVerifier.EXPECT().Validate(testCredential, &created) + mockVerifier.EXPECT().VerifySignature(testCredential, &created) keyResolver.EXPECT().ResolveKey(testDID, nil, resolver.NutsSigningKeyType).Return(ssi.MustParseURI(kid), key.Public(), nil) @@ -240,7 +240,7 @@ func TestWallet_BuildPresentation(t *testing.T) { keyResolver := resolver.NewMockKeyResolver(ctrl) mockVerifier := verifier.NewMockVerifier(ctrl) - mockVerifier.EXPECT().Validate(testCredential, &created).Return(errors.New("failed")) + mockVerifier.EXPECT().VerifySignature(testCredential, &created).Return(errors.New("failed")) keyResolver.EXPECT().ResolveKey(testDID, nil, resolver.NutsSigningKeyType).Return(ssi.MustParseURI(kid), key.Public(), nil) @@ -256,7 +256,7 @@ func TestWallet_BuildPresentation(t *testing.T) { keyResolver := resolver.NewMockKeyResolver(ctrl) mockVerifier := verifier.NewMockVerifier(ctrl) - mockVerifier.EXPECT().Validate(gomock.Any(), gomock.Any()) + mockVerifier.EXPECT().VerifySignature(gomock.Any(), gomock.Any()) keyResolver.EXPECT().ResolveKey(testDID, nil, resolver.NutsSigningKeyType).Return(ssi.MustParseURI(kid), key.Public(), nil) diff --git a/vcr/signature/proof/jsonld.go b/vcr/signature/proof/jsonld.go index ff5b71fd2b..f4914b9c00 100644 --- a/vcr/signature/proof/jsonld.go +++ b/vcr/signature/proof/jsonld.go @@ -80,6 +80,9 @@ func (o ProofOptions) ValidAt(at time.Time, maxSkew time.Duration) bool { return true } +var _ Proof = (*LDProof)(nil) +var _ ProofVerifier = (*LDProof)(nil) + // LDProof contains the fields of the Proof data model: https://w3c-ccg.github.io/data-integrity-spec/#proofs type LDProof struct { ProofOptions diff --git a/vcr/signature/proof/proof.go b/vcr/signature/proof/proof.go index 164b3d8a02..199b4b842a 100644 --- a/vcr/signature/proof/proof.go +++ b/vcr/signature/proof/proof.go @@ -19,6 +19,7 @@ package proof import ( + "context" "crypto" "encoding/json" nutsCrypto "github.com/nuts-foundation/nuts-node/crypto" @@ -70,7 +71,7 @@ func (d SignedDocument) UnmarshalProofValue(target interface{}) error { // Proof is the interface that defines a set of methods which a proof should implement. type Proof interface { // Sign defines the basic signing operation on the proof. - Sign(document Document, suite signature.Suite, key nutsCrypto.Key) (interface{}, error) + Sign(ctx context.Context, document Document, suite signature.Suite, key nutsCrypto.Key) (interface{}, error) } // ProofVerifier defines the generic verifier interface diff --git a/vcr/store.go b/vcr/store.go index e74c467711..2c1a12e8ea 100644 --- a/vcr/store.go +++ b/vcr/store.go @@ -57,7 +57,7 @@ func (c *vcr) StoreCredential(credential vc.VerifiableCredential, validAt *time. } // verify first - if err := c.verifier.Validate(credential, validAt); err != nil { + if err := c.verifier.VerifySignature(credential, validAt); err != nil { return err } diff --git a/vcr/verifier/interface.go b/vcr/verifier/interface.go index 793d92acf1..a1d9d40aaf 100644 --- a/vcr/verifier/interface.go +++ b/vcr/verifier/interface.go @@ -32,12 +32,12 @@ import ( // Verifier defines the interface for verifying verifiable credentials. type Verifier interface { // Verify checks credential on full correctness. It checks: - // validity of the signature + // validity of the signature (optional) // if it has been revoked - // if the issuer is registered as trusted + // if the issuer is registered as trusted (optional) Verify(credential vc.VerifiableCredential, allowUntrusted bool, checkSignature bool, validAt *time.Time) error - // Validate checks the verifiable credential technical correctness - Validate(credentialToVerify vc.VerifiableCredential, at *time.Time) error + // VerifySignature checks that the signature on the verifiable credential is correct and valid at the given time and nothing else + VerifySignature(credentialToVerify vc.VerifiableCredential, at *time.Time) error // IsRevoked checks if the credential is revoked IsRevoked(credentialID ssi.URI) (bool, error) // GetRevocation returns the first revocation by credential ID diff --git a/vcr/verifier/mock.go b/vcr/verifier/mock.go index 709e9acd4f..a0557a852d 100644 --- a/vcr/verifier/mock.go +++ b/vcr/verifier/mock.go @@ -86,32 +86,32 @@ func (mr *MockVerifierMockRecorder) RegisterRevocation(revocation any) *gomock.C return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "RegisterRevocation", reflect.TypeOf((*MockVerifier)(nil).RegisterRevocation), revocation) } -// Validate mocks base method. -func (m *MockVerifier) Validate(credentialToVerify vc.VerifiableCredential, at *time.Time) error { +// Verify mocks base method. +func (m *MockVerifier) Verify(credential vc.VerifiableCredential, allowUntrusted, checkSignature bool, validAt *time.Time) error { m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Validate", credentialToVerify, at) + ret := m.ctrl.Call(m, "Verify", credential, allowUntrusted, checkSignature, validAt) ret0, _ := ret[0].(error) return ret0 } -// Validate indicates an expected call of Validate. -func (mr *MockVerifierMockRecorder) Validate(credentialToVerify, at any) *gomock.Call { +// Verify indicates an expected call of Verify. +func (mr *MockVerifierMockRecorder) Verify(credential, allowUntrusted, checkSignature, validAt any) *gomock.Call { mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Validate", reflect.TypeOf((*MockVerifier)(nil).Validate), credentialToVerify, at) + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Verify", reflect.TypeOf((*MockVerifier)(nil).Verify), credential, allowUntrusted, checkSignature, validAt) } -// Verify mocks base method. -func (m *MockVerifier) Verify(credential vc.VerifiableCredential, allowUntrusted, checkSignature bool, validAt *time.Time) error { +// VerifySignature mocks base method. +func (m *MockVerifier) VerifySignature(credentialToVerify vc.VerifiableCredential, at *time.Time) error { m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Verify", credential, allowUntrusted, checkSignature, validAt) + ret := m.ctrl.Call(m, "VerifySignature", credentialToVerify, at) ret0, _ := ret[0].(error) return ret0 } -// Verify indicates an expected call of Verify. -func (mr *MockVerifierMockRecorder) Verify(credential, allowUntrusted, checkSignature, validAt any) *gomock.Call { +// VerifySignature indicates an expected call of VerifySignature. +func (mr *MockVerifierMockRecorder) VerifySignature(credentialToVerify, at any) *gomock.Call { mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Verify", reflect.TypeOf((*MockVerifier)(nil).Verify), credential, allowUntrusted, checkSignature, validAt) + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "VerifySignature", reflect.TypeOf((*MockVerifier)(nil).VerifySignature), credentialToVerify, at) } // VerifyVP mocks base method. diff --git a/vcr/verifier/signature_verifier.go b/vcr/verifier/signature_verifier.go new file mode 100644 index 0000000000..a8ae294937 --- /dev/null +++ b/vcr/verifier/signature_verifier.go @@ -0,0 +1,128 @@ +package verifier + +import ( + crypt "crypto" + "errors" + "fmt" + "github.com/nuts-foundation/nuts-node/vcr/credential" + "github.com/nuts-foundation/nuts-node/vcr/types" + "strings" + "time" + + "github.com/lestrrat-go/jwx/v2/jwt" + "github.com/nuts-foundation/go-did/vc" + "github.com/nuts-foundation/nuts-node/crypto" + "github.com/nuts-foundation/nuts-node/jsonld" + "github.com/nuts-foundation/nuts-node/vcr/issuer" + "github.com/nuts-foundation/nuts-node/vcr/signature" + "github.com/nuts-foundation/nuts-node/vcr/signature/proof" + "github.com/nuts-foundation/nuts-node/vdr/resolver" +) + +type signatureVerifier struct { + keyResolver resolver.KeyResolver + jsonldManager jsonld.JSONLD +} + +// VerifySignature implements the Proof Verification Algorithm: https://w3c-ccg.github.io/data-integrity-spec/#proof-verification-algorithm +func (sv *signatureVerifier) VerifySignature(credentialToVerify vc.VerifiableCredential, validateAt *time.Time) error { + switch credentialToVerify.Format() { + case issuer.JSONLDCredentialFormat: + return sv.jsonldProof(credentialToVerify, credentialToVerify.Issuer.String(), validateAt) + case issuer.JWTCredentialFormat: + return sv.jwtSignature(credentialToVerify.Raw(), credentialToVerify.Issuer.String(), validateAt) + default: + return errors.New("unsupported credential proof format") + } +} + +// verifyVPSignature implements the Proof Verification Algorithm: https://w3c-ccg.github.io/data-integrity-spec/#proof-verification-algorithm +func (sv *signatureVerifier) verifyVPSignature(presentation vc.VerifiablePresentation, validateAt *time.Time) error { + signerDID, err := credential.PresentationSigner(presentation) + if err != nil { + return toVerificationError(err) + } + + switch presentation.Format() { + case issuer.JSONLDPresentationFormat: + return sv.jsonldProof(presentation, signerDID.String(), validateAt) + case issuer.JWTPresentationFormat: + return sv.jwtSignature(presentation.Raw(), signerDID.String(), validateAt) + default: + return errors.New("unsupported presentation proof format") + } +} + +// jsonldProof checks the signature on a VC or VP in jsonld format (contains a proof) +func (sv *signatureVerifier) jsonldProof(documentToVerify any, issuer string, at *time.Time) error { + signedDocument, err := proof.NewSignedDocument(documentToVerify) + if err != nil { + return newVerificationError("invalid LD-JSON document: %w", err) + } + + ldProof := proof.LDProof{} + if err = signedDocument.UnmarshalProofValue(&ldProof); err != nil { + return newVerificationError("unsupported proof type: %w", err) + } + + // for a VP this will not fail + verificationMethod := ldProof.VerificationMethod.String() + verificationMethodIssuer := strings.Split(verificationMethod, "#")[0] + if verificationMethodIssuer == "" || verificationMethodIssuer != issuer { + return errVerificationMethodNotOfIssuer + } + + // verify signing time + validAt := time.Now() + if at != nil { + validAt = *at + } + if !ldProof.ValidAt(validAt, maxSkew) { + return toVerificationError(types.ErrPresentationNotValidAtTime) + } + + // find key + signingKey, err := sv.keyResolver.ResolveKeyByID(ldProof.VerificationMethod.String(), at, resolver.NutsSigningKeyType) + if err != nil { + return fmt.Errorf("unable to resolve valid signing key: %w", err) + } + + // verify signature + err = ldProof.Verify(signedDocument.DocumentWithoutProof(), signature.JSONWebSignature2020{ContextLoader: sv.jsonldManager.DocumentLoader()}, signingKey) + if err != nil { + return newVerificationError("invalid signature: %w", err) + } + return nil +} + +func (sv *signatureVerifier) jwtSignature(jwtDocumentToVerify string, issuer string, at *time.Time) error { + var keyID string + _, err := crypto.ParseJWT(jwtDocumentToVerify, func(kid string) (crypt.PublicKey, error) { + keyID = kid + return sv.resolveSigningKey(kid, issuer, at) + }, jwt.WithClock(jwt.ClockFunc(func() time.Time { + if at == nil { + return time.Now() + } + return *at + }))) + if err != nil { + return fmt.Errorf("unable to validate JWT credential: %w", err) + } + if keyID != "" && strings.Split(keyID, "#")[0] != issuer { + return errVerificationMethodNotOfIssuer + } + return nil +} + +func (sv *signatureVerifier) resolveSigningKey(kid string, issuer string, at *time.Time) (crypt.PublicKey, error) { + // Compatibility: VC data model v1 puts key discovery out of scope and does not require the `kid` header. + // When `kid` isn't present use the JWT issuer as `kid`, then it is at least compatible with DID methods that contain a single verification method (did:jwk). + if kid == "" { + kid = issuer + } + if strings.HasPrefix(kid, "did:jwk:") && !strings.Contains(kid, "#") { + kid += "#0" + } + return sv.keyResolver.ResolveKeyByID(kid, at, resolver.NutsSigningKeyType) +} diff --git a/vcr/verifier/signature_verifier_test.go b/vcr/verifier/signature_verifier_test.go new file mode 100644 index 0000000000..78bcf1ced3 --- /dev/null +++ b/vcr/verifier/signature_verifier_test.go @@ -0,0 +1,235 @@ +package verifier + +import ( + "context" + "crypto" + "crypto/ecdsa" + "crypto/elliptic" + "crypto/rand" + "encoding/base64" + "encoding/json" + "errors" + "os" + "testing" + "time" + + "github.com/lestrrat-go/jwx/v2/jwk" + ssi "github.com/nuts-foundation/go-did" + "github.com/nuts-foundation/go-did/did" + "github.com/nuts-foundation/go-did/vc" + "github.com/nuts-foundation/nuts-node/audit" + nutsCrypto "github.com/nuts-foundation/nuts-node/crypto" + "github.com/nuts-foundation/nuts-node/crypto/storage/spi" + "github.com/nuts-foundation/nuts-node/jsonld" + "github.com/nuts-foundation/nuts-node/vdr/didjwk" + "github.com/nuts-foundation/nuts-node/vdr/resolver" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "go.uber.org/mock/gomock" +) + +func TestSignatureVerifier_VerifySignature(t *testing.T) { + const testKID = "did:nuts:CuE3qeFGGLhEAS3gKzhMCeqd1dGa9at5JCbmCfyMU2Ey#sNGDQ3NlOe6Icv0E7_ufviOLG6Y25bSEyS5EbXBgp8Y" + + // load pub key + pke := spi.PublicKeyEntry{} + pkeJSON, _ := os.ReadFile("../test/public.json") + json.Unmarshal(pkeJSON, &pke) + var pk = new(ecdsa.PublicKey) + pke.JWK().Raw(pk) + + now := time.Now() + timeFunc = func() time.Time { + return now + } + defer func() { + timeFunc = time.Now + }() + + t.Run("JSON-LD", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + mockKeyResolver.EXPECT().ResolveKeyByID(testKID, gomock.Any(), resolver.NutsSigningKeyType).Return(pk, nil) + + err := sv.VerifySignature(testCredential(t), nil) + + assert.NoError(t, err) + }) + t.Run("JWT", func(t *testing.T) { + // Create did:jwk for issuer, and sign credential + keyStore := nutsCrypto.NewMemoryCryptoInstance() + key, err := keyStore.New(audit.TestContext(), func(key crypto.PublicKey) (string, error) { + keyAsJWK, _ := jwk.FromRaw(key) + keyJSON, _ := json.Marshal(keyAsJWK) + return "did:jwk:" + base64.RawStdEncoding.EncodeToString(keyJSON) + "#0", nil + }) + require.NoError(t, err) + + template := testCredential(t) + template.Issuer = did.MustParseDIDURL(key.KID()).DID.URI() + + cred, err := vc.CreateJWTVerifiableCredential(audit.TestContext(), template, func(ctx context.Context, claims map[string]interface{}, headers map[string]interface{}) (string, error) { + return keyStore.SignJWT(ctx, claims, headers, key) + }) + require.NoError(t, err) + + t.Run("with kid header", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + mockKeyResolver.EXPECT().ResolveKeyByID(key.KID(), gomock.Any(), resolver.NutsSigningKeyType).Return(key.Public(), nil) + err = sv.VerifySignature(*cred, nil) + + assert.NoError(t, err) + }) + t.Run("kid header does not match credential issuer", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + + cred, err := vc.CreateJWTVerifiableCredential(audit.TestContext(), template, func(ctx context.Context, claims map[string]interface{}, headers map[string]interface{}) (string, error) { + return keyStore.SignJWT(ctx, claims, headers, key) + }) + require.NoError(t, err) + cred.Issuer = ssi.MustParseURI("did:example:test") + + mockKeyResolver.EXPECT().ResolveKeyByID(key.KID(), gomock.Any(), resolver.NutsSigningKeyType).Return(key.Public(), nil) + err = sv.VerifySignature(*cred, nil) + + assert.ErrorIs(t, err, errVerificationMethodNotOfIssuer) + }) + t.Run("signature invalid", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + realKey, _ := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) + mockKeyResolver.EXPECT().ResolveKeyByID(key.KID(), gomock.Any(), resolver.NutsSigningKeyType).Return(realKey.Public(), nil) + + err = sv.VerifySignature(*cred, nil) + + assert.EqualError(t, err, "unable to validate JWT credential: could not verify message using any of the signatures or keys") + }) + t.Run("expired token", func(t *testing.T) { + // Credential taken from Sphereon Wallet, expires on Tue Oct 03 2023 + const credentialJSON = `eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCJ9.eyJleHAiOjE2OTYzMDE3MDgsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZmlyc3ROYW1lIjoiSGVsbG8iLCJsYXN0TmFtZSI6IlNwaGVyZW9uIiwiZW1haWwiOiJzcGhlcmVvbkBleGFtcGxlLmNvbSIsInR5cGUiOiJTcGhlcmVvbiBHdWVzdCIsImlkIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEifX0sIkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiZXhwaXJhdGlvbkRhdGUiOiIyMDIzLTEwLTAzVDAyOjU1OjA4LjEzM1oiLCJjcmVkZW50aWFsU3ViamVjdCI6eyJmaXJzdE5hbWUiOiJIZWxsbyIsImxhc3ROYW1lIjoiU3BoZXJlb24iLCJlbWFpbCI6InNwaGVyZW9uQGV4YW1wbGUuY29tIiwidHlwZSI6IlNwaGVyZW9uIEd1ZXN0IiwiaWQiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOa3NpTENKMWMyVWlPaUp6YVdjaUxDSnJkSGtpT2lKRlF5SXNJbU55ZGlJNkluTmxZM0F5TlRack1TSXNJbmdpT2lKak1WZFljemRYTTIxNWMyVlZaazVDY1hONFpGQlhRa2xIYUV0a05GUjZNRXhTTFVacU9FWk5XV0V3SWl3aWVTSTZJbGR0YTBOWWRURjNlWHBhWjBkT04xVjRUbUZ3Y0hGdVQxRmhUMnRYTWtOblQxTnVUMjk1VFVsVWRXTWlmUSJ9LCJpc3N1ZXIiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOaUlzSW5WelpTSTZJbk5wWnlJc0ltdDBlU0k2SWtWRElpd2lZM0oySWpvaVVDMHlOVFlpTENKNElqb2lWRWN5U0RKNE1tUlhXRTR6ZFVOeFduQnhSakY1YzBGUVVWWkVTa1ZPWDBndFEwMTBZbWRxWWkxT1p5SXNJbmtpT2lJNVRUaE9lR1F3VUU0eU1rMDViRkJFZUdSd1JIQnZWRXg2TVRWM1pubGFTbk0yV21oTFNWVktNek00SW4wIiwiaXNzdWFuY2VEYXRlIjoiMjAyMy0wOS0yOVQxMjozMTowOC4xMzNaIiwic3ViIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEiLCJuYmYiOjE2OTU5OTA2NjgsImlzcyI6ImRpZDpqd2s6ZXlKaGJHY2lPaUpGVXpJMU5pSXNJblZ6WlNJNkluTnBaeUlzSW10MGVTSTZJa1ZESWl3aVkzSjJJam9pVUMweU5UWWlMQ0o0SWpvaVZFY3lTREo0TW1SWFdFNHpkVU54V25CeFJqRjVjMEZRVVZaRVNrVk9YMGd0UTAxMFltZHFZaTFPWnlJc0lua2lPaUk1VFRoT2VHUXdVRTR5TWswNWJGQkVlR1J3UkhCdlZFeDZNVFYzWm5sYVNuTTJXbWhMU1ZWS016TTRJbjAifQ.wdhtLXE4jU1C-3YBBpP9-qE-yh1xOZ6lBLJ-0e5_Sa7fnrUHcAaU1n3kN2CeCyTVjtm1Uy3Tl6RzUOM6MjP3vQ` + cred, _ := vc.ParseVerifiableCredential(credentialJSON) + + sv := signatureVerifier{ + keyResolver: resolver.DIDKeyResolver{ + Resolver: didjwk.NewResolver(), + }, + } + err := sv.VerifySignature(*cred, nil) + + assert.EqualError(t, err, "unable to validate JWT credential: \"exp\" not satisfied") + }) + t.Run("without kid header, derived from issuer", func(t *testing.T) { + // Credential taken from Sphereon Wallet + const credentialJSON = `eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCJ9.eyJleHAiOjE2OTYzMDE3MDgsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZmlyc3ROYW1lIjoiSGVsbG8iLCJsYXN0TmFtZSI6IlNwaGVyZW9uIiwiZW1haWwiOiJzcGhlcmVvbkBleGFtcGxlLmNvbSIsInR5cGUiOiJTcGhlcmVvbiBHdWVzdCIsImlkIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEifX0sIkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiZXhwaXJhdGlvbkRhdGUiOiIyMDIzLTEwLTAzVDAyOjU1OjA4LjEzM1oiLCJjcmVkZW50aWFsU3ViamVjdCI6eyJmaXJzdE5hbWUiOiJIZWxsbyIsImxhc3ROYW1lIjoiU3BoZXJlb24iLCJlbWFpbCI6InNwaGVyZW9uQGV4YW1wbGUuY29tIiwidHlwZSI6IlNwaGVyZW9uIEd1ZXN0IiwiaWQiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOa3NpTENKMWMyVWlPaUp6YVdjaUxDSnJkSGtpT2lKRlF5SXNJbU55ZGlJNkluTmxZM0F5TlRack1TSXNJbmdpT2lKak1WZFljemRYTTIxNWMyVlZaazVDY1hONFpGQlhRa2xIYUV0a05GUjZNRXhTTFVacU9FWk5XV0V3SWl3aWVTSTZJbGR0YTBOWWRURjNlWHBhWjBkT04xVjRUbUZ3Y0hGdVQxRmhUMnRYTWtOblQxTnVUMjk1VFVsVWRXTWlmUSJ9LCJpc3N1ZXIiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOaUlzSW5WelpTSTZJbk5wWnlJc0ltdDBlU0k2SWtWRElpd2lZM0oySWpvaVVDMHlOVFlpTENKNElqb2lWRWN5U0RKNE1tUlhXRTR6ZFVOeFduQnhSakY1YzBGUVVWWkVTa1ZPWDBndFEwMTBZbWRxWWkxT1p5SXNJbmtpT2lJNVRUaE9lR1F3VUU0eU1rMDViRkJFZUdSd1JIQnZWRXg2TVRWM1pubGFTbk0yV21oTFNWVktNek00SW4wIiwiaXNzdWFuY2VEYXRlIjoiMjAyMy0wOS0yOVQxMjozMTowOC4xMzNaIiwic3ViIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEiLCJuYmYiOjE2OTU5OTA2NjgsImlzcyI6ImRpZDpqd2s6ZXlKaGJHY2lPaUpGVXpJMU5pSXNJblZ6WlNJNkluTnBaeUlzSW10MGVTSTZJa1ZESWl3aVkzSjJJam9pVUMweU5UWWlMQ0o0SWpvaVZFY3lTREo0TW1SWFdFNHpkVU54V25CeFJqRjVjMEZRVVZaRVNrVk9YMGd0UTAxMFltZHFZaTFPWnlJc0lua2lPaUk1VFRoT2VHUXdVRTR5TWswNWJGQkVlR1J3UkhCdlZFeDZNVFYzWm5sYVNuTTJXbWhMU1ZWS016TTRJbjAifQ.wdhtLXE4jU1C-3YBBpP9-qE-yh1xOZ6lBLJ-0e5_Sa7fnrUHcAaU1n3kN2CeCyTVjtm1Uy3Tl6RzUOM6MjP3vQ` + cred, _ := vc.ParseVerifiableCredential(credentialJSON) + + sv := signatureVerifier{ + keyResolver: resolver.DIDKeyResolver{ + Resolver: didjwk.NewResolver(), + }, + } + validAt := time.Date(2023, 9, 30, 0, 0, 0, 0, time.UTC) + err := sv.VerifySignature(*cred, &validAt) + + assert.NoError(t, err) + }) + }) + + t.Run("error - invalid vm", func(t *testing.T) { + sv, _ := signatureVerifierTestSetup(t) + + vc2 := testCredential(t) + pr := make([]vc.JSONWebSignature2020Proof, 0) + _ = vc2.UnmarshalProofValue(&pr) + u := ssi.MustParseURI(vc2.Issuer.String() + "2") + pr[0].VerificationMethod = u + vc2.Proof = []interface{}{pr[0]} + + err := sv.VerifySignature(vc2, nil) + + assert.Error(t, err) + assert.ErrorIs(t, err, errVerificationMethodNotOfIssuer) + }) + + t.Run("error - wrong hashed payload", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + vc2 := testCredential(t) + issuanceDate := time.Now() + vc2.IssuanceDate = &issuanceDate + + mockKeyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) + + err := sv.VerifySignature(vc2, nil) + + assert.ErrorContains(t, err, "failed to verify signature") + }) + + t.Run("error - wrong hashed proof", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + vc2 := testCredential(t) + pr := make([]vc.JSONWebSignature2020Proof, 0) + vc2.UnmarshalProofValue(&pr) + pr[0].Created = time.Now() + vc2.Proof = []interface{}{pr[0]} + + mockKeyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) + + err := sv.VerifySignature(vc2, nil) + + assert.ErrorContains(t, err, "failed to verify signature") + }) + + t.Run("error - no proof", func(t *testing.T) { + sv, _ := signatureVerifierTestSetup(t) + vc2 := testCredential(t) + vc2.Proof = []interface{}{} + + err := sv.VerifySignature(vc2, nil) + + assert.EqualError(t, err, "verification error: unsupported proof type: json: cannot unmarshal array into Go value of type proof.LDProof") + }) + + t.Run("error - wrong jws in proof", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + mockKeyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) + vc2 := testCredential(t) + pr := make([]vc.JSONWebSignature2020Proof, 0) + vc2.UnmarshalProofValue(&pr) + pr[0].Jws = "" + vc2.Proof = []interface{}{pr[0]} + + err := sv.VerifySignature(vc2, nil) + + assert.ErrorContains(t, err, "invalid 'jws' value in proof") + }) + + t.Run("error - wrong base64 encoding in jws", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + mockKeyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) + vc2 := testCredential(t) + pr := make([]vc.JSONWebSignature2020Proof, 0) + vc2.UnmarshalProofValue(&pr) + pr[0].Jws = "abac..ab//" + vc2.Proof = []interface{}{pr[0]} + + err := sv.VerifySignature(vc2, nil) + + assert.ErrorContains(t, err, "illegal base64 data") + }) + + t.Run("error - resolving key", func(t *testing.T) { + sv, mockKeyResolver := signatureVerifierTestSetup(t) + mockKeyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(nil, errors.New("b00m!")) + + err := sv.VerifySignature(testCredential(t), nil) + + assert.Error(t, err) + }) +} + +func signatureVerifierTestSetup(t testing.TB) (signatureVerifier, *resolver.MockKeyResolver) { + ctrl := gomock.NewController(t) + keyResolver := resolver.NewMockKeyResolver(ctrl) + return signatureVerifier{ + keyResolver: keyResolver, + jsonldManager: jsonld.NewTestJSONLDManager(t), + }, keyResolver +} diff --git a/vcr/verifier/verifier.go b/vcr/verifier/verifier.go index 31df90d8be..ece3f23437 100644 --- a/vcr/verifier/verifier.go +++ b/vcr/verifier/verifier.go @@ -19,14 +19,9 @@ package verifier import ( - crypt "crypto" "encoding/json" "errors" "fmt" - "github.com/lestrrat-go/jwx/v2/jwt" - "github.com/nuts-foundation/nuts-node/crypto" - "github.com/nuts-foundation/nuts-node/vcr/issuer" - "github.com/nuts-foundation/nuts-node/vdr/resolver" "strings" "time" @@ -39,6 +34,7 @@ import ( "github.com/nuts-foundation/nuts-node/vcr/signature/proof" "github.com/nuts-foundation/nuts-node/vcr/trust" "github.com/nuts-foundation/nuts-node/vcr/types" + "github.com/nuts-foundation/nuts-node/vdr/resolver" ) var timeFunc = time.Now @@ -58,6 +54,7 @@ type verifier struct { jsonldManager jsonld.JSONLD store Store trustConfig *trust.Config + signatureVerifier } // VerificationError is used to describe a VC/VP verification failure. @@ -86,86 +83,10 @@ func (e VerificationError) Error() string { // NewVerifier creates a new instance of the verifier. It needs a key resolver for validating signatures. func NewVerifier(store Store, didResolver resolver.DIDResolver, keyResolver resolver.KeyResolver, jsonldManager jsonld.JSONLD, trustConfig *trust.Config) Verifier { - return &verifier{store: store, didResolver: didResolver, keyResolver: keyResolver, jsonldManager: jsonldManager, trustConfig: trustConfig} -} - -// Validate implements the Proof Verification Algorithm: https://w3c-ccg.github.io/data-integrity-spec/#proof-verification-algorithm -func (v *verifier) Validate(credentialToVerify vc.VerifiableCredential, at *time.Time) error { - err := v.validateType(credentialToVerify) - if err != nil { - return err - } - - switch credentialToVerify.Format() { - case issuer.JSONLDCredentialFormat: - return v.validateJSONLDCredential(credentialToVerify, at) - case issuer.JWTCredentialFormat: - return v.validateJWTCredential(credentialToVerify, at) - default: - return errors.New("unsupported credential proof format") - } -} - -func (v *verifier) validateJSONLDCredential(credentialToVerify vc.VerifiableCredential, at *time.Time) error { - signedDocument, err := proof.NewSignedDocument(credentialToVerify) - if err != nil { - return fmt.Errorf("unable to build signed document from verifiable credential: %w", err) - } - - ldProof := proof.LDProof{} - if err := signedDocument.UnmarshalProofValue(&ldProof); err != nil { - return fmt.Errorf("unable to extract ldproof from signed document: %w", err) - } - - verificationMethod := ldProof.VerificationMethod.String() - verificationMethodIssuer := strings.Split(verificationMethod, "#")[0] - if verificationMethodIssuer == "" || verificationMethodIssuer != credentialToVerify.Issuer.String() { - return errVerificationMethodNotOfIssuer - } - - // find key - pk, err := v.keyResolver.ResolveKeyByID(ldProof.VerificationMethod.String(), at, resolver.NutsSigningKeyType) - if err != nil { - if at == nil { - return fmt.Errorf("unable to resolve signing key: %w", err) - } - return fmt.Errorf("unable to resolve valid signing key at given time: %w", err) - } - - // Try first with the correct LDProof implementation - return ldProof.Verify(signedDocument.DocumentWithoutProof(), signature.JSONWebSignature2020{ContextLoader: v.jsonldManager.DocumentLoader()}, pk) -} - -func (v *verifier) validateJWTCredential(credential vc.VerifiableCredential, at *time.Time) error { - var keyID string - _, err := crypto.ParseJWT(credential.Raw(), func(kid string) (crypt.PublicKey, error) { - keyID = kid - return v.resolveSigningKey(kid, credential.Issuer.String(), at) - }, jwt.WithClock(jwt.ClockFunc(func() time.Time { - if at == nil { - return time.Now() - } - return *at - }))) - if err != nil { - return fmt.Errorf("unable to validate JWT credential: %w", err) - } - if keyID != "" && strings.Split(keyID, "#")[0] != credential.Issuer.String() { - return errVerificationMethodNotOfIssuer - } - return nil -} - -func (v *verifier) resolveSigningKey(kid string, issuer string, at *time.Time) (crypt.PublicKey, error) { - // Compatibility: VC data model v1 puts key discovery out of scope and does not require the `kid` header. - // When `kid` isn't present use the JWT issuer as `kid`, then it is at least compatible with DID methods that contain a single verification method (did:jwk). - if kid == "" { - kid = issuer - } - if strings.HasPrefix(kid, "did:jwk:") && !strings.Contains(kid, "#") { - kid += "#0" - } - return v.keyResolver.ResolveKeyByID(kid, at, resolver.NutsSigningKeyType) + return &verifier{store: store, didResolver: didResolver, keyResolver: keyResolver, jsonldManager: jsonldManager, trustConfig: trustConfig, signatureVerifier: signatureVerifier{ + keyResolver: keyResolver, + jsonldManager: jsonldManager, + }} } // Verify implements the verify interface. @@ -176,14 +97,23 @@ func (v verifier) Verify(credentialToVerify vc.VerifiableCredential, allowUntrus if err := validator.Validate(credentialToVerify); err != nil { return err } + // We only accept VCs with at most 2 types: "VerifiableCredential" and a specific type + // The Validate above already checks "VerifiableCredential" is one of them + // This is a custom requirement + if len(credentialToVerify.Type) > 2 { + return errors.New("verifiable credential must list at most 2 types") + } // Check revocation status - revoked, err := v.IsRevoked(*credentialToVerify.ID) - if err != nil { - return err - } - if revoked { - return types.ErrRevoked + if credentialToVerify.ID != nil { + revoked, err := v.IsRevoked(*credentialToVerify.ID) + if err != nil { + return err + } + if revoked { + return types.ErrRevoked + } + } // Check trust status @@ -199,7 +129,8 @@ func (v verifier) Verify(credentialToVerify vc.VerifiableCredential, allowUntrus } } - // Check issuance/expiration time + // Check issuance/expiration time of the credential + // if the signing key is valid at the given time is checked during signature verification validAtNotNil := time.Now() if validAt != nil { validAtNotNil = *validAt @@ -211,12 +142,11 @@ func (v verifier) Verify(credentialToVerify vc.VerifiableCredential, allowUntrus // Check signature if checkSignature { issuerDID, _ := did.ParseDID(credentialToVerify.Issuer.String()) - _, _, err = v.didResolver.Resolve(*issuerDID, &resolver.ResolveMetadata{ResolveTime: validAt, AllowDeactivated: false}) + _, _, err := v.didResolver.Resolve(*issuerDID, &resolver.ResolveMetadata{ResolveTime: validAt, AllowDeactivated: false}) if err != nil { return fmt.Errorf("could not validate issuer: %w", err) } - - return v.Validate(credentialToVerify, validAt) + return v.VerifySignature(credentialToVerify, validAt) } return nil @@ -297,22 +227,14 @@ func (v verifier) VerifyVP(vp vc.VerifiablePresentation, verifyVCs bool, allowUn // doVerifyVP delegates VC verification to the supplied Verifier, to aid unit testing. func (v verifier) doVerifyVP(vcVerifier Verifier, presentation vc.VerifiablePresentation, verifyVCs bool, allowUntrustedVCs bool, validAt *time.Time) ([]vc.VerifiableCredential, error) { - var err error - switch presentation.Format() { - case issuer.JSONLDPresentationFormat: - err = v.validateJSONLDPresentation(presentation, validAt) - case issuer.JWTPresentationFormat: - err = v.validateJWTPresentation(presentation, validAt) - default: - err = errors.New("unsupported presentation proof format") - } + err := v.signatureVerifier.verifyVPSignature(presentation, validAt) if err != nil { return nil, err } if verifyVCs { for _, current := range presentation.VerifiableCredential { - err := vcVerifier.Verify(current, allowUntrustedVCs, true, validAt) + err = vcVerifier.Verify(current, allowUntrustedVCs, true, validAt) if err != nil { return nil, newVerificationError("invalid VC (id=%s): %w", current.ID, err) } @@ -321,80 +243,3 @@ func (v verifier) doVerifyVP(vcVerifier Verifier, presentation vc.VerifiablePres return presentation.VerifiableCredential, nil } - -func (v *verifier) validateJSONLDPresentation(presentation vc.VerifiablePresentation, validAt *time.Time) error { - // Multiple proofs might be supported in the future, when there's an actual use case. - if len(presentation.Proof) != 1 { - return newVerificationError("exactly 1 proof is expected") - } - // Make sure the proofs are LD-proofs - ldProof, err := credential.ParseLDProof(presentation) - if err != nil { - return newVerificationError("unsupported proof type: %w", err) - } - - // Validate signing time - at := timeFunc() - if validAt != nil { - at = *validAt - } - if !ldProof.ValidAt(at, maxSkew) { - return toVerificationError(types.ErrPresentationNotValidAtTime) - } - - // Validate signature - signingKey, err := v.keyResolver.ResolveKeyByID(ldProof.VerificationMethod.String(), validAt, resolver.NutsSigningKeyType) - if err != nil { - return fmt.Errorf("unable to resolve valid signing key: %w", err) - } - signedDocument, err := proof.NewSignedDocument(presentation) - if err != nil { - return newVerificationError("invalid LD-JSON document: %w", err) - } - err = ldProof.Verify(signedDocument.DocumentWithoutProof(), signature.JSONWebSignature2020{ContextLoader: v.jsonldManager.DocumentLoader()}, signingKey) - if err != nil { - return newVerificationError("invalid signature: %w", err) - } - return nil -} - -func (v *verifier) validateJWTPresentation(presentation vc.VerifiablePresentation, at *time.Time) error { - var keyID string - if len(presentation.VerifiableCredential) != 1 { - return errors.New("exactly 1 credential in JWT VP is expected") - } - subjectDID, err := presentation.VerifiableCredential[0].SubjectDID() - if err != nil { - return err - } - _, err = crypto.ParseJWT(presentation.Raw(), func(kid string) (crypt.PublicKey, error) { - keyID = kid - return v.resolveSigningKey(kid, subjectDID.String(), at) - }, jwt.WithClock(jwt.ClockFunc(func() time.Time { - if at == nil { - return time.Now() - } - return *at - })), jwt.WithAcceptableSkew(maxSkew)) - if err != nil { - return fmt.Errorf("unable to validate JWT credential: %w", err) - } - if keyID != "" && strings.Split(keyID, "#")[0] != subjectDID.String() { - return errVerificationMethodNotOfIssuer - } - return nil -} - -func (v *verifier) validateType(credential vc.VerifiableCredential) error { - // VCs must contain 2 types: "VerifiableCredential" and specific type - if len(credential.Type) > 2 { - return errors.New("verifiable credential must list at most 2 types") - } - // "VerifiableCredential" should be one of the types - for _, curr := range credential.Type { - if curr == vc.VerifiableCredentialTypeV1URI() { - return nil - } - } - return fmt.Errorf("verifiable credential does not list '%s' as type", vc.VerifiableCredentialTypeV1URI()) -} diff --git a/vcr/verifier/verifier_test.go b/vcr/verifier/verifier_test.go index 2b18e30c04..c528ea8901 100644 --- a/vcr/verifier/verifier_test.go +++ b/vcr/verifier/verifier_test.go @@ -19,18 +19,10 @@ package verifier import ( - "context" crypt "crypto" - "crypto/ecdsa" - "crypto/elliptic" - "crypto/rand" - "encoding/base64" "encoding/json" "errors" "github.com/lestrrat-go/jwx/v2/jwk" - "github.com/nuts-foundation/nuts-node/audit" - "github.com/nuts-foundation/nuts-node/crypto/storage/spi" - "github.com/nuts-foundation/nuts-node/vdr/didjwk" "github.com/nuts-foundation/nuts-node/vdr/resolver" "github.com/stretchr/testify/require" "os" @@ -58,234 +50,6 @@ func testCredential(t *testing.T) vc.VerifiableCredential { return subject } -func Test_verifier_Validate(t *testing.T) { - const testKID = "did:nuts:CuE3qeFGGLhEAS3gKzhMCeqd1dGa9at5JCbmCfyMU2Ey#sNGDQ3NlOe6Icv0E7_ufviOLG6Y25bSEyS5EbXBgp8Y" - - // load pub key - pke := spi.PublicKeyEntry{} - pkeJSON, _ := os.ReadFile("../test/public.json") - json.Unmarshal(pkeJSON, &pke) - var pk = new(ecdsa.PublicKey) - pke.JWK().Raw(pk) - - now := time.Now() - timeFunc = func() time.Time { - return now - } - defer func() { - timeFunc = time.Now - }() - - t.Run("JSON-LD", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - ctx.keyResolver.EXPECT().ResolveKeyByID(testKID, gomock.Any(), resolver.NutsSigningKeyType).Return(pk, nil) - - err := instance.Validate(testCredential(t), nil) - - assert.NoError(t, err) - }) - t.Run("JWT", func(t *testing.T) { - // Create did:jwk for issuer, and sign credential - keyStore := crypto.NewMemoryCryptoInstance() - key, err := keyStore.New(audit.TestContext(), func(key crypt.PublicKey) (string, error) { - keyAsJWK, _ := jwk.FromRaw(key) - keyJSON, _ := json.Marshal(keyAsJWK) - return "did:jwk:" + base64.RawStdEncoding.EncodeToString(keyJSON) + "#0", nil - }) - require.NoError(t, err) - - template := testCredential(t) - template.Issuer = did.MustParseDIDURL(key.KID()).DID.URI() - - cred, err := vc.CreateJWTVerifiableCredential(audit.TestContext(), template, func(ctx context.Context, claims map[string]interface{}, headers map[string]interface{}) (string, error) { - return keyStore.SignJWT(ctx, claims, headers, key) - }) - require.NoError(t, err) - - t.Run("with kid header", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - ctx.keyResolver.EXPECT().ResolveKeyByID(key.KID(), gomock.Any(), resolver.NutsSigningKeyType).Return(key.Public(), nil) - err = instance.Validate(*cred, nil) - - assert.NoError(t, err) - }) - t.Run("kid header does not match credential issuer", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - cred, err := vc.CreateJWTVerifiableCredential(audit.TestContext(), template, func(ctx context.Context, claims map[string]interface{}, headers map[string]interface{}) (string, error) { - return keyStore.SignJWT(ctx, claims, headers, key) - }) - require.NoError(t, err) - cred.Issuer = ssi.MustParseURI("did:example:test") - - ctx.keyResolver.EXPECT().ResolveKeyByID(key.KID(), gomock.Any(), resolver.NutsSigningKeyType).Return(key.Public(), nil) - err = instance.Validate(*cred, nil) - - assert.ErrorIs(t, err, errVerificationMethodNotOfIssuer) - }) - t.Run("signature invalid", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - realKey, _ := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) - - ctx.keyResolver.EXPECT().ResolveKeyByID(key.KID(), gomock.Any(), resolver.NutsSigningKeyType).Return(realKey.Public(), nil) - err = instance.Validate(*cred, nil) - - assert.EqualError(t, err, "unable to validate JWT credential: could not verify message using any of the signatures or keys") - }) - t.Run("expired token", func(t *testing.T) { - // Credential taken from Sphereon Wallet, expires on Tue Oct 03 2023 - const credentialJSON = `eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCJ9.eyJleHAiOjE2OTYzMDE3MDgsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZmlyc3ROYW1lIjoiSGVsbG8iLCJsYXN0TmFtZSI6IlNwaGVyZW9uIiwiZW1haWwiOiJzcGhlcmVvbkBleGFtcGxlLmNvbSIsInR5cGUiOiJTcGhlcmVvbiBHdWVzdCIsImlkIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEifX0sIkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiZXhwaXJhdGlvbkRhdGUiOiIyMDIzLTEwLTAzVDAyOjU1OjA4LjEzM1oiLCJjcmVkZW50aWFsU3ViamVjdCI6eyJmaXJzdE5hbWUiOiJIZWxsbyIsImxhc3ROYW1lIjoiU3BoZXJlb24iLCJlbWFpbCI6InNwaGVyZW9uQGV4YW1wbGUuY29tIiwidHlwZSI6IlNwaGVyZW9uIEd1ZXN0IiwiaWQiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOa3NpTENKMWMyVWlPaUp6YVdjaUxDSnJkSGtpT2lKRlF5SXNJbU55ZGlJNkluTmxZM0F5TlRack1TSXNJbmdpT2lKak1WZFljemRYTTIxNWMyVlZaazVDY1hONFpGQlhRa2xIYUV0a05GUjZNRXhTTFVacU9FWk5XV0V3SWl3aWVTSTZJbGR0YTBOWWRURjNlWHBhWjBkT04xVjRUbUZ3Y0hGdVQxRmhUMnRYTWtOblQxTnVUMjk1VFVsVWRXTWlmUSJ9LCJpc3N1ZXIiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOaUlzSW5WelpTSTZJbk5wWnlJc0ltdDBlU0k2SWtWRElpd2lZM0oySWpvaVVDMHlOVFlpTENKNElqb2lWRWN5U0RKNE1tUlhXRTR6ZFVOeFduQnhSakY1YzBGUVVWWkVTa1ZPWDBndFEwMTBZbWRxWWkxT1p5SXNJbmtpT2lJNVRUaE9lR1F3VUU0eU1rMDViRkJFZUdSd1JIQnZWRXg2TVRWM1pubGFTbk0yV21oTFNWVktNek00SW4wIiwiaXNzdWFuY2VEYXRlIjoiMjAyMy0wOS0yOVQxMjozMTowOC4xMzNaIiwic3ViIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEiLCJuYmYiOjE2OTU5OTA2NjgsImlzcyI6ImRpZDpqd2s6ZXlKaGJHY2lPaUpGVXpJMU5pSXNJblZ6WlNJNkluTnBaeUlzSW10MGVTSTZJa1ZESWl3aVkzSjJJam9pVUMweU5UWWlMQ0o0SWpvaVZFY3lTREo0TW1SWFdFNHpkVU54V25CeFJqRjVjMEZRVVZaRVNrVk9YMGd0UTAxMFltZHFZaTFPWnlJc0lua2lPaUk1VFRoT2VHUXdVRTR5TWswNWJGQkVlR1J3UkhCdlZFeDZNVFYzWm5sYVNuTTJXbWhMU1ZWS016TTRJbjAifQ.wdhtLXE4jU1C-3YBBpP9-qE-yh1xOZ6lBLJ-0e5_Sa7fnrUHcAaU1n3kN2CeCyTVjtm1Uy3Tl6RzUOM6MjP3vQ` - cred, _ := vc.ParseVerifiableCredential(credentialJSON) - - keyResolver := resolver.DIDKeyResolver{ - Resolver: didjwk.NewResolver(), - } - err := (&verifier{keyResolver: keyResolver}).Validate(*cred, nil) - - assert.EqualError(t, err, "unable to validate JWT credential: \"exp\" not satisfied") - }) - t.Run("without kid header, derived from issuer", func(t *testing.T) { - // Credential taken from Sphereon Wallet - const credentialJSON = `eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCJ9.eyJleHAiOjE2OTYzMDE3MDgsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZmlyc3ROYW1lIjoiSGVsbG8iLCJsYXN0TmFtZSI6IlNwaGVyZW9uIiwiZW1haWwiOiJzcGhlcmVvbkBleGFtcGxlLmNvbSIsInR5cGUiOiJTcGhlcmVvbiBHdWVzdCIsImlkIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEifX0sIkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIl0sInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJHdWVzdENyZWRlbnRpYWwiXSwiZXhwaXJhdGlvbkRhdGUiOiIyMDIzLTEwLTAzVDAyOjU1OjA4LjEzM1oiLCJjcmVkZW50aWFsU3ViamVjdCI6eyJmaXJzdE5hbWUiOiJIZWxsbyIsImxhc3ROYW1lIjoiU3BoZXJlb24iLCJlbWFpbCI6InNwaGVyZW9uQGV4YW1wbGUuY29tIiwidHlwZSI6IlNwaGVyZW9uIEd1ZXN0IiwiaWQiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOa3NpTENKMWMyVWlPaUp6YVdjaUxDSnJkSGtpT2lKRlF5SXNJbU55ZGlJNkluTmxZM0F5TlRack1TSXNJbmdpT2lKak1WZFljemRYTTIxNWMyVlZaazVDY1hONFpGQlhRa2xIYUV0a05GUjZNRXhTTFVacU9FWk5XV0V3SWl3aWVTSTZJbGR0YTBOWWRURjNlWHBhWjBkT04xVjRUbUZ3Y0hGdVQxRmhUMnRYTWtOblQxTnVUMjk1VFVsVWRXTWlmUSJ9LCJpc3N1ZXIiOiJkaWQ6andrOmV5SmhiR2NpT2lKRlV6STFOaUlzSW5WelpTSTZJbk5wWnlJc0ltdDBlU0k2SWtWRElpd2lZM0oySWpvaVVDMHlOVFlpTENKNElqb2lWRWN5U0RKNE1tUlhXRTR6ZFVOeFduQnhSakY1YzBGUVVWWkVTa1ZPWDBndFEwMTBZbWRxWWkxT1p5SXNJbmtpT2lJNVRUaE9lR1F3VUU0eU1rMDViRkJFZUdSd1JIQnZWRXg2TVRWM1pubGFTbk0yV21oTFNWVktNek00SW4wIiwiaXNzdWFuY2VEYXRlIjoiMjAyMy0wOS0yOVQxMjozMTowOC4xMzNaIiwic3ViIjoiZGlkOmp3azpleUpoYkdjaU9pSkZVekkxTmtzaUxDSjFjMlVpT2lKemFXY2lMQ0pyZEhraU9pSkZReUlzSW1OeWRpSTZJbk5sWTNBeU5UWnJNU0lzSW5naU9pSmpNVmRZY3pkWE0yMTVjMlZWWms1Q2NYTjRaRkJYUWtsSGFFdGtORlI2TUV4U0xVWnFPRVpOV1dFd0lpd2llU0k2SWxkdGEwTllkVEYzZVhwYVowZE9OMVY0VG1Gd2NIRnVUMUZoVDJ0WE1rTm5UMU51VDI5NVRVbFVkV01pZlEiLCJuYmYiOjE2OTU5OTA2NjgsImlzcyI6ImRpZDpqd2s6ZXlKaGJHY2lPaUpGVXpJMU5pSXNJblZ6WlNJNkluTnBaeUlzSW10MGVTSTZJa1ZESWl3aVkzSjJJam9pVUMweU5UWWlMQ0o0SWpvaVZFY3lTREo0TW1SWFdFNHpkVU54V25CeFJqRjVjMEZRVVZaRVNrVk9YMGd0UTAxMFltZHFZaTFPWnlJc0lua2lPaUk1VFRoT2VHUXdVRTR5TWswNWJGQkVlR1J3UkhCdlZFeDZNVFYzWm5sYVNuTTJXbWhMU1ZWS016TTRJbjAifQ.wdhtLXE4jU1C-3YBBpP9-qE-yh1xOZ6lBLJ-0e5_Sa7fnrUHcAaU1n3kN2CeCyTVjtm1Uy3Tl6RzUOM6MjP3vQ` - cred, _ := vc.ParseVerifiableCredential(credentialJSON) - - keyResolver := resolver.DIDKeyResolver{ - Resolver: didjwk.NewResolver(), - } - validAt := time.Date(2023, 9, 30, 0, 0, 0, 0, time.UTC) - err := (&verifier{keyResolver: keyResolver}).Validate(*cred, &validAt) - - assert.NoError(t, err) - }) - }) - - t.Run("type", func(t *testing.T) { - t.Run("incorrect number of types", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - err := instance.Validate(vc.VerifiableCredential{Type: []ssi.URI{vc.VerifiableCredentialTypeV1URI(), ssi.MustParseURI("a"), ssi.MustParseURI("b")}}, nil) - - assert.EqualError(t, err, "verifiable credential must list at most 2 types") - }) - t.Run("does not contain v1 context", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - err := instance.Validate(vc.VerifiableCredential{Type: []ssi.URI{ssi.MustParseURI("foo"), ssi.MustParseURI("bar")}}, nil) - - assert.EqualError(t, err, "verifiable credential does not list 'VerifiableCredential' as type") - }) - }) - - t.Run("error - invalid vm", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - vc2 := testCredential(t) - pr := make([]vc.JSONWebSignature2020Proof, 0) - _ = vc2.UnmarshalProofValue(&pr) - u := ssi.MustParseURI(vc2.Issuer.String() + "2") - pr[0].VerificationMethod = u - vc2.Proof = []interface{}{pr[0]} - - err := instance.Validate(vc2, nil) - - assert.Error(t, err) - assert.ErrorIs(t, err, errVerificationMethodNotOfIssuer) - }) - - t.Run("error - wrong hashed payload", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - vc2 := testCredential(t) - issuanceDate := time.Now() - vc2.IssuanceDate = &issuanceDate - - ctx.keyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) - - err := instance.Validate(vc2, nil) - - assert.ErrorContains(t, err, "failed to verify signature") - }) - - t.Run("error - wrong hashed proof", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - vc2 := testCredential(t) - pr := make([]vc.JSONWebSignature2020Proof, 0) - vc2.UnmarshalProofValue(&pr) - pr[0].Created = time.Now() - vc2.Proof = []interface{}{pr[0]} - - ctx.keyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) - - err := instance.Validate(vc2, nil) - - assert.ErrorContains(t, err, "failed to verify signature") - }) - - t.Run("error - no proof", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - vc2 := testCredential(t) - vc2.Proof = []interface{}{} - - err := instance.Validate(vc2, nil) - - assert.ErrorContains(t, err, "unable to extract ldproof from signed document: json: cannot unmarshal array into Go value of type proof.LDProof") - }) - - t.Run("error - wrong jws in proof", func(t *testing.T) { - ctx := newMockContext(t) - ctx.keyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) - instance := ctx.verifier - vc2 := testCredential(t) - pr := make([]vc.JSONWebSignature2020Proof, 0) - vc2.UnmarshalProofValue(&pr) - pr[0].Jws = "" - vc2.Proof = []interface{}{pr[0]} - - err := instance.Validate(vc2, nil) - - assert.ErrorContains(t, err, "invalid 'jws' value in proof") - }) - - t.Run("error - wrong base64 encoding in jws", func(t *testing.T) { - ctx := newMockContext(t) - ctx.keyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(pk, nil) - instance := ctx.verifier - vc2 := testCredential(t) - pr := make([]vc.JSONWebSignature2020Proof, 0) - vc2.UnmarshalProofValue(&pr) - pr[0].Jws = "abac..ab//" - vc2.Proof = []interface{}{pr[0]} - - err := instance.Validate(vc2, nil) - - assert.ErrorContains(t, err, "illegal base64 data") - }) - - t.Run("error - resolving key", func(t *testing.T) { - ctx := newMockContext(t) - instance := ctx.verifier - - ctx.keyResolver.EXPECT().ResolveKeyByID(testKID, nil, resolver.NutsSigningKeyType).Return(nil, errors.New("b00m!")) - - err := instance.Validate(testCredential(t), nil) - - assert.Error(t, err) - }) - -} - func TestVerifier_Verify(t *testing.T) { const testKID = "did:nuts:CuE3qeFGGLhEAS3gKzhMCeqd1dGa9at5JCbmCfyMU2Ey#sNGDQ3NlOe6Icv0E7_ufviOLG6Y25bSEyS5EbXBgp8Y" @@ -320,9 +84,9 @@ func TestVerifier_Verify(t *testing.T) { ctx.keyResolver.EXPECT().ResolveKeyByID(testKID, gomock.Any(), resolver.NutsSigningKeyType).Return(nil, errors.New("not found")) at := time.Now() - err := instance.Validate(subject, &at) + err := instance.VerifySignature(subject, &at) - assert.EqualError(t, err, "unable to resolve valid signing key at given time: not found") + assert.EqualError(t, err, "unable to resolve valid signing key: not found") }) // Verify calls other verifiers / validators. @@ -340,7 +104,7 @@ func TestVerifier_Verify(t *testing.T) { validationErr := ctx.verifier.Verify(vc, true, true, nil) - assert.EqualError(t, validationErr, "unable to resolve signing key: key not found in DID document") + assert.EqualError(t, validationErr, "unable to resolve valid signing key: key not found in DID document") }) t.Run("fails when controller or issuer is deactivated", func(t *testing.T) { @@ -434,6 +198,17 @@ func TestVerifier_Verify(t *testing.T) { }) }) + + t.Run("incorrect number of types", func(t *testing.T) { + ctx := newMockContext(t) + instance := ctx.verifier + testCred := testCredential(t) + testCred.Type = []ssi.URI{vc.VerifiableCredentialTypeV1URI(), ssi.MustParseURI("a"), ssi.MustParseURI("b")} + + err := instance.Verify(testCred, true, false, nil) + + assert.EqualError(t, err, "verifiable credential must list at most 2 types") + }) } func Test_verifier_CheckAndStoreRevocation(t *testing.T) { @@ -558,31 +333,32 @@ func TestVerifier_VerifyVP(t *testing.T) { assert.EqualError(t, err, "unable to validate JWT credential: \"exp\" not satisfied") assert.Empty(t, vcs) }) - t.Run("VP signer != VC credentialSubject.id", func(t *testing.T) { - // This VP was produced by a Sphereon Wallet, using did:key. The signer of the VP is a did:key, - // but the holder of the contained credential is a did:jwt. So the presenter is not the holder. Weird? - const rawVP = `eyJraWQiOiJkaWQ6a2V5Ono2TWtzRXl4NmQ1cEIxZWtvYVZtYUdzaWJiY1lIRTlWeHg3VjEzUFNxUHd4WVJ6TCN6Nk1rc0V5eDZkNXBCMWVrb2FWbWFHc2liYmNZSEU5Vnh4N1YxM1BTcVB3eFlSekwiLCJhbGciOiJFZERTQSIsInR5cCI6IkpXVCJ9.eyJ2cCI6eyJAY29udGV4dCI6WyJodHRwczovL3d3dy53My5vcmcvMjAxOC9jcmVkZW50aWFscy92MSIsImh0dHBzOi8vaWRlbnRpdHkuZm91bmRhdGlvbi9wcmVzZW50YXRpb24tZXhjaGFuZ2Uvc3VibWlzc2lvbi92MSJdLCJ0eXBlIjpbIlZlcmlmaWFibGVQcmVzZW50YXRpb24iLCJQcmVzZW50YXRpb25TdWJtaXNzaW9uIl0sInZlcmlmaWFibGVDcmVkZW50aWFsIjpbImV5SmhiR2NpT2lKRlV6STFOaUlzSW5SNWNDSTZJa3BYVkNKOS5leUpsZUhBaU9qRTJPVFl6TURFM01EZ3NJblpqSWpwN0lrQmpiMjUwWlhoMElqcGJJbWgwZEhCek9pOHZkM2QzTG5jekxtOXlaeTh5TURFNEwyTnlaV1JsYm5ScFlXeHpMM1l4SWwwc0luUjVjR1VpT2xzaVZtVnlhV1pwWVdKc1pVTnlaV1JsYm5ScFlXd2lMQ0pIZFdWemRFTnlaV1JsYm5ScFlXd2lYU3dpWTNKbFpHVnVkR2xoYkZOMVltcGxZM1FpT25zaVptbHljM1JPWVcxbElqb2lTR1ZzYkc4aUxDSnNZWE4wVG1GdFpTSTZJbE53YUdWeVpXOXVJaXdpWlcxaGFXd2lPaUp6Y0dobGNtVnZia0JsZUdGdGNHeGxMbU52YlNJc0luUjVjR1VpT2lKVGNHaGxjbVZ2YmlCSGRXVnpkQ0lzSW1sa0lqb2laR2xrT21wM2F6cGxlVXBvWWtkamFVOXBTa1pWZWtreFRtdHphVXhEU2pGak1sVnBUMmxLZW1GWFkybE1RMHB5WkVocmFVOXBTa1pSZVVselNXMU9lV1JwU1RaSmJrNXNXVE5CZVU1VVduSk5VMGx6U1c1bmFVOXBTbXBOVm1SWlkzcGtXRTB5TVRWak1sWldXbXMxUTJOWVRqUmFSa0pZVVd0c1NHRkZkR3RPUmxJMlRVVjRVMHhWV25GUFJWcE9WMWRGZDBscGQybGxVMGsyU1d4a2RHRXdUbGxrVkVZelpWaHdZVm93WkU5T01WWTBWRzFHZDJOSVJuVlVNVVpvVkRKMFdFMXJUbTVVTVU1MVZESTVOVlJWYkZWa1YwMXBabEVpZlgwc0lrQmpiMjUwWlhoMElqcGJJbWgwZEhCek9pOHZkM2QzTG5jekxtOXlaeTh5TURFNEwyTnlaV1JsYm5ScFlXeHpMM1l4SWwwc0luUjVjR1VpT2xzaVZtVnlhV1pwWVdKc1pVTnlaV1JsYm5ScFlXd2lMQ0pIZFdWemRFTnlaV1JsYm5ScFlXd2lYU3dpWlhod2FYSmhkR2x2YmtSaGRHVWlPaUl5TURJekxURXdMVEF6VkRBeU9qVTFPakE0TGpFek0xb2lMQ0pqY21Wa1pXNTBhV0ZzVTNWaWFtVmpkQ0k2ZXlKbWFYSnpkRTVoYldVaU9pSklaV3hzYnlJc0lteGhjM1JPWVcxbElqb2lVM0JvWlhKbGIyNGlMQ0psYldGcGJDSTZJbk53YUdWeVpXOXVRR1Y0WVcxd2JHVXVZMjl0SWl3aWRIbHdaU0k2SWxOd2FHVnlaVzl1SUVkMVpYTjBJaXdpYVdRaU9pSmthV1E2YW5kck9tVjVTbWhpUjJOcFQybEtSbFY2U1RGT2EzTnBURU5LTVdNeVZXbFBhVXA2WVZkamFVeERTbkprU0d0cFQybEtSbEY1U1hOSmJVNTVaR2xKTmtsdVRteFpNMEY1VGxSYWNrMVRTWE5KYm1kcFQybEthazFXWkZsamVtUllUVEl4TldNeVZsWmFhelZEWTFoT05GcEdRbGhSYTJ4SVlVVjBhMDVHVWpaTlJYaFRURlZhY1U5RldrNVhWMFYzU1dsM2FXVlRTVFpKYkdSMFlUQk9XV1JVUmpObFdIQmhXakJrVDA0eFZqUlViVVozWTBoR2RWUXhSbWhVTW5SWVRXdE9ibFF4VG5WVU1qazFWRlZzVldSWFRXbG1VU0o5TENKcGMzTjFaWElpT2lKa2FXUTZhbmRyT21WNVNtaGlSMk5wVDJsS1JsVjZTVEZPYVVselNXNVdlbHBUU1RaSmJrNXdXbmxKYzBsdGREQmxVMGsyU1d0V1JFbHBkMmxaTTBveVNXcHZhVlZETUhsT1ZGbHBURU5LTkVscWIybFdSV041VTBSS05FMXRVbGhYUlRSNlpGVk9lRmR1UW5oU2FrWTFZekJHVVZWV1drVlRhMVpQV0RCbmRGRXdNVEJaYldSeFdXa3hUMXA1U1hOSmJtdHBUMmxKTlZSVWFFOWxSMUYzVlVVMGVVMXJNRFZpUmtKRlpVZFNkMUpJUW5aV1JYZzJUVlJXTTFwdWJHRlRiazB5VjIxb1RGTldWa3ROZWswMFNXNHdJaXdpYVhOemRXRnVZMlZFWVhSbElqb2lNakF5TXkwd09TMHlPVlF4TWpvek1Ub3dPQzR4TXpOYUlpd2ljM1ZpSWpvaVpHbGtPbXAzYXpwbGVVcG9Za2RqYVU5cFNrWlZla2t4VG10emFVeERTakZqTWxWcFQybEtlbUZYWTJsTVEwcHlaRWhyYVU5cFNrWlJlVWx6U1cxT2VXUnBTVFpKYms1c1dUTkJlVTVVV25KTlUwbHpTVzVuYVU5cFNtcE5WbVJaWTNwa1dFMHlNVFZqTWxaV1dtczFRMk5ZVGpSYVJrSllVV3RzU0dGRmRHdE9SbEkyVFVWNFUweFZXbkZQUlZwT1YxZEZkMGxwZDJsbFUwazJTV3hrZEdFd1RsbGtWRVl6WlZod1lWb3daRTlPTVZZMFZHMUdkMk5JUm5WVU1VWm9WREowV0UxclRtNVVNVTUxVkRJNU5WUlZiRlZrVjAxcFpsRWlMQ0p1WW1ZaU9qRTJPVFU1T1RBMk5qZ3NJbWx6Y3lJNkltUnBaRHBxZDJzNlpYbEthR0pIWTJsUGFVcEdWWHBKTVU1cFNYTkpibFo2V2xOSk5rbHVUbkJhZVVselNXMTBNR1ZUU1RaSmExWkVTV2wzYVZrelNqSkphbTlwVlVNd2VVNVVXV2xNUTBvMFNXcHZhVlpGWTNsVFJFbzBUVzFTV0ZkRk5IcGtWVTU0VjI1Q2VGSnFSalZqTUVaUlZWWmFSVk5yVms5WU1HZDBVVEF4TUZsdFpIRlphVEZQV25sSmMwbHVhMmxQYVVrMVZGUm9UMlZIVVhkVlJUUjVUV3N3TldKR1FrVmxSMUozVWtoQ2RsWkZlRFpOVkZZeldtNXNZVk51VFRKWGJXaE1VMVpXUzAxNlRUUkpiakFpZlEud2RodExYRTRqVTFDLTNZQkJwUDktcUUteWgxeE9aNmxCTEotMGU1X1NhN2ZuclVIY0FhVTFuM2tOMkNlQ3lUVmp0bTFVeTNUbDZSelVPTTZNalAzdlEiXX0sInByZXNlbnRhdGlvbl9zdWJtaXNzaW9uIjp7ImlkIjoidG9DdGp5Y0V3QlZCWVBsbktBQTZGIiwiZGVmaW5pdGlvbl9pZCI6InNwaGVyZW9uIiwiZGVzY3JpcHRvcl9tYXAiOlt7ImlkIjoiNGNlN2FmZjEtMDIzNC00ZjM1LTlkMjEtMjUxNjY4YTYwOTUwIiwiZm9ybWF0Ijoiand0X3ZjIiwicGF0aCI6IiQudmVyaWZpYWJsZUNyZWRlbnRpYWxbMF0ifV19LCJuYmYiOjE2OTU5OTU2MzYsImlzcyI6ImRpZDprZXk6ejZNa3NFeXg2ZDVwQjFla29hVm1hR3NpYmJjWUhFOVZ4eDdWMTNQU3FQd3hZUnpMIn0.w3guHX-pmxJGGn5dGSSIKSba9xywnOutDk-l3tc_bpgHEOSbcR1mmmCqX5sSlZM_G0hgAbgpIv_YYI5iQNIfCw` - const keyID = "did:key:z6MksEyx6d5pB1ekoaVmaGsibbcYHE9Vxx7V13PSqPwxYRzL#z6MksEyx6d5pB1ekoaVmaGsibbcYHE9Vxx7V13PSqPwxYRzL" - keyAsJWK, err := jwk.ParseKey([]byte(`{ - "kty": "OKP", - "crv": "Ed25519", - "x": "vgLDESnU0TIlW-PmajyrvSlk9VysAsRkSYiEPBELj-U" - }`)) - require.NoError(t, err) - require.NoError(t, keyAsJWK.Set("kid", keyID)) - publicKey, err := keyAsJWK.PublicKey() - require.NoError(t, err) - - presentation, err := vc.ParseVerifiablePresentation(rawVP) - require.NoError(t, err) - ctx := newMockContext(t) - ctx.keyResolver.EXPECT().ResolveKeyByID(keyID, gomock.Any(), resolver.NutsSigningKeyType).Return(publicKey, nil) - - vcs, err := ctx.verifier.VerifyVP(*presentation, false, false, nil) - assert.EqualError(t, err, "verification method is not of issuer") - assert.Empty(t, vcs) - }) + //t.Run("VP signer != VC credentialSubject.id", func(t *testing.T) { + // // This VP was produced by a Sphereon Wallet, using did:key. The signer of the VP is a did:key, + // // but the holder of the contained credential is a did:jwt. So the presenter is not the holder. Weird? + // const rawVP = `eyJraWQiOiJkaWQ6a2V5Ono2TWtzRXl4NmQ1cEIxZWtvYVZtYUdzaWJiY1lIRTlWeHg3VjEzUFNxUHd4WVJ6TCN6Nk1rc0V5eDZkNXBCMWVrb2FWbWFHc2liYmNZSEU5Vnh4N1YxM1BTcVB3eFlSekwiLCJhbGciOiJFZERTQSIsInR5cCI6IkpXVCJ9.eyJ2cCI6eyJAY29udGV4dCI6WyJodHRwczovL3d3dy53My5vcmcvMjAxOC9jcmVkZW50aWFscy92MSIsImh0dHBzOi8vaWRlbnRpdHkuZm91bmRhdGlvbi9wcmVzZW50YXRpb24tZXhjaGFuZ2Uvc3VibWlzc2lvbi92MSJdLCJ0eXBlIjpbIlZlcmlmaWFibGVQcmVzZW50YXRpb24iLCJQcmVzZW50YXRpb25TdWJtaXNzaW9uIl0sInZlcmlmaWFibGVDcmVkZW50aWFsIjpbImV5SmhiR2NpT2lKRlV6STFOaUlzSW5SNWNDSTZJa3BYVkNKOS5leUpsZUhBaU9qRTJPVFl6TURFM01EZ3NJblpqSWpwN0lrQmpiMjUwWlhoMElqcGJJbWgwZEhCek9pOHZkM2QzTG5jekxtOXlaeTh5TURFNEwyTnlaV1JsYm5ScFlXeHpMM1l4SWwwc0luUjVjR1VpT2xzaVZtVnlhV1pwWVdKc1pVTnlaV1JsYm5ScFlXd2lMQ0pIZFdWemRFTnlaV1JsYm5ScFlXd2lYU3dpWTNKbFpHVnVkR2xoYkZOMVltcGxZM1FpT25zaVptbHljM1JPWVcxbElqb2lTR1ZzYkc4aUxDSnNZWE4wVG1GdFpTSTZJbE53YUdWeVpXOXVJaXdpWlcxaGFXd2lPaUp6Y0dobGNtVnZia0JsZUdGdGNHeGxMbU52YlNJc0luUjVjR1VpT2lKVGNHaGxjbVZ2YmlCSGRXVnpkQ0lzSW1sa0lqb2laR2xrT21wM2F6cGxlVXBvWWtkamFVOXBTa1pWZWtreFRtdHphVXhEU2pGak1sVnBUMmxLZW1GWFkybE1RMHB5WkVocmFVOXBTa1pSZVVselNXMU9lV1JwU1RaSmJrNXNXVE5CZVU1VVduSk5VMGx6U1c1bmFVOXBTbXBOVm1SWlkzcGtXRTB5TVRWak1sWldXbXMxUTJOWVRqUmFSa0pZVVd0c1NHRkZkR3RPUmxJMlRVVjRVMHhWV25GUFJWcE9WMWRGZDBscGQybGxVMGsyU1d4a2RHRXdUbGxrVkVZelpWaHdZVm93WkU5T01WWTBWRzFHZDJOSVJuVlVNVVpvVkRKMFdFMXJUbTVVTVU1MVZESTVOVlJWYkZWa1YwMXBabEVpZlgwc0lrQmpiMjUwWlhoMElqcGJJbWgwZEhCek9pOHZkM2QzTG5jekxtOXlaeTh5TURFNEwyTnlaV1JsYm5ScFlXeHpMM1l4SWwwc0luUjVjR1VpT2xzaVZtVnlhV1pwWVdKc1pVTnlaV1JsYm5ScFlXd2lMQ0pIZFdWemRFTnlaV1JsYm5ScFlXd2lYU3dpWlhod2FYSmhkR2x2YmtSaGRHVWlPaUl5TURJekxURXdMVEF6VkRBeU9qVTFPakE0TGpFek0xb2lMQ0pqY21Wa1pXNTBhV0ZzVTNWaWFtVmpkQ0k2ZXlKbWFYSnpkRTVoYldVaU9pSklaV3hzYnlJc0lteGhjM1JPWVcxbElqb2lVM0JvWlhKbGIyNGlMQ0psYldGcGJDSTZJbk53YUdWeVpXOXVRR1Y0WVcxd2JHVXVZMjl0SWl3aWRIbHdaU0k2SWxOd2FHVnlaVzl1SUVkMVpYTjBJaXdpYVdRaU9pSmthV1E2YW5kck9tVjVTbWhpUjJOcFQybEtSbFY2U1RGT2EzTnBURU5LTVdNeVZXbFBhVXA2WVZkamFVeERTbkprU0d0cFQybEtSbEY1U1hOSmJVNTVaR2xKTmtsdVRteFpNMEY1VGxSYWNrMVRTWE5KYm1kcFQybEthazFXWkZsamVtUllUVEl4TldNeVZsWmFhelZEWTFoT05GcEdRbGhSYTJ4SVlVVjBhMDVHVWpaTlJYaFRURlZhY1U5RldrNVhWMFYzU1dsM2FXVlRTVFpKYkdSMFlUQk9XV1JVUmpObFdIQmhXakJrVDA0eFZqUlViVVozWTBoR2RWUXhSbWhVTW5SWVRXdE9ibFF4VG5WVU1qazFWRlZzVldSWFRXbG1VU0o5TENKcGMzTjFaWElpT2lKa2FXUTZhbmRyT21WNVNtaGlSMk5wVDJsS1JsVjZTVEZPYVVselNXNVdlbHBUU1RaSmJrNXdXbmxKYzBsdGREQmxVMGsyU1d0V1JFbHBkMmxaTTBveVNXcHZhVlZETUhsT1ZGbHBURU5LTkVscWIybFdSV041VTBSS05FMXRVbGhYUlRSNlpGVk9lRmR1UW5oU2FrWTFZekJHVVZWV1drVlRhMVpQV0RCbmRGRXdNVEJaYldSeFdXa3hUMXA1U1hOSmJtdHBUMmxKTlZSVWFFOWxSMUYzVlVVMGVVMXJNRFZpUmtKRlpVZFNkMUpJUW5aV1JYZzJUVlJXTTFwdWJHRlRiazB5VjIxb1RGTldWa3ROZWswMFNXNHdJaXdpYVhOemRXRnVZMlZFWVhSbElqb2lNakF5TXkwd09TMHlPVlF4TWpvek1Ub3dPQzR4TXpOYUlpd2ljM1ZpSWpvaVpHbGtPbXAzYXpwbGVVcG9Za2RqYVU5cFNrWlZla2t4VG10emFVeERTakZqTWxWcFQybEtlbUZYWTJsTVEwcHlaRWhyYVU5cFNrWlJlVWx6U1cxT2VXUnBTVFpKYms1c1dUTkJlVTVVV25KTlUwbHpTVzVuYVU5cFNtcE5WbVJaWTNwa1dFMHlNVFZqTWxaV1dtczFRMk5ZVGpSYVJrSllVV3RzU0dGRmRHdE9SbEkyVFVWNFUweFZXbkZQUlZwT1YxZEZkMGxwZDJsbFUwazJTV3hrZEdFd1RsbGtWRVl6WlZod1lWb3daRTlPTVZZMFZHMUdkMk5JUm5WVU1VWm9WREowV0UxclRtNVVNVTUxVkRJNU5WUlZiRlZrVjAxcFpsRWlMQ0p1WW1ZaU9qRTJPVFU1T1RBMk5qZ3NJbWx6Y3lJNkltUnBaRHBxZDJzNlpYbEthR0pIWTJsUGFVcEdWWHBKTVU1cFNYTkpibFo2V2xOSk5rbHVUbkJhZVVselNXMTBNR1ZUU1RaSmExWkVTV2wzYVZrelNqSkphbTlwVlVNd2VVNVVXV2xNUTBvMFNXcHZhVlpGWTNsVFJFbzBUVzFTV0ZkRk5IcGtWVTU0VjI1Q2VGSnFSalZqTUVaUlZWWmFSVk5yVms5WU1HZDBVVEF4TUZsdFpIRlphVEZQV25sSmMwbHVhMmxQYVVrMVZGUm9UMlZIVVhkVlJUUjVUV3N3TldKR1FrVmxSMUozVWtoQ2RsWkZlRFpOVkZZeldtNXNZVk51VFRKWGJXaE1VMVpXUzAxNlRUUkpiakFpZlEud2RodExYRTRqVTFDLTNZQkJwUDktcUUteWgxeE9aNmxCTEotMGU1X1NhN2ZuclVIY0FhVTFuM2tOMkNlQ3lUVmp0bTFVeTNUbDZSelVPTTZNalAzdlEiXX0sInByZXNlbnRhdGlvbl9zdWJtaXNzaW9uIjp7ImlkIjoidG9DdGp5Y0V3QlZCWVBsbktBQTZGIiwiZGVmaW5pdGlvbl9pZCI6InNwaGVyZW9uIiwiZGVzY3JpcHRvcl9tYXAiOlt7ImlkIjoiNGNlN2FmZjEtMDIzNC00ZjM1LTlkMjEtMjUxNjY4YTYwOTUwIiwiZm9ybWF0Ijoiand0X3ZjIiwicGF0aCI6IiQudmVyaWZpYWJsZUNyZWRlbnRpYWxbMF0ifV19LCJuYmYiOjE2OTU5OTU2MzYsImlzcyI6ImRpZDprZXk6ejZNa3NFeXg2ZDVwQjFla29hVm1hR3NpYmJjWUhFOVZ4eDdWMTNQU3FQd3hZUnpMIn0.w3guHX-pmxJGGn5dGSSIKSba9xywnOutDk-l3tc_bpgHEOSbcR1mmmCqX5sSlZM_G0hgAbgpIv_YYI5iQNIfCw` + // const keyID = "did:key:z6MksEyx6d5pB1ekoaVmaGsibbcYHE9Vxx7V13PSqPwxYRzL#z6MksEyx6d5pB1ekoaVmaGsibbcYHE9Vxx7V13PSqPwxYRzL" + // keyAsJWK, err := jwk.ParseKey([]byte(`{ + // "kty": "OKP", + // "crv": "Ed25519", + // "x": "vgLDESnU0TIlW-PmajyrvSlk9VysAsRkSYiEPBELj-U" + //}`)) + // require.NoError(t, err) + // require.NoError(t, keyAsJWK.Set("kid", keyID)) + // publicKey, err := keyAsJWK.PublicKey() + // require.NoError(t, err) + // + // presentation, err := vc.ParseVerifiablePresentation(rawVP) + // require.NoError(t, err) + // ctx := newMockContext(t) + // ctx.keyResolver.EXPECT().ResolveKeyByID(keyID, gomock.Any(), resolver.NutsSigningKeyType).Return(publicKey, nil) + // + // vcs, err := ctx.verifier.VerifyVP(*presentation, false, false, nil) + // + // assert.EqualError(t, err, "verification method is not of issuer") + // assert.Empty(t, vcs) + //}) }) t.Run("JSONLD", func(t *testing.T) { rawVP := `{ @@ -744,7 +520,7 @@ func TestVerifier_VerifyVP(t *testing.T) { vcs, err := ctx.verifier.VerifyVP(vp, false, false, validAt) - assert.EqualError(t, err, "verification error: unsupported proof type: invalid LD-proof for presentation: json: cannot unmarshal string into Go value of type proof.LDProof") + assert.EqualError(t, err, "verification error: invalid LD-proof for presentation: json: cannot unmarshal string into Go value of type proof.LDProof") assert.Empty(t, vcs) }) t.Run("error - no proof", func(t *testing.T) { @@ -758,7 +534,7 @@ func TestVerifier_VerifyVP(t *testing.T) { vcs, err := ctx.verifier.VerifyVP(vp, false, false, validAt) - assert.EqualError(t, err, "verification error: exactly 1 proof is expected") + assert.EqualError(t, err, "verification error: presentation should have exactly 1 proof, got 0") assert.Empty(t, vcs) }) }) @@ -843,6 +619,11 @@ func newMockContext(t *testing.T) mockContext { verifierStore := NewMockStore(ctrl) trustConfig := trust.NewConfig(path.Join(io.TestDirectory(t), "trust.yaml")) verifier := NewVerifier(verifierStore, didResolver, keyResolver, jsonldManager, trustConfig).(*verifier) + sv := signatureVerifier{ + keyResolver: keyResolver, + jsonldManager: jsonldManager, + } + verifier.signatureVerifier = sv return mockContext{ ctrl: ctrl, verifier: verifier,