From 1cbdd80f1d45f8dd20c54b514e57d7bc5fbd3604 Mon Sep 17 00:00:00 2001 From: Julien Fraichot Date: Mon, 2 Dec 2024 14:28:52 +0100 Subject: [PATCH 1/2] feat(DataIntergrity): perform check for proofPurpose --- src/index.ts | 12 +++++++++++- .../data-integrity-proof-support.test.ts | 16 +++++++++++++++- 2 files changed, 26 insertions(+), 2 deletions(-) diff --git a/src/index.ts b/src/index.ts index 4058d45..fe0cb98 100644 --- a/src/index.ts +++ b/src/index.ts @@ -34,6 +34,8 @@ export interface MerkleProof2019API { verificationMethod?: IDidDocumentPublicKey; document: VCDocument; proof?: VCProof; + // the purpose of proof that the verifier will be used for, defaults to assertionMethod + proofPurpose?: string; } export interface MerkleProof2019VerificationResult { @@ -61,6 +63,7 @@ export class LDMerkleProof2019 extends LinkedDataProof { public verificationMethod: IDidDocumentPublicKey = null; public proof: VCProof = null; public proofValue: DecodedProof = null; + public proofPurpose: string; public document: VCDocument = null; public explorerAPIs: ExplorerAPI[] = []; public chain: IBlockchainObject; @@ -89,7 +92,8 @@ export class LDMerkleProof2019 extends LinkedDataProof { verificationMethod = null, document = null, proof = null, - options = {} + options = {}, + proofPurpose = 'assertionMethod' }: MerkleProof2019API) { super({ type: 'MerkleProof2019' }); @@ -100,6 +104,7 @@ export class LDMerkleProof2019 extends LinkedDataProof { this.issuer = issuer; this.verificationMethod = verificationMethod; this.document = document; + this.proofPurpose = proofPurpose; this.setProof(proof); this.setOptions(options); this.getChain(); @@ -131,6 +136,11 @@ export class LDMerkleProof2019 extends LinkedDataProof { this.documentLoader = documentLoader; let verified: boolean; let error: string = ''; + + if (this.proof.proofPurpose && this.proof.proofPurpose !== this.proofPurpose) { + throw new Error(`Invalid proof purpose. Expected ${this.proofPurpose} but received ${this.proof.proofPurpose}`); + } + try { await this.verifyProcess(this.proofVerificationProcess); if (verifyIdentity) { diff --git a/tests/verification/data-integrity-proof-support.test.ts b/tests/verification/data-integrity-proof-support.test.ts index d6adea1..c8d66db 100644 --- a/tests/verification/data-integrity-proof-support.test.ts +++ b/tests/verification/data-integrity-proof-support.test.ts @@ -2,7 +2,7 @@ import { describe, it, expect } from 'vitest'; import fixture from '../fixtures/mocknet-vc-v2-data-integrity-proof.json'; import { LDMerkleProof2019 } from '../../src'; -describe('given the documnet is signed following the DataIntegrityProof spec', function () { +describe('given the document is signed following the DataIntegrityProof spec', function () { describe('and is a valid MerkleProof2019 signature', function () { it('should verify successfully', async function () { const instance = new LDMerkleProof2019({ @@ -17,4 +17,18 @@ describe('given the documnet is signed following the DataIntegrityProof spec', f }); }); }); + + describe('given the proofPurpose does not match the one of the verifier', function () { + it('should throw an error', async function () { + const instance = new LDMerkleProof2019({ + document: fixture, + proof: fixture.proof, + proofPurpose: 'authentication' + }); + + await expect(async () => { + await instance.verifyProof(); + }).rejects.toThrow('Invalid proof purpose. Expected authentication but received assertionMethod'); + }); + }); }); From 7f12d9789091f346011afd9e47d9ba21c616f0e9 Mon Sep 17 00:00:00 2001 From: Julien Fraichot Date: Mon, 2 Dec 2024 15:55:14 +0100 Subject: [PATCH 2/2] feat(DataIntergrity): control issuer's key is approved for purpose --- src/index.ts | 10 +- tests/fixtures/issuer-blockcerts.json | 98 +++++++++++++++++++ .../data-integrity-proof-support.test.ts | 21 +++- 3 files changed, 126 insertions(+), 3 deletions(-) create mode 100644 tests/fixtures/issuer-blockcerts.json diff --git a/src/index.ts b/src/index.ts index fe0cb98..6beacb7 100644 --- a/src/index.ts +++ b/src/index.ts @@ -137,8 +137,14 @@ export class LDMerkleProof2019 extends LinkedDataProof { let verified: boolean; let error: string = ''; - if (this.proof.proofPurpose && this.proof.proofPurpose !== this.proofPurpose) { - throw new Error(`Invalid proof purpose. Expected ${this.proofPurpose} but received ${this.proof.proofPurpose}`); + if (this.proof.proofPurpose) { + if (this.proof.proofPurpose !== this.proofPurpose) { + throw new Error(`Invalid proof purpose. Expected ${this.proofPurpose} but received ${this.proof.proofPurpose}`); + } + + if (this.issuer && !this.issuer[this.proofPurpose]?.includes(this.proof.verificationMethod)) { + throw new Error(`The verification method ${this.proof.verificationMethod} is not allowed for the proof purpose ${this.proofPurpose}`); + } } try { diff --git a/tests/fixtures/issuer-blockcerts.json b/tests/fixtures/issuer-blockcerts.json new file mode 100644 index 0000000..0fa0208 --- /dev/null +++ b/tests/fixtures/issuer-blockcerts.json @@ -0,0 +1,98 @@ +{ + "@context": [ + "https://www.w3.org/ns/did/v1", + "https://w3id.org/openbadges/v2", + "https://w3id.org/blockcerts/v3" + ], + "type": "Profile", + "id": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json", + "name": "Blockcerts Organization", + "url": "https://www.blockcerts.org", + "publicKey": [ + { + "id": "ecdsa-koblitz-pubkey:mgdWjvq4RYAAP5goUNagTRMx7Xw534S5am", + "created": "2021-11-20T15:52:57.617381+00:00" + }, + { + "id": "ecdsa-koblitz-pubkey:1862cjGVHodmYyvfumSgrgfnWcpCMHK9sq", + "created": "2021-11-20T15:52:58.617381+00:00" + }, + { + "id": "ecdsa-koblitz-pubkey:0x40Cf9B7DB6FCc742ad0a76B8588C7f8De2b54a60", + "created": "2021-11-28T15:52:58.617381+00:00" + } + ], + "verificationMethod": [ + { + "id": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#key-1", + "controller": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json", + "type": "EcdsaSecp256k1VerificationKey2019", + "publicKeyJwk": { + "kty": "EC", + "crv": "secp256k1", + "x": "6_JKJIXL7PJQT9hnr03yQNda_fUfmfrcZpymkRqsmH4", + "y": "steT4D8LrgwmqASd1EMy6ZyyAqsl-KvNlD7rBhX3za8", + "kid": "_0qG5QVt8vd6pbVGs5ReFJA4-yvYNEi4Ov1HZHTsobM" + } + }, + { + "id": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#key-1-btc", + "controller": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json", + "type": "EcdsaSecp256k1VerificationKey2019", + "publicKeyJwk": { + "kty": "EC", + "crv": "secp256k1", + "x": "9PchaXjrV0R2nOQrIRKC7hOcX1oxkqNJSnI2sjMQEX8", + "y": "RsoYEcz5Hp7WNsOfhKQoEPpXtKHXoGPMA7slpVLxLnE", + "kid": "HL3p6Wtct4Tcto1NU1fNzSH-0353orBBSYMYtJnQt7E" + } + }, + { + "id": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#secp256k1-verification-public-key", + "type": "JsonWebKey2020", + "controller": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json", + "publicKeyJwk": { + "kty": "EC", + "crv": "secp256k1", + "x": "BpBt2iJlQGm3QpG5r0mJNDxwXXFlRAWdZsHWWIK_HBY", + "y": "tlbW6AMB3MmRk1JbWNZhc77q8TnSxgJidpvyA0rH0BI" + } + }, + { + "id": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#secp256k1-derived-public-key", + "controller": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json", + "type": "EcdsaSecp256k1VerificationKey2019", + "publicKeyJwk": { + "kty": "EC", + "crv": "secp256k1", + "x": "YHzDbKmTSWUisCw4AzY4hzOWDsjM-O8tyJTnFTSR79g", + "y": "bWCdCls4ukZq7KyWCTOJ8Sta_PJHvCdDIeiRaBD7e98" + } + }, + { + "id": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#key-1-multibase", + "controller": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json", + "type": "Multikey", + "publicKeyMultibase": "zQ3shvX9Dd7cAG7ZcJN4d9DksshpVYSGpqEyrLjopoGpk97CR", + "keyAgreement": false + }, + { + "id": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#btc-main-multibase", + "controller": "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json", + "type": "Multikey", + "publicKeyMultibase": "zQ3shw8MAkueKou9VhRyX1v2hDQ2WENWVQNkg6ifhn8DG1gQW", + "keyAgreement": false + } + ], + "assertionMethod": [ + "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#key-1", + "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#key-1-btc", + "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#secp256k1-verification-public-key", + "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#secp256k1-derived-public-key", + "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#key-1-multibase", + "https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#btc-main-multibase" + ], + "revocationList": "https://www.blockcerts.org/samples/3.0/revocation-list-blockcerts.json", + "email": "", + "image": "" +} diff --git a/tests/verification/data-integrity-proof-support.test.ts b/tests/verification/data-integrity-proof-support.test.ts index c8d66db..a82beb2 100644 --- a/tests/verification/data-integrity-proof-support.test.ts +++ b/tests/verification/data-integrity-proof-support.test.ts @@ -1,6 +1,7 @@ import { describe, it, expect } from 'vitest'; import fixture from '../fixtures/mocknet-vc-v2-data-integrity-proof.json'; import { LDMerkleProof2019 } from '../../src'; +import fixtureIssuerProfile from '../fixtures/issuer-blockcerts.json'; describe('given the document is signed following the DataIntegrityProof spec', function () { describe('and is a valid MerkleProof2019 signature', function () { @@ -18,7 +19,7 @@ describe('given the document is signed following the DataIntegrityProof spec', f }); }); - describe('given the proofPurpose does not match the one of the verifier', function () { + describe('given the proofPurpose of the proof does not match the one of the verifier', function () { it('should throw an error', async function () { const instance = new LDMerkleProof2019({ document: fixture, @@ -31,4 +32,22 @@ describe('given the document is signed following the DataIntegrityProof spec', f }).rejects.toThrow('Invalid proof purpose. Expected authentication but received assertionMethod'); }); }); + + describe('given the proofPurpose of the proof does not match the one of the issuer\'s key', function () { + it('should throw an error', async function () { + const instance = new LDMerkleProof2019({ + document: fixture, + proof: { + ...fixture.proof, + proofPurpose: 'authentication' + }, + proofPurpose: 'authentication', + issuer: fixtureIssuerProfile + }); + + await expect(async () => { + await instance.verifyProof(); + }).rejects.toThrow('The verification method https://www.blockcerts.org/samples/3.0/issuer-blockcerts.json#key-1 is not allowed for the proof purpose authentication'); + }); + }); });