github.com/hyperledger/aries-framework-go@v0.3.2/pkg/doc/verifiable/example_credential_test.go (about) 1 /* 2 Copyright SecureKey Technologies Inc. All Rights Reserved. 3 4 SPDX-License-Identifier: Apache-2.0 5 */ 6 7 package verifiable_test 8 9 import ( 10 "crypto/ed25519" 11 "encoding/base64" 12 "encoding/json" 13 "errors" 14 "fmt" 15 "time" 16 17 "github.com/hyperledger/aries-framework-go/pkg/common/log" 18 "github.com/hyperledger/aries-framework-go/pkg/doc/jose/jwk/jwksupport" 19 "github.com/hyperledger/aries-framework-go/pkg/doc/signature/jsonld" 20 "github.com/hyperledger/aries-framework-go/pkg/doc/signature/suite" 21 "github.com/hyperledger/aries-framework-go/pkg/doc/signature/suite/bbsblssignature2020" 22 "github.com/hyperledger/aries-framework-go/pkg/doc/signature/suite/ed25519signature2018" 23 "github.com/hyperledger/aries-framework-go/pkg/doc/signature/suite/jsonwebsignature2020" 24 sigverifier "github.com/hyperledger/aries-framework-go/pkg/doc/signature/verifier" 25 "github.com/hyperledger/aries-framework-go/pkg/doc/util" 26 "github.com/hyperledger/aries-framework-go/pkg/doc/util/signature" 27 "github.com/hyperledger/aries-framework-go/pkg/doc/verifiable" 28 "github.com/hyperledger/aries-framework-go/pkg/kms" 29 spi "github.com/hyperledger/aries-framework-go/spi/log" 30 ) 31 32 //nolint:gochecknoglobals 33 var ( 34 // Private key generated by ed25519.GenerateKey(rand.Reader). 35 issuerPrivKey = ed25519.PrivateKey{72, 67, 163, 188, 235, 199, 239, 146, 129, 52, 228, 34, 44, 106, 23, 144, 189, 57, 115, 171, 4, 217, 54, 121, 41, 155, 251, 83, 1, 240, 238, 65, 234, 100, 192, 93, 251, 181, 198, 73, 122, 220, 27, 48, 93, 73, 166, 33, 152, 140, 168, 36, 9, 205, 59, 161, 137, 7, 164, 9, 176, 252, 1, 171} 36 issuerPubKey = ed25519.PublicKey{234, 100, 192, 93, 251, 181, 198, 73, 122, 220, 27, 48, 93, 73, 166, 33, 152, 140, 168, 36, 9, 205, 59, 161, 137, 7, 164, 9, 176, 252, 1, 171} 37 issued = time.Date(2010, time.January, 1, 19, 23, 24, 0, time.UTC) 38 expired = time.Date(2020, time.January, 1, 19, 23, 24, 0, time.UTC) 39 40 bbsPrivKeyB64 = "PcVroyzTlmnYIIq8In8QOZhpK72AdTjj3EitB9tSNrg" 41 bbsPubKeyB64 = "l0Wtf3gy5f140G5vCoCJw2420hwk6Xw65/DX3ycv1W7/eMky8DyExw+o1s2bmq3sEIJatkiN8f5D4k0766x0UvfbupFX+vVkeqnlOvT6o2cag2osQdMFbBQqAybOM4Gm" 42 ) 43 44 const vcJSON = ` 45 { 46 "@context": [ 47 "https://www.w3.org/2018/credentials/v1", 48 "https://www.w3.org/2018/credentials/examples/v1" 49 ], 50 "credentialSchema": [], 51 "credentialSubject": { 52 "degree": { 53 "type": "BachelorDegree", 54 "university": "MIT" 55 }, 56 "id": "did:example:ebfeb1f712ebc6f1c276e12ec21", 57 "name": "Jayden Doe", 58 "spouse": "did:example:c276e12ec21ebfeb1f712ebc6f1" 59 }, 60 "expirationDate": "2020-01-01T19:23:24Z", 61 "id": "http://example.edu/credentials/1872", 62 "issuanceDate": "2009-01-01T19:23:24Z", 63 "issuer": { 64 "id": "did:example:76e12ec712ebc6f1c221ebfeb1f", 65 "name": "Example University" 66 }, 67 "referenceNumber": 83294849, 68 "type": [ 69 "VerifiableCredential", 70 "UniversityDegreeCredential" 71 ] 72 } 73 ` 74 75 func ExampleCredential_embedding() { 76 vc := &UniversityDegreeCredential{ 77 Credential: &verifiable.Credential{ 78 Context: []string{ 79 "https://www.w3.org/2018/credentials/v1", 80 "https://www.w3.org/2018/credentials/examples/v1", 81 }, 82 ID: "http://example.edu/credentials/1872", 83 Types: []string{ 84 "VerifiableCredential", 85 "UniversityDegreeCredential", 86 }, 87 Subject: UniversityDegreeSubject{ 88 ID: "did:example:ebfeb1f712ebc6f1c276e12ec21", 89 Name: "Jayden Doe", 90 Spouse: "did:example:c276e12ec21ebfeb1f712ebc6f1", 91 Degree: UniversityDegree{ 92 Type: "BachelorDegree", 93 University: "MIT", 94 }, 95 }, 96 Issuer: verifiable.Issuer{ 97 ID: "did:example:76e12ec712ebc6f1c221ebfeb1f", 98 CustomFields: verifiable.CustomFields{"name": "Example University"}, 99 }, 100 Issued: util.NewTime(issued), 101 Expired: util.NewTime(expired), 102 Schemas: []verifiable.TypedID{}, 103 }, 104 ReferenceNumber: 83294847, 105 } 106 107 // Marshal to JSON to verify the result of decoding. 108 vcBytes, err := json.Marshal(vc) 109 if err != nil { 110 panic("failed to marshal VC to JSON") 111 } 112 113 fmt.Println(string(vcBytes)) 114 115 // Marshal to JWS. 116 jwtClaims, err := vc.JWTClaims(true) 117 if err != nil { 118 panic(fmt.Errorf("failed to marshal JWT claims of VC: %w", err)) 119 } 120 121 signer := signature.GetEd25519Signer(issuerPrivKey, issuerPubKey) 122 123 jws, err := jwtClaims.MarshalJWS(verifiable.EdDSA, signer, "did:123#key1") 124 if err != nil { 125 panic(fmt.Errorf("failed to sign VC inside JWT: %w", err)) 126 } 127 128 fmt.Println(jws) 129 130 // Parse JWS and make sure it's coincide with JSON. 131 vcParsed, err := verifiable.ParseCredential( 132 []byte(jws), 133 verifiable.WithPublicKeyFetcher(verifiable.SingleKey(issuerPubKey, kms.ED25519)), 134 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader())) 135 if err != nil { 136 panic(fmt.Errorf("failed to encode VC from JWS: %w", err)) 137 } 138 139 // When a Credential was parsed from JWS, it Marshals into a JSON string containing the original JWS. 140 141 vcBytesFromJWS, err := vcParsed.MarshalJSON() 142 if err != nil { 143 panic(fmt.Errorf("failed to marshal VC: %w", err)) 144 } 145 146 // todo missing referenceNumber here (https://github.com/hyperledger/aries-framework-go/issues/847) 147 fmt.Println(string(vcBytesFromJWS)) 148 149 // To marshal the Credential into JSON-LD form, clear the JWT field. 150 151 vcParsed.JWT = "" 152 153 vcBytesFromJWS, err = vcParsed.MarshalJSON() 154 if err != nil { 155 panic(fmt.Errorf("failed to marshal VC: %w", err)) 156 } 157 158 fmt.Println(string(vcBytesFromJWS)) 159 160 // Output: 161 // {"@context":["https://www.w3.org/2018/credentials/v1","https://www.w3.org/2018/credentials/examples/v1"],"credentialSubject":{"degree":{"type":"BachelorDegree","university":"MIT"},"id":"did:example:ebfeb1f712ebc6f1c276e12ec21","name":"Jayden Doe","spouse":"did:example:c276e12ec21ebfeb1f712ebc6f1"},"expirationDate":"2020-01-01T19:23:24Z","id":"http://example.edu/credentials/1872","issuanceDate":"2010-01-01T19:23:24Z","issuer":{"id":"did:example:76e12ec712ebc6f1c221ebfeb1f","name":"Example University"},"referenceNumber":83294847,"type":["VerifiableCredential","UniversityDegreeCredential"]} 162 // eyJhbGciOiJFZERTQSIsImtpZCI6ImRpZDoxMjMja2V5MSJ9.eyJleHAiOjE1Nzc5MDY2MDQsImlhdCI6MTI2MjM3MzgwNCwiaXNzIjoiZGlkOmV4YW1wbGU6NzZlMTJlYzcxMmViYzZmMWMyMjFlYmZlYjFmIiwianRpIjoiaHR0cDovL2V4YW1wbGUuZWR1L2NyZWRlbnRpYWxzLzE4NzIiLCJuYmYiOjEyNjIzNzM4MDQsInN1YiI6ImRpZDpleGFtcGxlOmViZmViMWY3MTJlYmM2ZjFjMjc2ZTEyZWMyMSIsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIiwiaHR0cHM6Ly93d3cudzMub3JnLzIwMTgvY3JlZGVudGlhbHMvZXhhbXBsZXMvdjEiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZGVncmVlIjp7InR5cGUiOiJCYWNoZWxvckRlZ3JlZSIsInVuaXZlcnNpdHkiOiJNSVQifSwiaWQiOiJkaWQ6ZXhhbXBsZTplYmZlYjFmNzEyZWJjNmYxYzI3NmUxMmVjMjEiLCJuYW1lIjoiSmF5ZGVuIERvZSIsInNwb3VzZSI6ImRpZDpleGFtcGxlOmMyNzZlMTJlYzIxZWJmZWIxZjcxMmViYzZmMSJ9LCJpc3N1ZXIiOnsibmFtZSI6IkV4YW1wbGUgVW5pdmVyc2l0eSJ9LCJ0eXBlIjpbIlZlcmlmaWFibGVDcmVkZW50aWFsIiwiVW5pdmVyc2l0eURlZ3JlZUNyZWRlbnRpYWwiXX19.4VSZj1tPovsqtBihcWHxlGaE25bBNOqNdH85UoAz7XbV1rZ9hPYklwuV_sRRbcQOcEjTaRik2tzwsOPOz5EDAg 163 // "eyJhbGciOiJFZERTQSIsImtpZCI6ImRpZDoxMjMja2V5MSJ9.eyJleHAiOjE1Nzc5MDY2MDQsImlhdCI6MTI2MjM3MzgwNCwiaXNzIjoiZGlkOmV4YW1wbGU6NzZlMTJlYzcxMmViYzZmMWMyMjFlYmZlYjFmIiwianRpIjoiaHR0cDovL2V4YW1wbGUuZWR1L2NyZWRlbnRpYWxzLzE4NzIiLCJuYmYiOjEyNjIzNzM4MDQsInN1YiI6ImRpZDpleGFtcGxlOmViZmViMWY3MTJlYmM2ZjFjMjc2ZTEyZWMyMSIsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIiwiaHR0cHM6Ly93d3cudzMub3JnLzIwMTgvY3JlZGVudGlhbHMvZXhhbXBsZXMvdjEiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZGVncmVlIjp7InR5cGUiOiJCYWNoZWxvckRlZ3JlZSIsInVuaXZlcnNpdHkiOiJNSVQifSwiaWQiOiJkaWQ6ZXhhbXBsZTplYmZlYjFmNzEyZWJjNmYxYzI3NmUxMmVjMjEiLCJuYW1lIjoiSmF5ZGVuIERvZSIsInNwb3VzZSI6ImRpZDpleGFtcGxlOmMyNzZlMTJlYzIxZWJmZWIxZjcxMmViYzZmMSJ9LCJpc3N1ZXIiOnsibmFtZSI6IkV4YW1wbGUgVW5pdmVyc2l0eSJ9LCJ0eXBlIjpbIlZlcmlmaWFibGVDcmVkZW50aWFsIiwiVW5pdmVyc2l0eURlZ3JlZUNyZWRlbnRpYWwiXX19.4VSZj1tPovsqtBihcWHxlGaE25bBNOqNdH85UoAz7XbV1rZ9hPYklwuV_sRRbcQOcEjTaRik2tzwsOPOz5EDAg" 164 // {"@context":["https://www.w3.org/2018/credentials/v1","https://www.w3.org/2018/credentials/examples/v1"],"credentialSubject":{"degree":{"type":"BachelorDegree","university":"MIT"},"id":"did:example:ebfeb1f712ebc6f1c276e12ec21","name":"Jayden Doe","spouse":"did:example:c276e12ec21ebfeb1f712ebc6f1"},"expirationDate":"2020-01-01T19:23:24Z","id":"http://example.edu/credentials/1872","issuanceDate":"2010-01-01T19:23:24Z","issuer":{"id":"did:example:76e12ec712ebc6f1c221ebfeb1f","name":"Example University"},"type":["VerifiableCredential","UniversityDegreeCredential"]} 165 } 166 167 func ExampleCredential_extraFields() { 168 vc := &verifiable.Credential{ 169 Context: []string{ 170 "https://www.w3.org/2018/credentials/v1", 171 "https://www.w3.org/2018/credentials/examples/v1", 172 }, 173 ID: "http://example.edu/credentials/1872", 174 Types: []string{ 175 "VerifiableCredential", 176 "UniversityDegreeCredential", 177 }, 178 Subject: UniversityDegreeSubject{ 179 ID: "did:example:ebfeb1f712ebc6f1c276e12ec21", 180 Name: "Jayden Doe", 181 Spouse: "did:example:c276e12ec21ebfeb1f712ebc6f1", 182 Degree: UniversityDegree{ 183 Type: "BachelorDegree", 184 University: "MIT", 185 }, 186 }, 187 Issuer: verifiable.Issuer{ 188 ID: "did:example:76e12ec712ebc6f1c221ebfeb1f", 189 CustomFields: verifiable.CustomFields{"name": "Example University"}, 190 }, 191 Issued: util.NewTime(issued), 192 Expired: util.NewTime(expired), 193 Schemas: []verifiable.TypedID{}, 194 CustomFields: map[string]interface{}{ 195 "referenceNumber": 83294847, 196 }, 197 } 198 199 // Marshal to JSON. 200 vcBytes, err := json.Marshal(vc) 201 if err != nil { 202 panic("failed to marshal VC to JSON") 203 } 204 205 fmt.Println(string(vcBytes)) 206 207 // Marshal to JWS. 208 jwtClaims, err := vc.JWTClaims(true) 209 if err != nil { 210 panic(fmt.Errorf("failed to marshal JWT claims of VC: %w", err)) 211 } 212 213 signer := signature.GetEd25519Signer(issuerPrivKey, issuerPubKey) 214 215 jws, err := jwtClaims.MarshalJWS(verifiable.EdDSA, signer, "did:123#key1") 216 if err != nil { 217 panic(fmt.Errorf("failed to sign VC inside JWT: %w", err)) 218 } 219 220 fmt.Println(jws) 221 222 // Parse JWS and make sure it's coincide with JSON. 223 vcParsed, err := verifiable.ParseCredential( 224 []byte(jws), 225 verifiable.WithPublicKeyFetcher(verifiable.SingleKey(issuerPubKey, kms.ED25519)), 226 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader())) 227 if err != nil { 228 panic(fmt.Errorf("failed to encode VC from JWS: %w", err)) 229 } 230 231 vcParsed.JWT = "" 232 233 vcBytesFromJWS, err := vcParsed.MarshalJSON() 234 if err != nil { 235 panic(fmt.Errorf("failed to marshal VC: %w", err)) 236 } 237 238 fmt.Println(string(vcBytesFromJWS)) 239 240 // Output: 241 // {"@context":["https://www.w3.org/2018/credentials/v1","https://www.w3.org/2018/credentials/examples/v1"],"credentialSubject":{"degree":{"type":"BachelorDegree","university":"MIT"},"id":"did:example:ebfeb1f712ebc6f1c276e12ec21","name":"Jayden Doe","spouse":"did:example:c276e12ec21ebfeb1f712ebc6f1"},"expirationDate":"2020-01-01T19:23:24Z","id":"http://example.edu/credentials/1872","issuanceDate":"2010-01-01T19:23:24Z","issuer":{"id":"did:example:76e12ec712ebc6f1c221ebfeb1f","name":"Example University"},"referenceNumber":83294847,"type":["VerifiableCredential","UniversityDegreeCredential"]} 242 // eyJhbGciOiJFZERTQSIsImtpZCI6ImRpZDoxMjMja2V5MSJ9.eyJleHAiOjE1Nzc5MDY2MDQsImlhdCI6MTI2MjM3MzgwNCwiaXNzIjoiZGlkOmV4YW1wbGU6NzZlMTJlYzcxMmViYzZmMWMyMjFlYmZlYjFmIiwianRpIjoiaHR0cDovL2V4YW1wbGUuZWR1L2NyZWRlbnRpYWxzLzE4NzIiLCJuYmYiOjEyNjIzNzM4MDQsInN1YiI6ImRpZDpleGFtcGxlOmViZmViMWY3MTJlYmM2ZjFjMjc2ZTEyZWMyMSIsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIiwiaHR0cHM6Ly93d3cudzMub3JnLzIwMTgvY3JlZGVudGlhbHMvZXhhbXBsZXMvdjEiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZGVncmVlIjp7InR5cGUiOiJCYWNoZWxvckRlZ3JlZSIsInVuaXZlcnNpdHkiOiJNSVQifSwiaWQiOiJkaWQ6ZXhhbXBsZTplYmZlYjFmNzEyZWJjNmYxYzI3NmUxMmVjMjEiLCJuYW1lIjoiSmF5ZGVuIERvZSIsInNwb3VzZSI6ImRpZDpleGFtcGxlOmMyNzZlMTJlYzIxZWJmZWIxZjcxMmViYzZmMSJ9LCJpc3N1ZXIiOnsibmFtZSI6IkV4YW1wbGUgVW5pdmVyc2l0eSJ9LCJyZWZlcmVuY2VOdW1iZXIiOjguMzI5NDg0N2UrMDcsInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJVbml2ZXJzaXR5RGVncmVlQ3JlZGVudGlhbCJdfX0.rmOsOJbKp68XeAw3SR93A67bgDYeOdLP3VDFIwbaNguE9eGQgdYjyAA2q07RbUD-uPoQMIpQDH6uhVAWYBDWCg 243 // {"@context":["https://www.w3.org/2018/credentials/v1","https://www.w3.org/2018/credentials/examples/v1"],"credentialSubject":{"degree":{"type":"BachelorDegree","university":"MIT"},"id":"did:example:ebfeb1f712ebc6f1c276e12ec21","name":"Jayden Doe","spouse":"did:example:c276e12ec21ebfeb1f712ebc6f1"},"expirationDate":"2020-01-01T19:23:24Z","id":"http://example.edu/credentials/1872","issuanceDate":"2010-01-01T19:23:24Z","issuer":{"id":"did:example:76e12ec712ebc6f1c221ebfeb1f","name":"Example University"},"referenceNumber":83294847,"type":["VerifiableCredential","UniversityDegreeCredential"]} 244 } 245 246 func ExampleParseCredential() { 247 // Issuer is about to issue the university degree credential for the Holder 248 vcEncoded := &verifiable.Credential{ 249 Context: []string{ 250 "https://www.w3.org/2018/credentials/v1", 251 "https://www.w3.org/2018/credentials/examples/v1", 252 }, 253 ID: "http://example.edu/credentials/1872", 254 Types: []string{ 255 "VerifiableCredential", 256 "UniversityDegreeCredential", 257 }, 258 Subject: UniversityDegreeSubject{ 259 ID: "did:example:ebfeb1f712ebc6f1c276e12ec21", 260 Name: "Jayden Doe", 261 Spouse: "did:example:c276e12ec21ebfeb1f712ebc6f1", 262 Degree: UniversityDegree{ 263 Type: "BachelorDegree", 264 University: "MIT", 265 }, 266 }, 267 Issuer: verifiable.Issuer{ 268 ID: "did:example:76e12ec712ebc6f1c221ebfeb1f", 269 CustomFields: verifiable.CustomFields{"name": "Example University"}, 270 }, 271 Issued: util.NewTime(issued), 272 Expired: util.NewTime(expired), 273 Schemas: []verifiable.TypedID{}, 274 CustomFields: map[string]interface{}{ 275 "referenceNumber": 83294847, 276 }, 277 } 278 279 // ... in JWS form. 280 jwtClaims, err := vcEncoded.JWTClaims(true) 281 if err != nil { 282 panic(fmt.Errorf("failed to marshal JWT claims of VC: %w", err)) 283 } 284 285 signer := signature.GetEd25519Signer(issuerPrivKey, issuerPubKey) 286 287 jws, err := jwtClaims.MarshalJWS(verifiable.EdDSA, signer, "did:123#key1") 288 if err != nil { 289 panic(fmt.Errorf("failed to sign VC inside JWT: %w", err)) 290 } 291 292 // The Holder receives JWS and decodes it. 293 vcParsed, err := verifiable.ParseCredential( 294 []byte(jws), 295 verifiable.WithPublicKeyFetcher(verifiable.SingleKey(issuerPubKey, kms.ED25519)), 296 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader())) 297 if err != nil { 298 panic(fmt.Errorf("failed to decode VC JWS: %w", err)) 299 } 300 301 // When parsing a verifiable.Credential from JWS, Credential.JWT is set to the raw JWS value. 302 // This allows the user to save the Credential and verify it later. 303 304 // When Credential.JWT is set, the Credential Marshals into a JSON string containing the original JWS. 305 306 vcDecodedBytes, err := vcParsed.MarshalJSON() 307 if err != nil { 308 panic(fmt.Errorf("failed to marshal VC: %w", err)) 309 } 310 311 // The Holder then e.g. can save the credential to her personal verifiable credential wallet. 312 fmt.Println(string(vcDecodedBytes)) 313 314 // To marshal the Credential into JSON-LD form, clear the JWT field. 315 vcParsed.JWT = "" 316 317 vcDecodedBytes, err = vcParsed.MarshalJSON() 318 if err != nil { 319 panic(fmt.Errorf("failed to marshal VC: %w", err)) 320 } 321 322 // The Credential is now in JSON-LD form.. 323 fmt.Println(string(vcDecodedBytes)) 324 325 // Output: 326 // "eyJhbGciOiJFZERTQSIsImtpZCI6ImRpZDoxMjMja2V5MSJ9.eyJleHAiOjE1Nzc5MDY2MDQsImlhdCI6MTI2MjM3MzgwNCwiaXNzIjoiZGlkOmV4YW1wbGU6NzZlMTJlYzcxMmViYzZmMWMyMjFlYmZlYjFmIiwianRpIjoiaHR0cDovL2V4YW1wbGUuZWR1L2NyZWRlbnRpYWxzLzE4NzIiLCJuYmYiOjEyNjIzNzM4MDQsInN1YiI6ImRpZDpleGFtcGxlOmViZmViMWY3MTJlYmM2ZjFjMjc2ZTEyZWMyMSIsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIiwiaHR0cHM6Ly93d3cudzMub3JnLzIwMTgvY3JlZGVudGlhbHMvZXhhbXBsZXMvdjEiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZGVncmVlIjp7InR5cGUiOiJCYWNoZWxvckRlZ3JlZSIsInVuaXZlcnNpdHkiOiJNSVQifSwiaWQiOiJkaWQ6ZXhhbXBsZTplYmZlYjFmNzEyZWJjNmYxYzI3NmUxMmVjMjEiLCJuYW1lIjoiSmF5ZGVuIERvZSIsInNwb3VzZSI6ImRpZDpleGFtcGxlOmMyNzZlMTJlYzIxZWJmZWIxZjcxMmViYzZmMSJ9LCJpc3N1ZXIiOnsibmFtZSI6IkV4YW1wbGUgVW5pdmVyc2l0eSJ9LCJyZWZlcmVuY2VOdW1iZXIiOjguMzI5NDg0N2UrMDcsInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJVbml2ZXJzaXR5RGVncmVlQ3JlZGVudGlhbCJdfX0.rmOsOJbKp68XeAw3SR93A67bgDYeOdLP3VDFIwbaNguE9eGQgdYjyAA2q07RbUD-uPoQMIpQDH6uhVAWYBDWCg" 327 // {"@context":["https://www.w3.org/2018/credentials/v1","https://www.w3.org/2018/credentials/examples/v1"],"credentialSubject":{"degree":{"type":"BachelorDegree","university":"MIT"},"id":"did:example:ebfeb1f712ebc6f1c276e12ec21","name":"Jayden Doe","spouse":"did:example:c276e12ec21ebfeb1f712ebc6f1"},"expirationDate":"2020-01-01T19:23:24Z","id":"http://example.edu/credentials/1872","issuanceDate":"2010-01-01T19:23:24Z","issuer":{"id":"did:example:76e12ec712ebc6f1c221ebfeb1f","name":"Example University"},"referenceNumber":83294847,"type":["VerifiableCredential","UniversityDegreeCredential"]} 328 } 329 330 func ExampleCredential_JWTClaims() { 331 // The Holder wants to send the credential to the Verifier in JWS. 332 vc, err := verifiable.ParseCredential([]byte(vcJSON), 333 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader())) 334 if err != nil { 335 panic(fmt.Errorf("failed to decode VC JSON: %w", err)) 336 } 337 338 jwtClaims, err := vc.JWTClaims(true) 339 if err != nil { 340 panic(fmt.Errorf("failed to marshal JWT claims of VC: %w", err)) 341 } 342 343 signer := signature.GetEd25519Signer(issuerPrivKey, issuerPubKey) 344 345 jws, err := jwtClaims.MarshalJWS(verifiable.EdDSA, signer, "") 346 if err != nil { 347 panic(fmt.Errorf("failed to sign VC inside JWT: %w", err)) 348 } 349 350 // The Holder passes JWS to Verifier 351 fmt.Println(jws) 352 353 // Output: eyJhbGciOiJFZERTQSIsImtpZCI6IiJ9.eyJleHAiOjE1Nzc5MDY2MDQsImlhdCI6MTIzMDgzNzgwNCwiaXNzIjoiZGlkOmV4YW1wbGU6NzZlMTJlYzcxMmViYzZmMWMyMjFlYmZlYjFmIiwianRpIjoiaHR0cDovL2V4YW1wbGUuZWR1L2NyZWRlbnRpYWxzLzE4NzIiLCJuYmYiOjEyMzA4Mzc4MDQsInN1YiI6ImRpZDpleGFtcGxlOmViZmViMWY3MTJlYmM2ZjFjMjc2ZTEyZWMyMSIsInZjIjp7IkBjb250ZXh0IjpbImh0dHBzOi8vd3d3LnczLm9yZy8yMDE4L2NyZWRlbnRpYWxzL3YxIiwiaHR0cHM6Ly93d3cudzMub3JnLzIwMTgvY3JlZGVudGlhbHMvZXhhbXBsZXMvdjEiXSwiY3JlZGVudGlhbFN1YmplY3QiOnsiZGVncmVlIjp7InR5cGUiOiJCYWNoZWxvckRlZ3JlZSIsInVuaXZlcnNpdHkiOiJNSVQifSwiaWQiOiJkaWQ6ZXhhbXBsZTplYmZlYjFmNzEyZWJjNmYxYzI3NmUxMmVjMjEiLCJuYW1lIjoiSmF5ZGVuIERvZSIsInNwb3VzZSI6ImRpZDpleGFtcGxlOmMyNzZlMTJlYzIxZWJmZWIxZjcxMmViYzZmMSJ9LCJpc3N1ZXIiOnsibmFtZSI6IkV4YW1wbGUgVW5pdmVyc2l0eSJ9LCJyZWZlcmVuY2VOdW1iZXIiOjguMzI5NDg0OWUrMDcsInR5cGUiOlsiVmVyaWZpYWJsZUNyZWRlbnRpYWwiLCJVbml2ZXJzaXR5RGVncmVlQ3JlZGVudGlhbCJdfX0.3rENm7FBKgw-04J5_RRKS_36UNdiD9DT2-zqdy2mJbpQvKalAt-r49LTSWU50XVxuCUepbo6K_SWDBErsIZ2Aw 354 } 355 356 func ExampleCredential_AddLinkedDataProof() { 357 vc, err := verifiable.ParseCredential([]byte(vcJSON), 358 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader())) 359 if err != nil { 360 panic(fmt.Errorf("failed to decode VC JSON: %w", err)) 361 } 362 363 signer := signature.GetEd25519Signer(issuerPrivKey, issuerPubKey) 364 365 err = vc.AddLinkedDataProof(&verifiable.LinkedDataProofContext{ 366 Created: &issued, 367 SignatureType: "Ed25519Signature2018", 368 Suite: ed25519signature2018.New(suite.WithSigner(signer)), 369 SignatureRepresentation: verifiable.SignatureJWS, 370 VerificationMethod: "did:example:123456#key1", 371 }, jsonld.WithDocumentLoader(getJSONLDDocumentLoader())) 372 if err != nil { 373 panic(fmt.Errorf("failed to add linked data proof: %w", err)) 374 } 375 376 vcJSONWithProof, err := json.MarshalIndent(vc, "", "\t") 377 if err != nil { 378 panic(fmt.Errorf("failed to marshal VC to JSON: %w", err)) 379 } 380 381 fmt.Println(string(vcJSONWithProof)) 382 383 // Output: { 384 // "@context": [ 385 // "https://www.w3.org/2018/credentials/v1", 386 // "https://www.w3.org/2018/credentials/examples/v1" 387 // ], 388 // "credentialSubject": { 389 // "degree": { 390 // "type": "BachelorDegree", 391 // "university": "MIT" 392 // }, 393 // "id": "did:example:ebfeb1f712ebc6f1c276e12ec21", 394 // "name": "Jayden Doe", 395 // "spouse": "did:example:c276e12ec21ebfeb1f712ebc6f1" 396 // }, 397 // "expirationDate": "2020-01-01T19:23:24Z", 398 // "id": "http://example.edu/credentials/1872", 399 // "issuanceDate": "2009-01-01T19:23:24Z", 400 // "issuer": { 401 // "id": "did:example:76e12ec712ebc6f1c221ebfeb1f", 402 // "name": "Example University" 403 // }, 404 // "proof": { 405 // "created": "2010-01-01T19:23:24Z", 406 // "jws": "eyJhbGciOiJFZERTQSIsImI2NCI6ZmFsc2UsImNyaXQiOlsiYjY0Il19..lrkhpRH4tWl6KzQKHlcyAwSm8qUTXIMSKmD3QASF_uI5QW8NWLxLebXmnQpIM8H7umhLA6dINSYVowcaPdpwBw", 407 // "proofPurpose": "assertionMethod", 408 // "type": "Ed25519Signature2018", 409 // "verificationMethod": "did:example:123456#key1" 410 // }, 411 // "referenceNumber": 83294849, 412 // "type": [ 413 // "VerifiableCredential", 414 // "UniversityDegreeCredential" 415 // ] 416 //} 417 } 418 419 //nolint:govet 420 func ExampleCredential_AddLinkedDataProofMultiProofs() { 421 log.SetLevel("aries-framework/json-ld-processor", spi.ERROR) 422 423 vc, err := verifiable.ParseCredential([]byte(vcJSON), 424 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader())) 425 if err != nil { 426 panic(fmt.Errorf("failed to decode VC JSON: %w", err)) 427 } 428 429 ed25519Signer := signature.GetEd25519Signer(issuerPrivKey, issuerPubKey) 430 431 err = vc.AddLinkedDataProof(&verifiable.LinkedDataProofContext{ 432 Created: &issued, 433 SignatureType: "Ed25519Signature2018", 434 Suite: ed25519signature2018.New(suite.WithSigner(ed25519Signer)), 435 SignatureRepresentation: verifiable.SignatureJWS, 436 VerificationMethod: "did:example:123456#key1", 437 }, jsonld.WithDocumentLoader(getJSONLDDocumentLoader())) 438 if err != nil { 439 panic(err) 440 } 441 442 ecdsaSigner, err := signature.NewSigner(kms.ECDSASecp256k1TypeIEEEP1363) 443 if err != nil { 444 panic(err) 445 } 446 447 err = vc.AddLinkedDataProof(&verifiable.LinkedDataProofContext{ 448 Created: &issued, 449 SignatureType: "JsonWebSignature2020", 450 Suite: jsonwebsignature2020.New(suite.WithSigner(ecdsaSigner)), 451 SignatureRepresentation: verifiable.SignatureJWS, 452 VerificationMethod: "did:example:123456#key2", 453 }, jsonld.WithDocumentLoader(getJSONLDDocumentLoader())) 454 if err != nil { 455 panic(err) 456 } 457 458 vcBytes, err := json.Marshal(vc) 459 if err != nil { 460 panic(err) 461 } 462 463 // Verify the VC with two embedded proofs. 464 ed25519Suite := ed25519signature2018.New(suite.WithVerifier(ed25519signature2018.NewPublicKeyVerifier())) 465 jsonWebSignatureSuite := jsonwebsignature2020.New(suite.WithVerifier(jsonwebsignature2020.NewPublicKeyVerifier())) 466 467 j, err := jwksupport.JWKFromKey(ecdsaSigner.PublicKey()) 468 if err != nil { 469 panic(err) 470 } 471 472 _, err = verifiable.ParseCredential(vcBytes, 473 verifiable.WithEmbeddedSignatureSuites(ed25519Suite, jsonWebSignatureSuite), 474 verifiable.WithPublicKeyFetcher(func(issuerID, keyID string) (*sigverifier.PublicKey, error) { 475 switch keyID { 476 case "#key1": 477 return &sigverifier.PublicKey{ 478 Type: "Ed25519Signature2018", 479 Value: issuerPubKey, 480 }, nil 481 482 case "#key2": 483 return &sigverifier.PublicKey{ 484 Type: "JsonWebKey2020", 485 Value: ecdsaSigner.PublicKeyBytes(), 486 JWK: j, 487 }, nil 488 } 489 490 return nil, errors.New("unsupported keyID") 491 }), 492 verifiable.WithJSONLDOnlyValidRDF(), 493 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader())) 494 if err != nil { 495 panic(err) 496 } 497 // Output: 498 } 499 500 //nolint:gocyclo 501 func ExampleCredential_GenerateBBSSelectiveDisclosure() { 502 log.SetLevel("aries-framework/json-ld-processor", spi.ERROR) 503 504 vcStr := ` 505 { 506 "@context": [ 507 "https://www.w3.org/2018/credentials/v1", 508 "https://w3id.org/citizenship/v1", 509 "https://w3id.org/security/bbs/v1" 510 ], 511 "id": "https://issuer.oidp.uscis.gov/credentials/83627465", 512 "type": [ 513 "VerifiableCredential", 514 "PermanentResidentCard" 515 ], 516 "issuer": "did:example:b34ca6cd37bbf23", 517 "identifier": "83627465", 518 "name": "Permanent Resident Card", 519 "description": "Government of Example Permanent Resident Card.", 520 "issuanceDate": "2019-12-03T12:19:52Z", 521 "expirationDate": "2029-12-03T12:19:52Z", 522 "credentialSubject": { 523 "id": "did:example:b34ca6cd37bbf23", 524 "type": [ 525 "PermanentResident", 526 "Person" 527 ], 528 "givenName": "JOHN", 529 "familyName": "SMITH", 530 "gender": "Male", 531 "image": "", 532 "residentSince": "2015-01-01", 533 "lprCategory": "C09", 534 "lprNumber": "999-999-999", 535 "commuterClassification": "C1", 536 "birthCountry": "Bahamas", 537 "birthDate": "1958-07-17" 538 } 539 } 540 ` 541 542 vc, err := verifiable.ParseCredential([]byte(vcStr), 543 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader()), 544 verifiable.WithDisabledProofCheck()) 545 if err != nil { 546 panic(fmt.Errorf("failed to decode VC JSON: %w", err)) 547 } 548 549 ed25519Signer := signature.GetEd25519Signer(issuerPrivKey, issuerPubKey) 550 551 err = vc.AddLinkedDataProof(&verifiable.LinkedDataProofContext{ 552 Created: &issued, 553 SignatureType: "Ed25519Signature2018", 554 Suite: ed25519signature2018.New(suite.WithSigner(ed25519Signer)), 555 SignatureRepresentation: verifiable.SignatureJWS, 556 VerificationMethod: "did:example:123456#key1", 557 }, jsonld.WithDocumentLoader(getJSONLDDocumentLoader())) 558 if err != nil { 559 panic(err) 560 } 561 562 pubKey, privKey, err := loadBBSKeyPair(bbsPubKeyB64, bbsPrivKeyB64) 563 if err != nil { 564 panic(err) 565 } 566 567 bbsSigner, err := newBBSSigner(privKey) 568 if err != nil { 569 panic(err) 570 } 571 572 err = vc.AddLinkedDataProof(&verifiable.LinkedDataProofContext{ 573 Created: &issued, 574 SignatureType: "BbsBlsSignature2020", 575 Suite: bbsblssignature2020.New(suite.WithSigner(bbsSigner)), 576 SignatureRepresentation: verifiable.SignatureProofValue, 577 VerificationMethod: "did:example:123456#key1", 578 }, jsonld.WithDocumentLoader(getJSONLDDocumentLoader())) 579 if err != nil { 580 panic(err) 581 } 582 583 // BBS+ signature is generated each time unique, that's why we substitute it with some constant value 584 // for a reason of keeping constant test output. 585 originalProofValue := hideProofValue(vc.Proofs[1], "dummy signature value") 586 587 vcJSONWithProof, err := json.MarshalIndent(vc, "", "\t") 588 if err != nil { 589 panic(fmt.Errorf("failed to marshal VC to JSON: %w", err)) 590 } 591 592 fmt.Println(string(vcJSONWithProof)) 593 594 restoreProofValue(vc.Proofs[1], originalProofValue) 595 596 // Create BBS+ selective disclosure. We explicitly state the fields we want to reveal in the output document. 597 // For example, "credentialSubject.birthDate" is not mentioned and thus will be hidden. 598 // To hide top-level VC fields, "@explicit": true is used on top level of reveal doc. 599 // For example, we can reveal "identifier" top-level VC field only. "issuer" and "issuanceDate" are mandatory 600 // and thus must be defined in reveal doc in case of hiding top-level VC fields. 601 revealDoc := ` 602 { 603 "@context": [ 604 "https://www.w3.org/2018/credentials/v1", 605 "https://w3id.org/citizenship/v1", 606 "https://w3id.org/security/bbs/v1" 607 ], 608 "type": ["VerifiableCredential", "PermanentResidentCard"], 609 "@explicit": true, 610 "identifier": {}, 611 "issuer": {}, 612 "issuanceDate": {}, 613 "credentialSubject": { 614 "@explicit": true, 615 "type": ["PermanentResident", "Person"], 616 "givenName": {}, 617 "familyName": {}, 618 "gender": {} 619 } 620 } 621 ` 622 623 var revealDocMap map[string]interface{} 624 625 err = json.Unmarshal([]byte(revealDoc), &revealDocMap) 626 if err != nil { 627 panic(err) 628 } 629 630 pubKeyBytes, err := pubKey.Marshal() 631 if err != nil { 632 panic(err) 633 } 634 635 vcWithSelectiveDisclosure, err := vc.GenerateBBSSelectiveDisclosure(revealDocMap, []byte("some nonce"), 636 verifiable.WithJSONLDDocumentLoader(getJSONLDDocumentLoader()), 637 verifiable.WithPublicKeyFetcher(verifiable.SingleKey(pubKeyBytes, "Bls12381G2Key2020"))) 638 if err != nil { 639 panic(err) 640 } 641 642 // Only BBS+ related proof left. 643 hideProofValue(vcWithSelectiveDisclosure.Proofs[0], "dummy signature proof value") 644 645 vcJSONWithProof, err = json.MarshalIndent(vcWithSelectiveDisclosure, "", "\t") 646 if err != nil { 647 panic(fmt.Errorf("failed to marshal VC to JSON: %w", err)) 648 } 649 650 fmt.Println() 651 fmt.Println(string(vcJSONWithProof)) 652 // Output:{ 653 // "@context": [ 654 // "https://www.w3.org/2018/credentials/v1", 655 // "https://w3id.org/citizenship/v1", 656 // "https://w3id.org/security/bbs/v1" 657 // ], 658 // "credentialSubject": { 659 // "birthCountry": "Bahamas", 660 // "birthDate": "1958-07-17", 661 // "commuterClassification": "C1", 662 // "familyName": "SMITH", 663 // "gender": "Male", 664 // "givenName": "JOHN", 665 // "id": "did:example:b34ca6cd37bbf23", 666 // "image": "", 667 // "lprCategory": "C09", 668 // "lprNumber": "999-999-999", 669 // "residentSince": "2015-01-01", 670 // "type": [ 671 // "PermanentResident", 672 // "Person" 673 // ] 674 // }, 675 // "description": "Government of Example Permanent Resident Card.", 676 // "expirationDate": "2029-12-03T12:19:52Z", 677 // "id": "https://issuer.oidp.uscis.gov/credentials/83627465", 678 // "identifier": "83627465", 679 // "issuanceDate": "2019-12-03T12:19:52Z", 680 // "issuer": "did:example:b34ca6cd37bbf23", 681 // "name": "Permanent Resident Card", 682 // "proof": [ 683 // { 684 // "created": "2010-01-01T19:23:24Z", 685 // "jws": "eyJhbGciOiJFZERTQSIsImI2NCI6ZmFsc2UsImNyaXQiOlsiYjY0Il19..HsBapUAZDdaZZy6hrn951768kJaRmNAwTWvVnTDM-Bp5k08eEnnxrii5n47AeWVLDJJo7P0dEPafyC_gMjFPAA", 686 // "proofPurpose": "assertionMethod", 687 // "type": "Ed25519Signature2018", 688 // "verificationMethod": "did:example:123456#key1" 689 // }, 690 // { 691 // "created": "2010-01-01T19:23:24Z", 692 // "proofPurpose": "assertionMethod", 693 // "proofValue": "ZHVtbXkgc2lnbmF0dXJlIHZhbHVl", 694 // "type": "BbsBlsSignature2020", 695 // "verificationMethod": "did:example:123456#key1" 696 // } 697 // ], 698 // "type": [ 699 // "VerifiableCredential", 700 // "PermanentResidentCard" 701 // ] 702 //} 703 // 704 //{ 705 // "@context": [ 706 // "https://www.w3.org/2018/credentials/v1", 707 // "https://w3id.org/citizenship/v1", 708 // "https://w3id.org/security/bbs/v1" 709 // ], 710 // "credentialSubject": { 711 // "familyName": "SMITH", 712 // "gender": "Male", 713 // "givenName": "JOHN", 714 // "id": "did:example:b34ca6cd37bbf23", 715 // "type": [ 716 // "Person", 717 // "PermanentResident" 718 // ] 719 // }, 720 // "id": "https://issuer.oidp.uscis.gov/credentials/83627465", 721 // "identifier": "83627465", 722 // "issuanceDate": "2019-12-03T12:19:52Z", 723 // "issuer": "did:example:b34ca6cd37bbf23", 724 // "proof": { 725 // "created": "2010-01-01T19:23:24Z", 726 // "nonce": "c29tZSBub25jZQ==", 727 // "proofPurpose": "assertionMethod", 728 // "proofValue": "ZHVtbXkgc2lnbmF0dXJlIHByb29mIHZhbHVl", 729 // "type": "BbsBlsSignatureProof2020", 730 // "verificationMethod": "did:example:123456#key1" 731 // }, 732 // "type": [ 733 // "PermanentResidentCard", 734 // "VerifiableCredential" 735 // ] 736 //} 737 } 738 739 func hideProofValue(proof verifiable.Proof, dummyValue string) interface{} { 740 oldProofValue := proof["proofValue"] 741 proof["proofValue"] = base64.StdEncoding.EncodeToString([]byte(dummyValue)) 742 743 return oldProofValue 744 } 745 746 func restoreProofValue(proof verifiable.Proof, proofValue interface{}) { 747 proof["proofValue"] = proofValue 748 }