diff --git a/.golangci.yml b/.golangci.yml index d7cae30..7ff72de 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -71,12 +71,11 @@ linters-settings: allow-cuddle-declarations: false godot: check-all: false - gomoddirectives: - replace-local: true linters: enable-all: true disable: + - gomoddirectives - nolintlint - varnamelen - tenv diff --git a/dataintegrity/suite/ecdsa2019/ecdsa2019.go b/dataintegrity/suite/ecdsa2019/ecdsa2019.go index a5f0155..b4d8fcf 100644 --- a/dataintegrity/suite/ecdsa2019/ecdsa2019.go +++ b/dataintegrity/suite/ecdsa2019/ecdsa2019.go @@ -311,19 +311,19 @@ func canonicalize(data map[string]interface{}, loader ld.DocumentLoader) ([]byte return out, nil } -func hashData(transformedDoc, confData []byte, h hash.Hash) []byte { - h.Write(transformedDoc) +func hashData(docData, proofData []byte, h hash.Hash) []byte { + h.Write(docData) docHash := h.Sum(nil) h.Reset() - h.Write(confData) - result := h.Sum(docHash) + h.Write(proofData) + proofHash := h.Sum(nil) - return result + return append(proofHash, docHash...) } func proofConfig(docCtx interface{}, opts *models.ProofOptions) map[string]interface{} { - return map[string]interface{}{ + proof := map[string]interface{}{ ldCtxKey: docCtx, "type": models.DataIntegrityProof, "cryptosuite": opts.SuiteType, @@ -331,6 +331,15 @@ func proofConfig(docCtx interface{}, opts *models.ProofOptions) map[string]inter "created": opts.Created.Format(models.DateTimeFormat), "proofPurpose": opts.Purpose, } + + if opts.Challenge != "" { + proof["challenge"] = opts.Challenge + } + if opts.Domain != "" { + proof["domain"] = opts.Domain + } + + return proof } func sign(sigBase []byte, key *jwk.JWK, signerGetter SignerGetter) ([]byte, error) { diff --git a/dataintegrity/suite/eddsa2022/eddsa2022.go b/dataintegrity/suite/eddsa2022/eddsa2022.go index 7211e0e..5e7b2a3 100644 --- a/dataintegrity/suite/eddsa2022/eddsa2022.go +++ b/dataintegrity/suite/eddsa2022/eddsa2022.go @@ -31,6 +31,9 @@ const ( // implementing eddsa signatures with RDF canonicalization as per this // spec:https://w3c.github.io/vc-di-eddsa/#verify-proof-eddsa-rdfc-2022 SuiteType = "eddsa-rdfc-2022" + + // SuiteType2 "eddsa-2022" is the data integrity Type identifier for the suite. Alias (vc playground). + SuiteType2 = "eddsa-2022" ) // SignerGetter returns a Signer, which must sign with the private key matching @@ -126,7 +129,7 @@ func (i initializer) Verifier() (suite.Verifier, error) { // Type private, implements suite.SignerInitializer and // suite.VerifierInitializer. func (i initializer) Type() []string { - return []string{SuiteType} + return []string{SuiteType, SuiteType2} } // SignerInitializerOptions provides options for a SignerInitializer. @@ -200,6 +203,7 @@ func (s *Suite) CreateProof(doc []byte, opts *models.ProofOptions) (*models.Proo return p, nil } +// nolint:gocyclo func (s *Suite) transformAndHash(doc []byte, opts *models.ProofOptions) ([]byte, *pubkey.PublicKey, Verifier, error) { docData := make(map[string]interface{}) @@ -208,11 +212,6 @@ func (s *Suite) transformAndHash(doc []byte, opts *models.ProofOptions) ([]byte, return nil, nil, nil, fmt.Errorf("eddsa-2022 suite expects JSON-LD payload: %w", err) } - vmKey := opts.VerificationMethod.JSONWebKey() - if vmKey == nil { - return nil, nil, nil, errors.New("verification method needs JWK") - } - var ( keyType kms.KeyType h hash.Hash @@ -221,11 +220,22 @@ func (s *Suite) transformAndHash(doc []byte, opts *models.ProofOptions) ([]byte, verifier = s.eD25519Verifier keyType = kms.ED25519Type + + finalKey := &pubkey.PublicKey{Type: keyType, JWK: opts.VerificationMethod.JSONWebKey()} + if finalKey.JWK == nil && len(opts.VerificationMethod.Value) > 0 { + finalKey.BytesKey = &pubkey.BytesKey{Bytes: opts.VerificationMethod.Value} + } + + if finalKey.JWK == nil && finalKey.BytesKey == nil { + return nil, nil, nil, errors.New("verification method needs JWK") + } + h = sha256.New() confData := proofConfig(docData[ldCtxKey], opts) - if opts.ProofType != "DataIntegrityProof" || opts.SuiteType != SuiteType { + if opts.ProofType != "DataIntegrityProof" || (opts.SuiteType != SuiteType && + opts.SuiteType != SuiteType2) { return nil, nil, nil, suite.ErrProofTransformation } @@ -241,7 +251,7 @@ func (s *Suite) transformAndHash(doc []byte, opts *models.ProofOptions) ([]byte, docHash := hashData(canonDoc, canonConf, h) - return docHash, &pubkey.PublicKey{Type: keyType, JWK: vmKey}, verifier, nil + return docHash, finalKey, verifier, nil } // VerifyProof implements the eddsa-2022 cryptographic suite for CheckJWTProof Proof. @@ -279,26 +289,41 @@ func canonicalize(data map[string]interface{}, loader ld.DocumentLoader) ([]byte return out, nil } -func hashData(transformedDoc, confData []byte, h hash.Hash) []byte { - h.Write(transformedDoc) +func hashData(docData, proofData []byte, h hash.Hash) []byte { + h.Write(docData) docHash := h.Sum(nil) h.Reset() - h.Write(confData) - result := h.Sum(docHash) + h.Write(proofData) + proofHash := h.Sum(nil) - return result + return append(proofHash, docHash...) } func proofConfig(docCtx interface{}, opts *models.ProofOptions) map[string]interface{} { - return map[string]interface{}{ + suiteType := SuiteType + if opts.SuiteType != "" { + suiteType = opts.SuiteType + } + + proof := map[string]interface{}{ ldCtxKey: docCtx, "type": models.DataIntegrityProof, - "cryptosuite": SuiteType, + "cryptosuite": suiteType, "verificationMethod": opts.VerificationMethodID, "created": opts.Created.Format(models.DateTimeFormat), "proofPurpose": opts.Purpose, } + + if opts.Challenge != "" { + proof["challenge"] = opts.Challenge + } + + if opts.Domain != "" { + proof["domain"] = opts.Domain + } + + return proof } func sign(sigBase []byte, key *jwk.JWK, signerGetter SignerGetter) ([]byte, error) { diff --git a/dataintegrity/suite/eddsa2022/eddsa2022_test.go b/dataintegrity/suite/eddsa2022/eddsa2022_test.go index 7fbc198..bff9495 100644 --- a/dataintegrity/suite/eddsa2022/eddsa2022_test.go +++ b/dataintegrity/suite/eddsa2022/eddsa2022_test.go @@ -51,7 +51,7 @@ func TestNew(t *testing.T) { require.NotNil(t, signer) require.False(t, signer.RequiresCreated()) - require.EqualValues(t, []string{"eddsa-rdfc-2022"}, sigInit.Type()) + require.EqualValues(t, []string{"eddsa-rdfc-2022", "eddsa-2022"}, sigInit.Type()) }) t.Run("verifier success", func(t *testing.T) { @@ -250,18 +250,6 @@ func TestSharedFailures(t *testing.T) { testSign(t, tc) }) - t.Run("no jwk in vm", func(t *testing.T) { - tc := successCase(t) - - tc.proofOpts.VerificationMethod = &did.VerificationMethod{ - ID: tc.proofOpts.VerificationMethodID, - Value: []byte(fooBar), - } - tc.errStr = "verification method needs JWK" - - testSign(t, tc) - }) - t.Run("invalid proof/suite type", func(t *testing.T) { tc := successCase(t) diff --git a/go.mod b/go.mod index 55d5b5d..998cfd4 100644 --- a/go.mod +++ b/go.mod @@ -21,11 +21,11 @@ require ( github.com/multiformats/go-multibase v0.1.1 github.com/piprate/json-gold v0.5.1-0.20230111113000-6ddbe6e6f19f github.com/samber/lo v1.47.0 - github.com/stretchr/testify v1.8.2 + github.com/stretchr/testify v1.8.3 github.com/tidwall/gjson v1.14.3 github.com/tidwall/sjson v1.1.4 github.com/trustbloc/bbs-signature-go v1.0.2 - github.com/trustbloc/did-go v1.3.1-0.20240910130808-bf0188fdfe70 + github.com/trustbloc/did-go v1.3.1-0.20241021165331-5721a3ff7396 github.com/trustbloc/kms-go v1.1.2 github.com/veraison/go-cose v1.1.1-0.20240126165338-2300d5c96dbd github.com/xeipuuv/gojsonschema v1.2.0 @@ -67,3 +67,5 @@ require ( gopkg.in/yaml.v3 v3.0.1 // indirect rsc.io/tmplfunc v0.0.3 // indirect ) + +replace github.com/piprate/json-gold v0.5.1-0.20230111113000-6ddbe6e6f19f => github.com/trustbloc/json-gold v0.5.1 diff --git a/go.sum b/go.sum index 205305f..bcaff40 100644 --- a/go.sum +++ b/go.sum @@ -96,8 +96,6 @@ github.com/multiformats/go-multibase v0.1.1/go.mod h1:ZEjHE+IsUrgp5mhlEAYjMtZwK1 github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= -github.com/piprate/json-gold v0.5.1-0.20230111113000-6ddbe6e6f19f h1:HlPa7RcxTCrva5izPfTEfvYecO7LTahgmMRD1Qp13xg= -github.com/piprate/json-gold v0.5.1-0.20230111113000-6ddbe6e6f19f/go.mod h1:WZ501QQMbZZ+3pXFPhQKzNwS1+jls0oqov3uQ2WasLs= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= @@ -111,14 +109,10 @@ github.com/samber/lo v1.47.0/go.mod h1:RmDH9Ct32Qy3gduHQuKJ3gW1fMHAnE/fAzQuf6He5 github.com/shopspring/decimal v1.3.1 h1:2Usl1nmF/WZucqkFZhnfFYxxxu8LG21F6nPQBE5gKV8= github.com/shopspring/decimal v1.3.1/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= -github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= -github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= -github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= -github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY= +github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= github.com/teserakt-io/golang-ed25519 v0.0.0-20210104091850-3888c087a4c8 h1:RBkacARv7qY5laaXGlF4wFB/tk5rnthhPb8oIBGoagY= github.com/teserakt-io/golang-ed25519 v0.0.0-20210104091850-3888c087a4c8/go.mod h1:9PdLyPiZIiW3UopXyRnPYyjUXSpiQNHRLu8fOsR3o8M= github.com/tidwall/gjson v1.6.7/go.mod h1:zeFuBCIqD4sN/gmqBzZ4j7Jd6UcA2Fc56x7QFsv+8fI= @@ -134,8 +128,10 @@ github.com/tidwall/sjson v1.1.4 h1:bTSsPLdAYF5QNLSwYsKfBKKTnlGbIuhqL3CpRsjzGhg= github.com/tidwall/sjson v1.1.4/go.mod h1:wXpKXu8CtDjKAZ+3DrKY5ROCorDFahq8l0tey/Lx1fg= github.com/trustbloc/bbs-signature-go v1.0.2 h1:gepEsbLiZHv/vva9FKG5gF38mGtOIyGez7desZxiI1o= github.com/trustbloc/bbs-signature-go v1.0.2/go.mod h1:xYotcXHAbcE0TO+SteW0J6XI3geQaXq4wdnXR2k+XCU= -github.com/trustbloc/did-go v1.3.1-0.20240910130808-bf0188fdfe70 h1:5GOoXZcKYDTsobaTCeFUEBjbognYbjBh38fCjkvClU4= -github.com/trustbloc/did-go v1.3.1-0.20240910130808-bf0188fdfe70/go.mod h1:packTRoBoo8DrwOE7QKsI98xXS3Vf6ovUXYD4FUAcB4= +github.com/trustbloc/did-go v1.3.1-0.20241021165331-5721a3ff7396 h1:z9x5gLgDeUtcPUS8uQgHD/KQ/PL5VK2QV9oENYjsWbU= +github.com/trustbloc/did-go v1.3.1-0.20241021165331-5721a3ff7396/go.mod h1:L5m4TVlPwe7VN5FRrANPMg6EJN8wIlthC8CvossDZVI= +github.com/trustbloc/json-gold v0.5.1 h1:0HHf0ildMnN4rUr7Rgxwnm1CO116JoGMrgoWIFngM1U= +github.com/trustbloc/json-gold v0.5.1/go.mod h1:RVhE35veDX19r5gfUAR+IYHkAUuPwJO8Ie/qVeFaIzw= github.com/trustbloc/kms-go v1.1.2 h1:nAlhDoHkSyX1eQFRz/sJsdgmJuNadyX7FJEy/9ROwys= github.com/trustbloc/kms-go v1.1.2/go.mod h1:OKOtsLbE6W5s4mpjWkvk8XEqcmt9vTgVmDNkHELpWO0= github.com/veraison/go-cose v1.1.1-0.20240126165338-2300d5c96dbd h1:QhdCHSW1/oosJbzBTEYLU6xcKxXbQzzqFnhCtW2UWbA= diff --git a/proof/ldproofs/ed25519signature2020/proof.go b/proof/ldproofs/ed25519signature2020/proof.go index 6a2e747..df70d2a 100644 --- a/proof/ldproofs/ed25519signature2020/proof.go +++ b/proof/ldproofs/ed25519signature2020/proof.go @@ -50,6 +50,12 @@ func New() *Proof { JWKKeyType: JWKKeyType, JWKCurve: JWKCurve, }, + { + VerificationMethodType: "Ed25519VerificationKey2018", + KMSKeyType: kms.ED25519Type, + JWKKeyType: JWKKeyType, + JWKCurve: JWKCurve, + }, { VerificationMethodType: "JsonWebKey2020", KMSKeyType: kms.ED25519Type, diff --git a/verifiable/data_integrity_proof_test.go b/verifiable/data_integrity_proof_test.go index c4cbe5f..bd85605 100644 --- a/verifiable/data_integrity_proof_test.go +++ b/verifiable/data_integrity_proof_test.go @@ -7,17 +7,27 @@ SPDX-License-Identifier: Apache-2.0 package verifiable import ( + _ "embed" + "net/http" "testing" "time" + "github.com/piprate/json-gold/ld" + "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" "github.com/trustbloc/did-go/doc/did" + "github.com/trustbloc/did-go/method/jwk" + "github.com/trustbloc/did-go/method/key" + vdrpkg "github.com/trustbloc/did-go/vdr" vdrapi "github.com/trustbloc/did-go/vdr/api" kmsapi "github.com/trustbloc/kms-go/spi/kms" "github.com/trustbloc/vc-go/dataintegrity" "github.com/trustbloc/vc-go/dataintegrity/suite/ecdsa2019" + "github.com/trustbloc/vc-go/dataintegrity/suite/eddsa2022" "github.com/trustbloc/vc-go/internal/testutil/kmscryptoutil" + "github.com/trustbloc/vc-go/proof/defaults" + "github.com/trustbloc/vc-go/vermethod" ) func Test_DataIntegrity_SignVerify(t *testing.T) { @@ -178,6 +188,62 @@ func Test_DataIntegrity_SignVerify(t *testing.T) { }) } +//go:embed testdata/example_presentation.jsonld +var examplePresentation []byte + +//go:embed testdata/example_presentation_2.json +var examplePresentation2 []byte + +//go:embed testdata/context/credential_v2.jsonld +var credentialV2Context []byte + +func TestCanParseRDFC2022Presentation(t *testing.T) { + vdr := vdrpkg.New(vdrpkg.WithVDR(jwk.New()), vdrpkg.WithVDR(key.New())) + + loader := ld.NewDefaultDocumentLoader(http.DefaultClient) + verifier, err := dataintegrity.NewVerifier(&dataintegrity.Options{ + DIDResolver: vdr, + }, eddsa2022.NewVerifierInitializer(&eddsa2022.VerifierInitializerOptions{ + LDDocumentLoader: loader, + })) + + resp, err := ParsePresentation(examplePresentation, + WithPresDataIntegrityVerifier(verifier), + WithPresJSONLDDocumentLoader(loader), + WithPresExpectedDataIntegrityFields("authentication", + "github.com/w3c/vc-data-model-2.0-test-suite", "ubXbWYV5hUDu1VCy2b75qKg"), + ) + require.NoError(t, err) + assert.NotNil(t, resp) +} + +func TestCanParsePlaygroundPresentation(t *testing.T) { + vdr := vdrpkg.New(vdrpkg.WithVDR(jwk.New()), vdrpkg.WithVDR(key.New())) + + loader := ld.NewDefaultDocumentLoader(http.DefaultClient) + verifier, err := dataintegrity.NewVerifier(&dataintegrity.Options{ + DIDResolver: vdr, + }, eddsa2022.NewVerifierInitializer(&eddsa2022.VerifierInitializerOptions{ + LDDocumentLoader: loader, + }), ecdsa2019.NewVerifierInitializer(&ecdsa2019.VerifierInitializerOptions{ + LDDocumentLoader: loader, + })) + + proofChecker := defaults.NewDefaultProofChecker(vermethod.NewVDRResolver(vdr)) + + resp, err := ParsePresentation(examplePresentation2, + WithPresDataIntegrityVerifier(verifier), + WithPresJSONLDDocumentLoader(loader), + WithPresProofChecker(proofChecker), + WithPresExpectedDataIntegrityFields("authentication", + "https://playground.chapi.io", + "3779e883a51a8086039db1d4e773aec26faeb3ee99643706345c572cddded857", + ), + ) + require.NoError(t, err) + assert.NotNil(t, resp) +} + type resolveFunc func(id string) (*did.DocResolution, error) func (f resolveFunc) Resolve(id string, opts ...vdrapi.DIDMethodOption) (*did.DocResolution, error) { diff --git a/verifiable/testdata/context/credential_v2.jsonld b/verifiable/testdata/context/credential_v2.jsonld new file mode 100644 index 0000000..827fe1d --- /dev/null +++ b/verifiable/testdata/context/credential_v2.jsonld @@ -0,0 +1,340 @@ +{ + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "description": "https://schema.org/description", + "digestMultibase": { + "@id": "https://w3id.org/security#digestMultibase", + "@type": "https://w3id.org/security#multibase" + }, + "digestSRI": { + "@id": "https://www.w3.org/2018/credentials#digestSRI", + "@type": "https://www.w3.org/2018/credentials#sriString" + }, + "mediaType": { + "@id": "https://schema.org/encodingFormat" + }, + "name": "https://schema.org/name", + + "VerifiableCredential": { + "@id": "https://www.w3.org/2018/credentials#VerifiableCredential", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "confidenceMethod": { + "@id": "https://www.w3.org/2018/credentials#confidenceMethod", + "@type": "@id" + }, + "credentialSchema": { + "@id": "https://www.w3.org/2018/credentials#credentialSchema", + "@type": "@id" + }, + "credentialStatus": { + "@id": "https://www.w3.org/2018/credentials#credentialStatus", + "@type": "@id" + }, + "credentialSubject": { + "@id": "https://www.w3.org/2018/credentials#credentialSubject", + "@type": "@id" + }, + "description": "https://schema.org/description", + "evidence": { + "@id": "https://www.w3.org/2018/credentials#evidence", + "@type": "@id" + }, + "issuer": { + "@id": "https://www.w3.org/2018/credentials#issuer", + "@type": "@id" + }, + "name": "https://schema.org/name", + "proof": { + "@id": "https://w3id.org/security#proof", + "@type": "@id", + "@container": "@graph" + }, + "refreshService": { + "@id": "https://www.w3.org/2018/credentials#refreshService", + "@type": "@id" + }, + "relatedResource": { + "@id": "https://www.w3.org/2018/credentials#relatedResource", + "@type": "@id" + }, + "renderMethod": { + "@id": "https://www.w3.org/2018/credentials#renderMethod", + "@type": "@id" + }, + "termsOfUse": { + "@id": "https://www.w3.org/2018/credentials#termsOfUse", + "@type": "@id" + }, + "validFrom": { + "@id": "https://www.w3.org/2018/credentials#validFrom", + "@type": "http://www.w3.org/2001/XMLSchema#dateTime" + }, + "validUntil": { + "@id": "https://www.w3.org/2018/credentials#validUntil", + "@type": "http://www.w3.org/2001/XMLSchema#dateTime" + } + } + }, + + "EnvelopedVerifiableCredential": + "https://www.w3.org/2018/credentials#EnvelopedVerifiableCredential", + + "VerifiablePresentation": { + "@id": "https://www.w3.org/2018/credentials#VerifiablePresentation", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "holder": { + "@id": "https://www.w3.org/2018/credentials#holder", + "@type": "@id" + }, + "proof": { + "@id": "https://w3id.org/security#proof", + "@type": "@id", + "@container": "@graph" + }, + "termsOfUse": { + "@id": "https://www.w3.org/2018/credentials#termsOfUse", + "@type": "@id" + }, + "verifiableCredential": { + "@id": "https://www.w3.org/2018/credentials#verifiableCredential", + "@type": "@id", + "@container": "@graph", + "@context": null + } + } + }, + + "EnvelopedVerifiablePresentation": + "https://www.w3.org/2018/credentials#EnvelopedVerifiablePresentation", + + "JsonSchemaCredential": + "https://www.w3.org/2018/credentials#JsonSchemaCredential", + + "JsonSchema": { + "@id": "https://www.w3.org/2018/credentials#JsonSchema", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "jsonSchema": { + "@id": "https://www.w3.org/2018/credentials#jsonSchema", + "@type": "@json" + } + } + }, + + "BitstringStatusListCredential": + "https://www.w3.org/ns/credentials/status#BitstringStatusListCredential", + + "BitstringStatusList": { + "@id": "https://www.w3.org/ns/credentials/status#BitstringStatusList", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "encodedList": { + "@id": "https://www.w3.org/ns/credentials/status#encodedList", + "@type": "https://w3id.org/security#multibase" + }, + "statusMessage": { + "@id": "https://www.w3.org/ns/credentials/status#statusMessage", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "message": "https://www.w3.org/ns/credentials/status#message", + "status": "https://www.w3.org/ns/credentials/status#status" + } + }, + "statusPurpose": + "https://www.w3.org/ns/credentials/status#statusPurpose", + "statusReference": { + "@id": "https://www.w3.org/ns/credentials/status#statusReference", + "@type": "@id" + }, + "statusSize": { + "@id": "https://www.w3.org/ns/credentials/status#statusSize", + "@type": "https://www.w3.org/2001/XMLSchema#positiveInteger" + }, + "ttl": "https://www.w3.org/ns/credentials/status#ttl" + } + }, + + "BitstringStatusListEntry": { + "@id": + "https://www.w3.org/ns/credentials/status#BitstringStatusListEntry", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "statusListCredential": { + "@id": + "https://www.w3.org/ns/credentials/status#statusListCredential", + "@type": "@id" + }, + "statusListIndex": + "https://www.w3.org/ns/credentials/status#statusListIndex", + "statusPurpose": + "https://www.w3.org/ns/credentials/status#statusPurpose" + } + }, + + "DataIntegrityProof": { + "@id": "https://w3id.org/security#DataIntegrityProof", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "challenge": "https://w3id.org/security#challenge", + "created": { + "@id": "http://purl.org/dc/terms/created", + "@type": "http://www.w3.org/2001/XMLSchema#dateTime" + }, + "cryptosuite": { + "@id": "https://w3id.org/security#cryptosuite", + "@type": "https://w3id.org/security#cryptosuiteString" + }, + "domain": "https://w3id.org/security#domain", + "expires": { + "@id": "https://w3id.org/security#expiration", + "@type": "http://www.w3.org/2001/XMLSchema#dateTime" + }, + "nonce": "https://w3id.org/security#nonce", + "previousProof": { + "@id": "https://w3id.org/security#previousProof", + "@type": "@id" + }, + "proofPurpose": { + "@id": "https://w3id.org/security#proofPurpose", + "@type": "@vocab", + "@context": { + "@protected": true, + + "id": "@id", + "type": "@type", + + "assertionMethod": { + "@id": "https://w3id.org/security#assertionMethod", + "@type": "@id", + "@container": "@set" + }, + "authentication": { + "@id": "https://w3id.org/security#authenticationMethod", + "@type": "@id", + "@container": "@set" + }, + "capabilityDelegation": { + "@id": "https://w3id.org/security#capabilityDelegationMethod", + "@type": "@id", + "@container": "@set" + }, + "capabilityInvocation": { + "@id": "https://w3id.org/security#capabilityInvocationMethod", + "@type": "@id", + "@container": "@set" + }, + "keyAgreement": { + "@id": "https://w3id.org/security#keyAgreementMethod", + "@type": "@id", + "@container": "@set" + } + } + }, + "proofValue": { + "@id": "https://w3id.org/security#proofValue", + "@type": "https://w3id.org/security#multibase" + }, + "verificationMethod": { + "@id": "https://w3id.org/security#verificationMethod", + "@type": "@id" + } + } + }, + + "...": { + "@id": "https://www.iana.org/assignments/jwt#..." + }, + "_sd": { + "@id": "https://www.iana.org/assignments/jwt#_sd", + "@type": "@json" + }, + "_sd_alg": { + "@id": "https://www.iana.org/assignments/jwt#_sd_alg" + }, + "aud": { + "@id": "https://www.iana.org/assignments/jwt#aud", + "@type": "@id" + }, + "cnf": { + "@id": "https://www.iana.org/assignments/jwt#cnf", + "@context": { + "@protected": true, + + "kid": { + "@id": "https://www.iana.org/assignments/jwt#kid", + "@type": "@id" + }, + "jwk": { + "@id": "https://www.iana.org/assignments/jwt#jwk", + "@type": "@json" + } + } + }, + "exp": { + "@id": "https://www.iana.org/assignments/jwt#exp", + "@type": "https://www.w3.org/2001/XMLSchema#nonNegativeInteger" + }, + "iat": { + "@id": "https://www.iana.org/assignments/jwt#iat", + "@type": "https://www.w3.org/2001/XMLSchema#nonNegativeInteger" + }, + "iss": { + "@id": "https://www.iana.org/assignments/jose#iss", + "@type": "@id" + }, + "jku": { + "@id": "https://www.iana.org/assignments/jose#jku", + "@type": "@id" + }, + "kid": { + "@id": "https://www.iana.org/assignments/jose#kid", + "@type": "@id" + }, + "nbf": { + "@id": "https://www.iana.org/assignments/jwt#nbf", + "@type": "https://www.w3.org/2001/XMLSchema#nonNegativeInteger" + }, + "sub": { + "@id": "https://www.iana.org/assignments/jose#sub", + "@type": "@id" + }, + "x5u": { + "@id": "https://www.iana.org/assignments/jose#x5u", + "@type": "@id" + } + } +} diff --git a/verifiable/testdata/example_presentation.jsonld b/verifiable/testdata/example_presentation.jsonld new file mode 100644 index 0000000..f808017 --- /dev/null +++ b/verifiable/testdata/example_presentation.jsonld @@ -0,0 +1,40 @@ +{ + "@context": [ + "https://www.w3.org/ns/credentials/v2" + ], + "type": [ + "VerifiablePresentation" + ], + "verifiableCredential": [ + { + "@context": [ + "https://www.w3.org/ns/credentials/v2" + ], + "type": [ + "VerifiableCredential" + ], + "issuer": "did:key:z6MkpJySvETLnxhQG9DzEdmKJtysBDjuuTeDfUj1uNNCUqcj", + "credentialSubject": { + "id": "did:example:subject" + }, + "proof": { + "type": "DataIntegrityProof", + "created": "2024-10-21T19:27:16Z", + "verificationMethod": "did:key:z6MkpJySvETLnxhQG9DzEdmKJtysBDjuuTeDfUj1uNNCUqcj#z6MkpJySvETLnxhQG9DzEdmKJtysBDjuuTeDfUj1uNNCUqcj", + "cryptosuite": "eddsa-rdfc-2022", + "proofPurpose": "assertionMethod", + "proofValue": "z2kb5qAGHsTYw9zeGqEScYFFHmWj8qoYrN5JeK6rp3PJ88Ma5sPAgvoZpvPbbfRFwAZZhmDG1w3L5A2jdzSKX4TPJ" + } + } + ], + "proof": { + "type": "DataIntegrityProof", + "created": "2024-10-21T19:27:40Z", + "verificationMethod": "did:key:z6MkpJySvETLnxhQG9DzEdmKJtysBDjuuTeDfUj1uNNCUqcj#z6MkpJySvETLnxhQG9DzEdmKJtysBDjuuTeDfUj1uNNCUqcj", + "cryptosuite": "eddsa-rdfc-2022", + "proofPurpose": "authentication", + "challenge": "ubXbWYV5hUDu1VCy2b75qKg", + "domain": "github.com/w3c/vc-data-model-2.0-test-suite", + "proofValue": "z2dmNLqEfBenLXXJnd79ucZr1UJR2qkNdWkNuohmTfXfKMwc7U2MKuMVSje1mJKmSWpNtfVmZRDKeSeLsNrWPRk9f" + } +} diff --git a/verifiable/testdata/example_presentation_2.json b/verifiable/testdata/example_presentation_2.json new file mode 100644 index 0000000..fc1ac6c --- /dev/null +++ b/verifiable/testdata/example_presentation_2.json @@ -0,0 +1,76 @@ +{ + "@context": [ + "https://www.w3.org/2018/credentials/v1", + "https://purl.imsglobal.org/spec/ob/v3p0/context.json", + "https://w3id.org/security/suites/ed25519-2020/v1" + ], + "type": [ + "VerifiablePresentation" + ], + "holder": "did:key:z6Mkt4Jf2FtLZ52N2ffVN1mFhmVV4SvUJL69jGBEvgx3y3FK", + "verifiableCredential": [ + { + "@context": [ + "https://www.w3.org/2018/credentials/v1", + "https://purl.imsglobal.org/spec/ob/v3p0/context.json", + "https://w3id.org/security/suites/ed25519-2020/v1" + ], + "id": "urn:uuid:29f07569-ab0f-4980-ab81-998c1c11ddbe", + "type": [ + "VerifiableCredential", + "OpenBadgeCredential" + ], + "name": "Intro to CHAPI", + "issuer": { + "type": [ + "Profile" + ], + "id": "did:key:z6Mkv4v7tCJurgqgGxD7biXmXH8YGdHSggNhrMLpkMEoPi7e", + "name": "W3C Verifiable Credentials Education Task Force", + "image": { + "id": "", + "type": "Image" + } + }, + "credentialSubject": { + "type": [ + "AchievementSubject" + ], + "achievement": { + "id": "urn:uuid:1a197ca4-7a29-457c-ae74-a7f8db1fe8e5", + "type": [ + "Achievement" + ], + "achievementType": "Badge", + "name": "VC-EDU - Introduction to CHAPI", + "description": "The holder of this badge used the CHAPI Playground to issue and obtain an Open Badges 3.0 credential", + "criteria": { + "narrative": "The Verifiable Credentials Task Force (VC-EDU) aims to increase understanding of how to issue and use verifiable credentials (VCs) as learning and employment records (LERs). [CHAPI (Credential Handler API)](https://chapi.io/) is one way for VCs to be securely transported on the web. This intro badge demonstrates how CHAPI works with a real Open Badges 3.0 VC. To earn this badge, the holder used the tools at [CHAPI Playground](https://playground.chapi.io) to issue the badge and store it in a CHAPI-enabled digital wallet." + }, + "image": { + "id": "", + "type": "Image" + } + }, + "id": "did:key:z6MkiQZT4nphaKsVhjWWi2ef3ZeeX3JGaKJKTykaFBXns3M2" + }, + "issuanceDate": "2023-05-10T06:54:58Z", + "proof": { + "type": "Ed25519Signature2020", + "created": "2023-05-10T06:54:58Z", + "verificationMethod": "did:key:z6Mkv4v7tCJurgqgGxD7biXmXH8YGdHSggNhrMLpkMEoPi7e#z6Mkv4v7tCJurgqgGxD7biXmXH8YGdHSggNhrMLpkMEoPi7e", + "proofPurpose": "assertionMethod", + "proofValue": "z5n9iiRd6FsNVkuXQrUKXj7JYww5gfoU2A2eoeX5hyBKCiMJho3FH8LQ1cB4SVKLzdFtRmYj79J8yp6yaE6iegBNK" + } + } + ], + "proof": { + "type": "Ed25519Signature2020", + "proofPurpose": "authentication", + "verificationMethod": "did:key:z6MktTmBQWV6FmHZyfvSV7ubQ1UCPgLF6VQkRgeGBx7NRzCP#z6MktTmBQWV6FmHZyfvSV7ubQ1UCPgLF6VQkRgeGBx7NRzCP", + "created": "2023-05-10T07:38:42Z", + "proofValue": "z5fLhzuxCf4Zn4t4zvsCaoJhf36T93e17d8hW6j8DUpXbtw1on9zL1KVwnjDEBwR22WbYKQ39b5U3dAykFM6E8wiT", + "challenge": "3779e883a51a8086039db1d4e773aec26faeb3ee99643706345c572cddded857", + "domain": "https://playground.chapi.io" + } +}