@originals/sdk 1.1.0 → 1.4.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +3 -2
- package/src/did/DIDManager.ts +1 -1
- package/src/did/WebVHManager.ts +11 -2
- package/src/examples/create-module-original.ts +435 -0
- package/src/examples/full-lifecycle-flow.ts +514 -0
- package/src/examples/run.ts +59 -4
- package/src/index.ts +69 -3
- package/src/kinds/KindRegistry.ts +290 -0
- package/src/kinds/index.ts +74 -0
- package/src/kinds/types.ts +470 -0
- package/src/kinds/validators/AgentValidator.ts +257 -0
- package/src/kinds/validators/AppValidator.ts +211 -0
- package/src/kinds/validators/DatasetValidator.ts +242 -0
- package/src/kinds/validators/DocumentValidator.ts +311 -0
- package/src/kinds/validators/MediaValidator.ts +269 -0
- package/src/kinds/validators/ModuleValidator.ts +225 -0
- package/src/kinds/validators/base.ts +276 -0
- package/src/kinds/validators/index.ts +12 -0
- package/src/lifecycle/LifecycleManager.ts +909 -1
- package/src/resources/ResourceManager.ts +655 -0
- package/src/resources/index.ts +21 -0
- package/src/resources/types.ts +202 -0
- package/src/types/common.ts +1 -1
- package/src/vc/CredentialManager.ts +647 -2
- package/tests/integration/createTypedOriginal.test.ts +379 -0
- package/tests/performance/BatchOperations.perf.test.ts +2 -2
- package/tests/unit/kinds/KindRegistry.test.ts +329 -0
- package/tests/unit/kinds/types.test.ts +409 -0
- package/tests/unit/kinds/validators.test.ts +651 -0
- package/tests/unit/lifecycle/LifecycleManager.cleanapi.test.ts +441 -0
- package/tests/unit/resources/ResourceManager.test.ts +740 -0
- package/tests/unit/vc/CredentialManager.helpers.test.ts +527 -0
- package/tsconfig.json +0 -1
- package/.turbo/turbo-build.log +0 -1
- package/.turbo/turbo-test.log +0 -68353
- package/dist/adapters/FeeOracleMock.d.ts +0 -6
- package/dist/adapters/FeeOracleMock.js +0 -8
- package/dist/adapters/index.d.ts +0 -4
- package/dist/adapters/index.js +0 -4
- package/dist/adapters/providers/OrdHttpProvider.d.ts +0 -56
- package/dist/adapters/providers/OrdHttpProvider.js +0 -110
- package/dist/adapters/providers/OrdMockProvider.d.ts +0 -70
- package/dist/adapters/providers/OrdMockProvider.js +0 -75
- package/dist/adapters/types.d.ts +0 -71
- package/dist/adapters/types.js +0 -1
- package/dist/bitcoin/BitcoinManager.d.ts +0 -15
- package/dist/bitcoin/BitcoinManager.js +0 -262
- package/dist/bitcoin/BroadcastClient.d.ts +0 -30
- package/dist/bitcoin/BroadcastClient.js +0 -35
- package/dist/bitcoin/OrdinalsClient.d.ts +0 -21
- package/dist/bitcoin/OrdinalsClient.js +0 -105
- package/dist/bitcoin/PSBTBuilder.d.ts +0 -24
- package/dist/bitcoin/PSBTBuilder.js +0 -80
- package/dist/bitcoin/fee-calculation.d.ts +0 -14
- package/dist/bitcoin/fee-calculation.js +0 -31
- package/dist/bitcoin/providers/OrdNodeProvider.d.ts +0 -38
- package/dist/bitcoin/providers/OrdNodeProvider.js +0 -67
- package/dist/bitcoin/providers/OrdinalsProvider.d.ts +0 -33
- package/dist/bitcoin/providers/OrdinalsProvider.js +0 -50
- package/dist/bitcoin/providers/types.d.ts +0 -63
- package/dist/bitcoin/providers/types.js +0 -1
- package/dist/bitcoin/transactions/commit.d.ts +0 -89
- package/dist/bitcoin/transactions/commit.js +0 -311
- package/dist/bitcoin/transactions/index.d.ts +0 -7
- package/dist/bitcoin/transactions/index.js +0 -8
- package/dist/bitcoin/transfer.d.ts +0 -9
- package/dist/bitcoin/transfer.js +0 -26
- package/dist/bitcoin/utxo-selection.d.ts +0 -78
- package/dist/bitcoin/utxo-selection.js +0 -237
- package/dist/bitcoin/utxo.d.ts +0 -26
- package/dist/bitcoin/utxo.js +0 -78
- package/dist/contexts/credentials-v1.json +0 -195
- package/dist/contexts/credentials-v2-examples.json +0 -5
- package/dist/contexts/credentials-v2.json +0 -301
- package/dist/contexts/credentials.json +0 -195
- package/dist/contexts/data-integrity-v2.json +0 -81
- package/dist/contexts/dids.json +0 -57
- package/dist/contexts/ed255192020.json +0 -93
- package/dist/contexts/ordinals-plus.json +0 -23
- package/dist/contexts/originals.json +0 -22
- package/dist/core/OriginalsSDK.d.ts +0 -158
- package/dist/core/OriginalsSDK.js +0 -274
- package/dist/crypto/Multikey.d.ts +0 -30
- package/dist/crypto/Multikey.js +0 -149
- package/dist/crypto/Signer.d.ts +0 -21
- package/dist/crypto/Signer.js +0 -196
- package/dist/crypto/noble-init.d.ts +0 -18
- package/dist/crypto/noble-init.js +0 -106
- package/dist/did/BtcoDidResolver.d.ts +0 -57
- package/dist/did/BtcoDidResolver.js +0 -166
- package/dist/did/DIDManager.d.ts +0 -101
- package/dist/did/DIDManager.js +0 -493
- package/dist/did/Ed25519Verifier.d.ts +0 -30
- package/dist/did/Ed25519Verifier.js +0 -59
- package/dist/did/KeyManager.d.ts +0 -17
- package/dist/did/KeyManager.js +0 -207
- package/dist/did/WebVHManager.d.ts +0 -100
- package/dist/did/WebVHManager.js +0 -304
- package/dist/did/createBtcoDidDocument.d.ts +0 -10
- package/dist/did/createBtcoDidDocument.js +0 -42
- package/dist/did/providers/OrdinalsClientProviderAdapter.d.ts +0 -23
- package/dist/did/providers/OrdinalsClientProviderAdapter.js +0 -51
- package/dist/events/EventEmitter.d.ts +0 -115
- package/dist/events/EventEmitter.js +0 -198
- package/dist/events/index.d.ts +0 -7
- package/dist/events/index.js +0 -6
- package/dist/events/types.d.ts +0 -286
- package/dist/events/types.js +0 -9
- package/dist/examples/basic-usage.d.ts +0 -3
- package/dist/examples/basic-usage.js +0 -62
- package/dist/examples/run.d.ts +0 -1
- package/dist/examples/run.js +0 -4
- package/dist/index.d.ts +0 -39
- package/dist/index.js +0 -47
- package/dist/lifecycle/BatchOperations.d.ts +0 -147
- package/dist/lifecycle/BatchOperations.js +0 -251
- package/dist/lifecycle/LifecycleManager.d.ts +0 -116
- package/dist/lifecycle/LifecycleManager.js +0 -971
- package/dist/lifecycle/OriginalsAsset.d.ts +0 -164
- package/dist/lifecycle/OriginalsAsset.js +0 -380
- package/dist/lifecycle/ProvenanceQuery.d.ts +0 -126
- package/dist/lifecycle/ProvenanceQuery.js +0 -220
- package/dist/lifecycle/ResourceVersioning.d.ts +0 -73
- package/dist/lifecycle/ResourceVersioning.js +0 -127
- package/dist/migration/MigrationManager.d.ts +0 -86
- package/dist/migration/MigrationManager.js +0 -412
- package/dist/migration/audit/AuditLogger.d.ts +0 -51
- package/dist/migration/audit/AuditLogger.js +0 -156
- package/dist/migration/checkpoint/CheckpointManager.d.ts +0 -31
- package/dist/migration/checkpoint/CheckpointManager.js +0 -96
- package/dist/migration/checkpoint/CheckpointStorage.d.ts +0 -26
- package/dist/migration/checkpoint/CheckpointStorage.js +0 -89
- package/dist/migration/index.d.ts +0 -22
- package/dist/migration/index.js +0 -27
- package/dist/migration/operations/BaseMigration.d.ts +0 -48
- package/dist/migration/operations/BaseMigration.js +0 -83
- package/dist/migration/operations/PeerToBtcoMigration.d.ts +0 -25
- package/dist/migration/operations/PeerToBtcoMigration.js +0 -67
- package/dist/migration/operations/PeerToWebvhMigration.d.ts +0 -19
- package/dist/migration/operations/PeerToWebvhMigration.js +0 -46
- package/dist/migration/operations/WebvhToBtcoMigration.d.ts +0 -25
- package/dist/migration/operations/WebvhToBtcoMigration.js +0 -67
- package/dist/migration/rollback/RollbackManager.d.ts +0 -29
- package/dist/migration/rollback/RollbackManager.js +0 -146
- package/dist/migration/state/StateMachine.d.ts +0 -25
- package/dist/migration/state/StateMachine.js +0 -76
- package/dist/migration/state/StateTracker.d.ts +0 -36
- package/dist/migration/state/StateTracker.js +0 -123
- package/dist/migration/types.d.ts +0 -306
- package/dist/migration/types.js +0 -33
- package/dist/migration/validation/BitcoinValidator.d.ts +0 -13
- package/dist/migration/validation/BitcoinValidator.js +0 -83
- package/dist/migration/validation/CredentialValidator.d.ts +0 -13
- package/dist/migration/validation/CredentialValidator.js +0 -46
- package/dist/migration/validation/DIDCompatibilityValidator.d.ts +0 -16
- package/dist/migration/validation/DIDCompatibilityValidator.js +0 -127
- package/dist/migration/validation/LifecycleValidator.d.ts +0 -10
- package/dist/migration/validation/LifecycleValidator.js +0 -52
- package/dist/migration/validation/StorageValidator.d.ts +0 -10
- package/dist/migration/validation/StorageValidator.js +0 -65
- package/dist/migration/validation/ValidationPipeline.d.ts +0 -29
- package/dist/migration/validation/ValidationPipeline.js +0 -180
- package/dist/storage/LocalStorageAdapter.d.ts +0 -11
- package/dist/storage/LocalStorageAdapter.js +0 -53
- package/dist/storage/MemoryStorageAdapter.d.ts +0 -6
- package/dist/storage/MemoryStorageAdapter.js +0 -21
- package/dist/storage/StorageAdapter.d.ts +0 -16
- package/dist/storage/StorageAdapter.js +0 -1
- package/dist/storage/index.d.ts +0 -2
- package/dist/storage/index.js +0 -2
- package/dist/types/bitcoin.d.ts +0 -84
- package/dist/types/bitcoin.js +0 -1
- package/dist/types/common.d.ts +0 -82
- package/dist/types/common.js +0 -1
- package/dist/types/credentials.d.ts +0 -75
- package/dist/types/credentials.js +0 -1
- package/dist/types/did.d.ts +0 -26
- package/dist/types/did.js +0 -1
- package/dist/types/index.d.ts +0 -5
- package/dist/types/index.js +0 -5
- package/dist/types/network.d.ts +0 -78
- package/dist/types/network.js +0 -145
- package/dist/utils/EventLogger.d.ts +0 -71
- package/dist/utils/EventLogger.js +0 -232
- package/dist/utils/Logger.d.ts +0 -106
- package/dist/utils/Logger.js +0 -257
- package/dist/utils/MetricsCollector.d.ts +0 -110
- package/dist/utils/MetricsCollector.js +0 -264
- package/dist/utils/bitcoin-address.d.ts +0 -38
- package/dist/utils/bitcoin-address.js +0 -113
- package/dist/utils/cbor.d.ts +0 -2
- package/dist/utils/cbor.js +0 -9
- package/dist/utils/encoding.d.ts +0 -37
- package/dist/utils/encoding.js +0 -120
- package/dist/utils/hash.d.ts +0 -1
- package/dist/utils/hash.js +0 -5
- package/dist/utils/retry.d.ts +0 -10
- package/dist/utils/retry.js +0 -35
- package/dist/utils/satoshi-validation.d.ts +0 -60
- package/dist/utils/satoshi-validation.js +0 -156
- package/dist/utils/serialization.d.ts +0 -14
- package/dist/utils/serialization.js +0 -76
- package/dist/utils/telemetry.d.ts +0 -17
- package/dist/utils/telemetry.js +0 -24
- package/dist/utils/validation.d.ts +0 -5
- package/dist/utils/validation.js +0 -98
- package/dist/vc/CredentialManager.d.ts +0 -22
- package/dist/vc/CredentialManager.js +0 -227
- package/dist/vc/Issuer.d.ts +0 -27
- package/dist/vc/Issuer.js +0 -70
- package/dist/vc/Verifier.d.ts +0 -16
- package/dist/vc/Verifier.js +0 -50
- package/dist/vc/cryptosuites/bbs.d.ts +0 -44
- package/dist/vc/cryptosuites/bbs.js +0 -213
- package/dist/vc/cryptosuites/bbsSimple.d.ts +0 -9
- package/dist/vc/cryptosuites/bbsSimple.js +0 -12
- package/dist/vc/cryptosuites/eddsa.d.ts +0 -30
- package/dist/vc/cryptosuites/eddsa.js +0 -81
- package/dist/vc/documentLoader.d.ts +0 -16
- package/dist/vc/documentLoader.js +0 -59
- package/dist/vc/proofs/data-integrity.d.ts +0 -21
- package/dist/vc/proofs/data-integrity.js +0 -15
- package/dist/vc/utils/jsonld.d.ts +0 -2
- package/dist/vc/utils/jsonld.js +0 -15
- package/test/logs/did_webvh_QmQsRNhXxPSCSeLjpbKYcNMZj8b1kBQAoC6cZmkFAgmpHt_example_com.jsonl +0 -1
- package/test/logs/did_webvh_QmSQkpD58qxcqMWHYcEmDUn3wk7hHvJwzYTrZmhh6zjPQ8_example_com_users_alice123_profile.jsonl +0 -1
- package/test/logs/did_webvh_QmTMda6VW3cUPdKk5Yc3onnv1vdgEumvWWdP2noAYFSjeG_example_com.jsonl +0 -1
- package/test/logs/did_webvh_QmTkb8KnCYcsnKKDCY4eUQuKQdKJLrCinvhw13v3zETxpE_example_com_users_etc_passwd.jsonl +0 -1
- package/test/logs/did_webvh_QmTn9FdCfpXFDrxHH52pwB4iNrDFVvNDjJ5FQTcDbmM3Fg_example_com.jsonl +0 -1
- package/test/logs/did_webvh_QmUCQUi1xjtJjnSQ1XJZgKqcWgErx1v7E2dz4DAPraAyJP_example_com_etc_passwd.jsonl +0 -1
- package/test/logs/did_webvh_QmUENQJCDKBJVRS5BkL6zjaUvcRjkb9xHmy7foCgRjmv3W_example_com.jsonl +0 -1
- package/test/logs/did_webvh_QmUPdGyjYBEnQ3aQUkmqyyBKTyjvCP5RZQGiaEDeTtf6dc_example_com.jsonl +0 -1
- package/test/logs/did_webvh_QmUoHTuHMWzQM29ZFrE9VLtMxkZ5u869yqee8LwcCLN39M_example_com.jsonl +0 -1
- package/test/logs/did_webvh_QmUrnms8G65ggVKsr9oQeWrLUBuGChwQPPb2LCFvaoNxaw_example_com_users_alice.jsonl +0 -1
- package/test/logs/did_webvh_QmUwiw3eSXdHG1hPvoAGu3cuK5jF4aXRYDLBAjPXfv1qzb_example_com_level1_level2.jsonl +0 -1
- package/test/logs/did_webvh_QmW7bzKh6yFEKNAtmVsrPGvvsMHTUQdzJSNsTZkbuGFpbj_example_com_secret.jsonl +0 -1
- package/test/logs/did_webvh_QmXbFTFBBJ8zpjdz9WE1DNN44A2wprFmdvAubjSffeyoAG_example_com.jsonl +0 -1
- package/test/logs/did_webvh_QmXyVXFPCTffGb2mTUFDeMCsScjnpLWkyUkVkB6q6QoeBf_example_com_C_Windows_System32.jsonl +0 -1
- package/test/logs/did_webvh_QmZK9B81gxZtvo5fYHYKDtKt8zZfZZPhmCMhbujBJuRRzE_example_com_etc_passwd.jsonl +0 -1
- package/test/logs/did_webvh_QmbNLCVSdXSVLrwFBvCBQPAabjtRb1SGHjkGVyw3QUbfBL_example_com_users_etc_passwd.jsonl +0 -1
- package/test/logs/did_webvh_QmbeaicmGW3Q7Yzbqmftc8a9jLBngokveb5A2KVKfVGZRb_example_com_my_org_user_name_test_123.jsonl +0 -1
- package/test/logs/did_webvh_Qmdv7c7AjUreUfoKyvkN2UpAWTozxKsv99srQetPJMJEnp_example_com_users_etc_passwd.jsonl +0 -1
- package/test/logs/did_webvh_QmeioWY3uypYLkYpCXe9eCYnn4xBVruP9C1d79azMrTEHG_example_com.jsonl +0 -1
- package/test/logs/did_webvh_Qmf4QH5dsA6Ecr5HJ6KaJL9uJRyY8RxrQdqoRCM25DzvPi_example_com_users_alice.jsonl +0 -1
- package/tests/e2e/README.md +0 -97
- package/tests/e2e/example.spec.ts +0 -78
- package/tsconfig.tsbuildinfo +0 -1
|
@@ -4,16 +4,149 @@ import {
|
|
|
4
4
|
CredentialSubject,
|
|
5
5
|
OriginalsConfig,
|
|
6
6
|
Proof,
|
|
7
|
-
ExternalSigner
|
|
7
|
+
ExternalSigner,
|
|
8
|
+
LayerType,
|
|
9
|
+
AssetResource
|
|
8
10
|
} from '../types';
|
|
9
11
|
import { canonicalizeDocument } from '../utils/serialization';
|
|
10
12
|
import { encodeBase64UrlMultibase, decodeBase64UrlMultibase } from '../utils/encoding';
|
|
11
13
|
import { sha256 } from '@noble/hashes/sha2.js';
|
|
14
|
+
import { bytesToHex } from '@noble/hashes/utils.js';
|
|
12
15
|
import { Signer, ES256KSigner, Ed25519Signer, ES256Signer } from '../crypto/Signer';
|
|
13
16
|
import { DIDManager } from '../did/DIDManager';
|
|
14
17
|
import { Issuer, VerificationMethodLike } from './Issuer';
|
|
15
18
|
import { createDocumentLoader } from './documentLoader';
|
|
16
19
|
import { Verifier } from './Verifier';
|
|
20
|
+
import { BBSCryptosuiteUtils } from './cryptosuites/bbs';
|
|
21
|
+
|
|
22
|
+
// ===== Credential Factory Types =====
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Subject data for a ResourceCreated credential
|
|
26
|
+
*/
|
|
27
|
+
export interface ResourceCreatedSubject {
|
|
28
|
+
/** ID of the subject (typically the resource DID or asset DID) */
|
|
29
|
+
id: string;
|
|
30
|
+
/** Resource identifier */
|
|
31
|
+
resourceId: string;
|
|
32
|
+
/** Resource type (e.g., 'code', 'text', 'image') */
|
|
33
|
+
resourceType: string;
|
|
34
|
+
/** Content hash of the resource */
|
|
35
|
+
contentHash: string;
|
|
36
|
+
/** MIME content type */
|
|
37
|
+
contentType: string;
|
|
38
|
+
/** Creator DID */
|
|
39
|
+
creator: string;
|
|
40
|
+
/** Creation timestamp */
|
|
41
|
+
createdAt: string;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Subject data for a ResourceUpdated credential
|
|
46
|
+
*/
|
|
47
|
+
export interface ResourceUpdatedSubject {
|
|
48
|
+
/** ID of the subject (typically the asset DID) */
|
|
49
|
+
id: string;
|
|
50
|
+
/** Resource identifier */
|
|
51
|
+
resourceId: string;
|
|
52
|
+
/** Previous content hash */
|
|
53
|
+
previousHash: string;
|
|
54
|
+
/** New content hash */
|
|
55
|
+
newHash: string;
|
|
56
|
+
/** Previous version number */
|
|
57
|
+
fromVersion: number;
|
|
58
|
+
/** New version number */
|
|
59
|
+
toVersion: number;
|
|
60
|
+
/** Update timestamp */
|
|
61
|
+
updatedAt: string;
|
|
62
|
+
/** Optional description of changes */
|
|
63
|
+
updateReason?: string;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* Subject data for a MigrationCompleted credential
|
|
68
|
+
*/
|
|
69
|
+
export interface MigrationSubject {
|
|
70
|
+
/** ID of the subject (typically the asset DID) */
|
|
71
|
+
id: string;
|
|
72
|
+
/** Source DID (before migration) */
|
|
73
|
+
sourceDid: string;
|
|
74
|
+
/** Target DID (after migration) */
|
|
75
|
+
targetDid?: string;
|
|
76
|
+
/** Layer migrated from */
|
|
77
|
+
fromLayer: LayerType;
|
|
78
|
+
/** Layer migrated to */
|
|
79
|
+
toLayer: LayerType;
|
|
80
|
+
/** Migration timestamp */
|
|
81
|
+
migratedAt: string;
|
|
82
|
+
/** Transaction ID (for Bitcoin migrations) */
|
|
83
|
+
transactionId?: string;
|
|
84
|
+
/** Inscription ID (for Bitcoin migrations) */
|
|
85
|
+
inscriptionId?: string;
|
|
86
|
+
/** Satoshi number (for Bitcoin migrations) */
|
|
87
|
+
satoshi?: string;
|
|
88
|
+
/** Optional reason for migration */
|
|
89
|
+
migrationReason?: string;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Subject data for an OwnershipTransferred credential
|
|
94
|
+
*/
|
|
95
|
+
export interface OwnershipSubject {
|
|
96
|
+
/** ID of the subject (typically the asset DID) */
|
|
97
|
+
id: string;
|
|
98
|
+
/** Previous owner DID or address */
|
|
99
|
+
previousOwner: string;
|
|
100
|
+
/** New owner DID or address */
|
|
101
|
+
newOwner: string;
|
|
102
|
+
/** Transfer timestamp */
|
|
103
|
+
transferredAt: string;
|
|
104
|
+
/** Transaction ID for the transfer */
|
|
105
|
+
transactionId: string;
|
|
106
|
+
/** Satoshi number of the inscription */
|
|
107
|
+
satoshi?: string;
|
|
108
|
+
/** Optional transfer reason or notes */
|
|
109
|
+
transferReason?: string;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
/**
|
|
113
|
+
* Options for creating credentials with chaining
|
|
114
|
+
*/
|
|
115
|
+
export interface CredentialChainOptions {
|
|
116
|
+
/** Previous credential ID to chain from */
|
|
117
|
+
previousCredentialId?: string;
|
|
118
|
+
/** Hash of the previous credential for verification */
|
|
119
|
+
previousCredentialHash?: string;
|
|
120
|
+
/** Optional expiration date */
|
|
121
|
+
expirationDate?: string;
|
|
122
|
+
/** Optional credential status information */
|
|
123
|
+
credentialStatus?: {
|
|
124
|
+
id: string;
|
|
125
|
+
type: string;
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
/**
|
|
130
|
+
* Options for BBS+ selective disclosure
|
|
131
|
+
*/
|
|
132
|
+
export interface SelectiveDisclosureOptions {
|
|
133
|
+
/** JSON Pointer paths to fields that must always be disclosed */
|
|
134
|
+
mandatoryPointers: string[];
|
|
135
|
+
/** JSON Pointer paths to fields the holder can selectively disclose */
|
|
136
|
+
selectivePointers?: string[];
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
/**
|
|
140
|
+
* Result of creating a derived proof with selective disclosure
|
|
141
|
+
*/
|
|
142
|
+
export interface DerivedProofResult {
|
|
143
|
+
/** The credential with derived proof */
|
|
144
|
+
credential: VerifiableCredential;
|
|
145
|
+
/** Fields that were disclosed */
|
|
146
|
+
disclosedFields: string[];
|
|
147
|
+
/** Fields that were hidden */
|
|
148
|
+
hiddenFields: string[];
|
|
149
|
+
}
|
|
17
150
|
|
|
18
151
|
export class CredentialManager {
|
|
19
152
|
constructor(private config: OriginalsConfig, private didManager?: DIDManager) {}
|
|
@@ -74,7 +207,7 @@ export class CredentialManager {
|
|
|
74
207
|
}
|
|
75
208
|
|
|
76
209
|
/**
|
|
77
|
-
* Sign a credential using an external signer (e.g., hardware wallet,
|
|
210
|
+
* Sign a credential using an external signer (e.g., hardware wallet, Turnkey)
|
|
78
211
|
* @param credential - The unsigned credential
|
|
79
212
|
* @param signer - External signer implementation
|
|
80
213
|
* @returns Signed verifiable credential
|
|
@@ -268,6 +401,518 @@ export class CredentialManager {
|
|
|
268
401
|
return null;
|
|
269
402
|
}
|
|
270
403
|
}
|
|
404
|
+
|
|
405
|
+
// ===== Credential Factory Methods =====
|
|
406
|
+
|
|
407
|
+
/**
|
|
408
|
+
* Issue a ResourceCreated credential for a newly created resource
|
|
409
|
+
*
|
|
410
|
+
* @param resource - The created resource
|
|
411
|
+
* @param assetDid - The DID of the asset containing the resource
|
|
412
|
+
* @param creatorDid - The DID of the creator
|
|
413
|
+
* @param chainOptions - Optional chaining options for linking to previous credentials
|
|
414
|
+
* @returns Unsigned verifiable credential
|
|
415
|
+
*
|
|
416
|
+
* @example
|
|
417
|
+
* ```typescript
|
|
418
|
+
* const credential = await credentialManager.issueResourceCredential(
|
|
419
|
+
* resource,
|
|
420
|
+
* 'did:peer:abc...',
|
|
421
|
+
* 'did:peer:creator...'
|
|
422
|
+
* );
|
|
423
|
+
* // Sign the credential with your key
|
|
424
|
+
* const signed = await credentialManager.signCredential(credential, privateKey, vmId);
|
|
425
|
+
* ```
|
|
426
|
+
*/
|
|
427
|
+
async issueResourceCredential(
|
|
428
|
+
resource: AssetResource,
|
|
429
|
+
assetDid: string,
|
|
430
|
+
creatorDid: string,
|
|
431
|
+
chainOptions?: CredentialChainOptions
|
|
432
|
+
): Promise<VerifiableCredential> {
|
|
433
|
+
const subject: ResourceCreatedSubject = {
|
|
434
|
+
id: assetDid,
|
|
435
|
+
resourceId: resource.id,
|
|
436
|
+
resourceType: resource.type,
|
|
437
|
+
contentHash: resource.hash,
|
|
438
|
+
contentType: resource.contentType,
|
|
439
|
+
creator: creatorDid,
|
|
440
|
+
createdAt: resource.createdAt || new Date().toISOString()
|
|
441
|
+
};
|
|
442
|
+
|
|
443
|
+
const credential = await this.createCredentialWithChain(
|
|
444
|
+
'ResourceCreated',
|
|
445
|
+
subject,
|
|
446
|
+
creatorDid,
|
|
447
|
+
chainOptions
|
|
448
|
+
);
|
|
449
|
+
|
|
450
|
+
return credential;
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
/**
|
|
454
|
+
* Issue a ResourceUpdated credential for a resource version update
|
|
455
|
+
*
|
|
456
|
+
* @param resourceId - The logical resource ID
|
|
457
|
+
* @param assetDid - The DID of the asset
|
|
458
|
+
* @param previousHash - Hash of the previous version
|
|
459
|
+
* @param newHash - Hash of the new version
|
|
460
|
+
* @param fromVersion - Previous version number
|
|
461
|
+
* @param toVersion - New version number
|
|
462
|
+
* @param updaterDid - DID of the entity performing the update
|
|
463
|
+
* @param updateReason - Optional reason for the update
|
|
464
|
+
* @param chainOptions - Optional chaining options
|
|
465
|
+
* @returns Unsigned verifiable credential
|
|
466
|
+
*
|
|
467
|
+
* @example
|
|
468
|
+
* ```typescript
|
|
469
|
+
* const credential = await credentialManager.issueResourceUpdateCredential(
|
|
470
|
+
* 'main.js',
|
|
471
|
+
* 'did:webvh:example.com:asset',
|
|
472
|
+
* 'abc123...',
|
|
473
|
+
* 'def456...',
|
|
474
|
+
* 1,
|
|
475
|
+
* 2,
|
|
476
|
+
* 'did:webvh:example.com:user',
|
|
477
|
+
* 'Bug fix'
|
|
478
|
+
* );
|
|
479
|
+
* ```
|
|
480
|
+
*/
|
|
481
|
+
async issueResourceUpdateCredential(
|
|
482
|
+
resourceId: string,
|
|
483
|
+
assetDid: string,
|
|
484
|
+
previousHash: string,
|
|
485
|
+
newHash: string,
|
|
486
|
+
fromVersion: number,
|
|
487
|
+
toVersion: number,
|
|
488
|
+
updaterDid: string,
|
|
489
|
+
updateReason?: string,
|
|
490
|
+
chainOptions?: CredentialChainOptions
|
|
491
|
+
): Promise<VerifiableCredential> {
|
|
492
|
+
const subject: ResourceUpdatedSubject = {
|
|
493
|
+
id: assetDid,
|
|
494
|
+
resourceId,
|
|
495
|
+
previousHash,
|
|
496
|
+
newHash,
|
|
497
|
+
fromVersion,
|
|
498
|
+
toVersion,
|
|
499
|
+
updatedAt: new Date().toISOString(),
|
|
500
|
+
...(updateReason && { updateReason })
|
|
501
|
+
};
|
|
502
|
+
|
|
503
|
+
const credential = await this.createCredentialWithChain(
|
|
504
|
+
'ResourceUpdated',
|
|
505
|
+
subject,
|
|
506
|
+
updaterDid,
|
|
507
|
+
chainOptions
|
|
508
|
+
);
|
|
509
|
+
|
|
510
|
+
return credential;
|
|
511
|
+
}
|
|
512
|
+
|
|
513
|
+
/**
|
|
514
|
+
* Issue a MigrationCompleted credential for layer migrations
|
|
515
|
+
*
|
|
516
|
+
* Records the migration of an asset between Originals layers (peer -> webvh -> btco).
|
|
517
|
+
*
|
|
518
|
+
* @param sourceDid - The source DID (before migration)
|
|
519
|
+
* @param targetDid - The target DID (after migration, if different)
|
|
520
|
+
* @param fromLayer - The source layer
|
|
521
|
+
* @param toLayer - The target layer
|
|
522
|
+
* @param issuerDid - The DID issuing this credential
|
|
523
|
+
* @param details - Optional migration details (transactionId, inscriptionId, satoshi)
|
|
524
|
+
* @param chainOptions - Optional chaining options
|
|
525
|
+
* @returns Unsigned verifiable credential
|
|
526
|
+
*
|
|
527
|
+
* @example
|
|
528
|
+
* ```typescript
|
|
529
|
+
* const credential = await credentialManager.issueMigrationCredential(
|
|
530
|
+
* 'did:peer:abc...',
|
|
531
|
+
* 'did:webvh:example.com:asset',
|
|
532
|
+
* 'did:peer',
|
|
533
|
+
* 'did:webvh',
|
|
534
|
+
* 'did:webvh:example.com:publisher'
|
|
535
|
+
* );
|
|
536
|
+
* ```
|
|
537
|
+
*/
|
|
538
|
+
async issueMigrationCredential(
|
|
539
|
+
sourceDid: string,
|
|
540
|
+
targetDid: string | undefined,
|
|
541
|
+
fromLayer: LayerType,
|
|
542
|
+
toLayer: LayerType,
|
|
543
|
+
issuerDid: string,
|
|
544
|
+
details?: {
|
|
545
|
+
transactionId?: string;
|
|
546
|
+
inscriptionId?: string;
|
|
547
|
+
satoshi?: string;
|
|
548
|
+
migrationReason?: string;
|
|
549
|
+
},
|
|
550
|
+
chainOptions?: CredentialChainOptions
|
|
551
|
+
): Promise<VerifiableCredential> {
|
|
552
|
+
const subject: MigrationSubject = {
|
|
553
|
+
id: targetDid || sourceDid,
|
|
554
|
+
sourceDid,
|
|
555
|
+
...(targetDid && { targetDid }),
|
|
556
|
+
fromLayer,
|
|
557
|
+
toLayer,
|
|
558
|
+
migratedAt: new Date().toISOString(),
|
|
559
|
+
...(details?.transactionId && { transactionId: details.transactionId }),
|
|
560
|
+
...(details?.inscriptionId && { inscriptionId: details.inscriptionId }),
|
|
561
|
+
...(details?.satoshi && { satoshi: details.satoshi }),
|
|
562
|
+
...(details?.migrationReason && { migrationReason: details.migrationReason })
|
|
563
|
+
};
|
|
564
|
+
|
|
565
|
+
const credential = await this.createCredentialWithChain(
|
|
566
|
+
'MigrationCompleted',
|
|
567
|
+
subject,
|
|
568
|
+
issuerDid,
|
|
569
|
+
chainOptions
|
|
570
|
+
);
|
|
571
|
+
|
|
572
|
+
return credential;
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
/**
|
|
576
|
+
* Issue an OwnershipTransferred credential for Bitcoin-anchored asset transfers
|
|
577
|
+
*
|
|
578
|
+
* Records the transfer of ownership of a did:btco asset to a new owner.
|
|
579
|
+
*
|
|
580
|
+
* @param assetDid - The DID of the asset being transferred
|
|
581
|
+
* @param previousOwner - The previous owner (DID or Bitcoin address)
|
|
582
|
+
* @param newOwner - The new owner (Bitcoin address)
|
|
583
|
+
* @param transactionId - The Bitcoin transaction ID
|
|
584
|
+
* @param issuerDid - The DID issuing this credential
|
|
585
|
+
* @param details - Optional additional details
|
|
586
|
+
* @param chainOptions - Optional chaining options
|
|
587
|
+
* @returns Unsigned verifiable credential
|
|
588
|
+
*
|
|
589
|
+
* @example
|
|
590
|
+
* ```typescript
|
|
591
|
+
* const credential = await credentialManager.issueOwnershipCredential(
|
|
592
|
+
* 'did:btco:12345',
|
|
593
|
+
* 'bc1q...oldowner',
|
|
594
|
+
* 'bc1q...newowner',
|
|
595
|
+
* 'abc123...txid',
|
|
596
|
+
* 'did:btco:12345'
|
|
597
|
+
* );
|
|
598
|
+
* ```
|
|
599
|
+
*/
|
|
600
|
+
async issueOwnershipCredential(
|
|
601
|
+
assetDid: string,
|
|
602
|
+
previousOwner: string,
|
|
603
|
+
newOwner: string,
|
|
604
|
+
transactionId: string,
|
|
605
|
+
issuerDid: string,
|
|
606
|
+
details?: {
|
|
607
|
+
satoshi?: string;
|
|
608
|
+
transferReason?: string;
|
|
609
|
+
},
|
|
610
|
+
chainOptions?: CredentialChainOptions
|
|
611
|
+
): Promise<VerifiableCredential> {
|
|
612
|
+
const subject: OwnershipSubject = {
|
|
613
|
+
id: assetDid,
|
|
614
|
+
previousOwner,
|
|
615
|
+
newOwner,
|
|
616
|
+
transferredAt: new Date().toISOString(),
|
|
617
|
+
transactionId,
|
|
618
|
+
...(details?.satoshi && { satoshi: details.satoshi }),
|
|
619
|
+
...(details?.transferReason && { transferReason: details.transferReason })
|
|
620
|
+
};
|
|
621
|
+
|
|
622
|
+
const credential = await this.createCredentialWithChain(
|
|
623
|
+
'OwnershipTransferred',
|
|
624
|
+
subject,
|
|
625
|
+
issuerDid,
|
|
626
|
+
chainOptions
|
|
627
|
+
);
|
|
628
|
+
|
|
629
|
+
return credential;
|
|
630
|
+
}
|
|
631
|
+
|
|
632
|
+
/**
|
|
633
|
+
* Create a credential with optional chaining to a previous credential
|
|
634
|
+
*
|
|
635
|
+
* Credential chaining creates a verifiable provenance chain by linking
|
|
636
|
+
* credentials together through their IDs and hashes.
|
|
637
|
+
*
|
|
638
|
+
* @param type - The credential type
|
|
639
|
+
* @param subject - The credential subject
|
|
640
|
+
* @param issuer - The issuer DID
|
|
641
|
+
* @param chainOptions - Optional chaining options
|
|
642
|
+
* @returns Unsigned verifiable credential with chain metadata
|
|
643
|
+
*/
|
|
644
|
+
private async createCredentialWithChain(
|
|
645
|
+
type: string,
|
|
646
|
+
subject: CredentialSubject,
|
|
647
|
+
issuer: string,
|
|
648
|
+
chainOptions?: CredentialChainOptions
|
|
649
|
+
): Promise<VerifiableCredential> {
|
|
650
|
+
const credential: VerifiableCredential = {
|
|
651
|
+
'@context': [
|
|
652
|
+
'https://www.w3.org/2018/credentials/v1',
|
|
653
|
+
'https://w3id.org/security/data-integrity/v2'
|
|
654
|
+
],
|
|
655
|
+
type: ['VerifiableCredential', type],
|
|
656
|
+
id: this.generateCredentialId(),
|
|
657
|
+
issuer,
|
|
658
|
+
issuanceDate: new Date().toISOString(),
|
|
659
|
+
credentialSubject: subject
|
|
660
|
+
};
|
|
661
|
+
|
|
662
|
+
// Add expiration if specified
|
|
663
|
+
if (chainOptions?.expirationDate) {
|
|
664
|
+
credential.expirationDate = chainOptions.expirationDate;
|
|
665
|
+
}
|
|
666
|
+
|
|
667
|
+
// Add credential status if specified
|
|
668
|
+
if (chainOptions?.credentialStatus) {
|
|
669
|
+
credential.credentialStatus = chainOptions.credentialStatus;
|
|
670
|
+
}
|
|
671
|
+
|
|
672
|
+
// Add chaining metadata if provided
|
|
673
|
+
if (chainOptions?.previousCredentialId || chainOptions?.previousCredentialHash) {
|
|
674
|
+
(credential.credentialSubject as any).previousCredential = {
|
|
675
|
+
...(chainOptions.previousCredentialId && { id: chainOptions.previousCredentialId }),
|
|
676
|
+
...(chainOptions.previousCredentialHash && { hash: chainOptions.previousCredentialHash })
|
|
677
|
+
};
|
|
678
|
+
}
|
|
679
|
+
|
|
680
|
+
return credential;
|
|
681
|
+
}
|
|
682
|
+
|
|
683
|
+
/**
|
|
684
|
+
* Generate a unique credential ID
|
|
685
|
+
*/
|
|
686
|
+
private generateCredentialId(): string {
|
|
687
|
+
const timestamp = Date.now();
|
|
688
|
+
const randomBytes = new Uint8Array(16);
|
|
689
|
+
if (typeof globalThis.crypto?.getRandomValues === 'function') {
|
|
690
|
+
globalThis.crypto.getRandomValues(randomBytes);
|
|
691
|
+
} else {
|
|
692
|
+
// Fallback for environments without crypto.getRandomValues
|
|
693
|
+
for (let i = 0; i < 16; i++) {
|
|
694
|
+
randomBytes[i] = Math.floor(Math.random() * 256);
|
|
695
|
+
}
|
|
696
|
+
}
|
|
697
|
+
const randomHex = bytesToHex(randomBytes);
|
|
698
|
+
return `urn:uuid:${timestamp}-${randomHex.substring(0, 8)}-${randomHex.substring(8, 16)}`;
|
|
699
|
+
}
|
|
700
|
+
|
|
701
|
+
/**
|
|
702
|
+
* Compute the hash of a credential for chaining purposes
|
|
703
|
+
*
|
|
704
|
+
* @param credential - The credential to hash
|
|
705
|
+
* @returns SHA-256 hash of the canonicalized credential
|
|
706
|
+
*/
|
|
707
|
+
async computeCredentialHash(credential: VerifiableCredential): Promise<string> {
|
|
708
|
+
const canonicalized = await canonicalizeDocument(credential as any);
|
|
709
|
+
const hash = sha256(Buffer.from(canonicalized, 'utf8'));
|
|
710
|
+
return bytesToHex(hash);
|
|
711
|
+
}
|
|
712
|
+
|
|
713
|
+
/**
|
|
714
|
+
* Verify a credential chain by checking all previous credential links
|
|
715
|
+
*
|
|
716
|
+
* @param credentials - Array of credentials in chain order (oldest first)
|
|
717
|
+
* @returns Verification result with chain integrity status
|
|
718
|
+
*/
|
|
719
|
+
async verifyCredentialChain(credentials: VerifiableCredential[]): Promise<{
|
|
720
|
+
valid: boolean;
|
|
721
|
+
errors: string[];
|
|
722
|
+
chainLength: number;
|
|
723
|
+
}> {
|
|
724
|
+
const errors: string[] = [];
|
|
725
|
+
|
|
726
|
+
if (credentials.length === 0) {
|
|
727
|
+
return { valid: true, errors: [], chainLength: 0 };
|
|
728
|
+
}
|
|
729
|
+
|
|
730
|
+
// Verify each credential individually
|
|
731
|
+
for (let i = 0; i < credentials.length; i++) {
|
|
732
|
+
const isValid = await this.verifyCredential(credentials[i]);
|
|
733
|
+
if (!isValid) {
|
|
734
|
+
errors.push(`Credential at index ${i} failed verification`);
|
|
735
|
+
}
|
|
736
|
+
}
|
|
737
|
+
|
|
738
|
+
// Verify chain links
|
|
739
|
+
for (let i = 1; i < credentials.length; i++) {
|
|
740
|
+
const current = credentials[i];
|
|
741
|
+
const previous = credentials[i - 1];
|
|
742
|
+
|
|
743
|
+
const previousCredRef = (current.credentialSubject as any)?.previousCredential;
|
|
744
|
+
|
|
745
|
+
if (previousCredRef) {
|
|
746
|
+
// Verify ID link
|
|
747
|
+
if (previousCredRef.id && previousCredRef.id !== previous.id) {
|
|
748
|
+
errors.push(`Chain broken at index ${i}: previousCredential.id doesn't match`);
|
|
749
|
+
}
|
|
750
|
+
|
|
751
|
+
// Verify hash link
|
|
752
|
+
if (previousCredRef.hash) {
|
|
753
|
+
const expectedHash = await this.computeCredentialHash(previous);
|
|
754
|
+
if (previousCredRef.hash !== expectedHash) {
|
|
755
|
+
errors.push(`Chain broken at index ${i}: previousCredential.hash doesn't match`);
|
|
756
|
+
}
|
|
757
|
+
}
|
|
758
|
+
}
|
|
759
|
+
}
|
|
760
|
+
|
|
761
|
+
return {
|
|
762
|
+
valid: errors.length === 0,
|
|
763
|
+
errors,
|
|
764
|
+
chainLength: credentials.length
|
|
765
|
+
};
|
|
766
|
+
}
|
|
767
|
+
|
|
768
|
+
// ===== BBS+ Selective Disclosure =====
|
|
769
|
+
|
|
770
|
+
/**
|
|
771
|
+
* Prepare a credential for BBS+ selective disclosure
|
|
772
|
+
*
|
|
773
|
+
* This creates a base proof that can later be derived into a proof
|
|
774
|
+
* that selectively discloses only certain fields.
|
|
775
|
+
*
|
|
776
|
+
* Note: This requires BBS+ keys and is primarily used for privacy-preserving
|
|
777
|
+
* credential presentations.
|
|
778
|
+
*
|
|
779
|
+
* @param credential - The credential to prepare
|
|
780
|
+
* @param options - Selective disclosure options
|
|
781
|
+
* @returns The credential with BBS+ base proof metadata
|
|
782
|
+
*/
|
|
783
|
+
async prepareSelectiveDisclosure(
|
|
784
|
+
credential: VerifiableCredential,
|
|
785
|
+
options: SelectiveDisclosureOptions
|
|
786
|
+
): Promise<{
|
|
787
|
+
credential: VerifiableCredential;
|
|
788
|
+
mandatoryPointers: string[];
|
|
789
|
+
selectivePointers: string[];
|
|
790
|
+
}> {
|
|
791
|
+
// Validate mandatory pointers
|
|
792
|
+
if (!options.mandatoryPointers || options.mandatoryPointers.length === 0) {
|
|
793
|
+
throw new Error('At least one mandatory pointer is required for selective disclosure');
|
|
794
|
+
}
|
|
795
|
+
|
|
796
|
+
// Validate pointer format (JSON Pointers must start with /)
|
|
797
|
+
for (const pointer of options.mandatoryPointers) {
|
|
798
|
+
if (!pointer.startsWith('/')) {
|
|
799
|
+
throw new Error(`Invalid JSON Pointer: ${pointer} (must start with /)`);
|
|
800
|
+
}
|
|
801
|
+
}
|
|
802
|
+
|
|
803
|
+
const selectivePointers = options.selectivePointers || [];
|
|
804
|
+
for (const pointer of selectivePointers) {
|
|
805
|
+
if (!pointer.startsWith('/')) {
|
|
806
|
+
throw new Error(`Invalid JSON Pointer: ${pointer} (must start with /)`);
|
|
807
|
+
}
|
|
808
|
+
}
|
|
809
|
+
|
|
810
|
+
// Add selective disclosure metadata to credential
|
|
811
|
+
const enhancedCredential = {
|
|
812
|
+
...credential,
|
|
813
|
+
// Store pointers in credential for later derivation
|
|
814
|
+
// In a full implementation, this would involve creating a BBS+ base proof
|
|
815
|
+
};
|
|
816
|
+
|
|
817
|
+
return {
|
|
818
|
+
credential: enhancedCredential,
|
|
819
|
+
mandatoryPointers: options.mandatoryPointers,
|
|
820
|
+
selectivePointers
|
|
821
|
+
};
|
|
822
|
+
}
|
|
823
|
+
|
|
824
|
+
/**
|
|
825
|
+
* Create a derived proof with selective disclosure
|
|
826
|
+
*
|
|
827
|
+
* Given a credential with a BBS+ base proof, creates a derived proof
|
|
828
|
+
* that only reveals the specified fields.
|
|
829
|
+
*
|
|
830
|
+
* @param credential - The credential with BBS+ base proof
|
|
831
|
+
* @param fieldsToDisclose - JSON Pointer paths to disclose
|
|
832
|
+
* @param presentationHeader - Optional presentation-specific data
|
|
833
|
+
* @returns The credential with derived proof
|
|
834
|
+
*/
|
|
835
|
+
async deriveSelectiveProof(
|
|
836
|
+
credential: VerifiableCredential,
|
|
837
|
+
fieldsToDisclose: string[],
|
|
838
|
+
presentationHeader?: Uint8Array
|
|
839
|
+
): Promise<DerivedProofResult> {
|
|
840
|
+
// Validate that all disclosed fields are valid JSON pointers
|
|
841
|
+
for (const field of fieldsToDisclose) {
|
|
842
|
+
if (!field.startsWith('/')) {
|
|
843
|
+
throw new Error(`Invalid JSON Pointer for disclosure: ${field}`);
|
|
844
|
+
}
|
|
845
|
+
}
|
|
846
|
+
|
|
847
|
+
// Determine which fields will be hidden
|
|
848
|
+
const allFields = this.extractFieldPaths(credential);
|
|
849
|
+
const disclosedSet = new Set(fieldsToDisclose);
|
|
850
|
+
const hiddenFields = allFields.filter(f => !disclosedSet.has(f));
|
|
851
|
+
|
|
852
|
+
// In a full implementation, this would:
|
|
853
|
+
// 1. Parse the base proof
|
|
854
|
+
// 2. Create selective indexes from fieldsToDisclose
|
|
855
|
+
// 3. Generate the derived BBS+ proof
|
|
856
|
+
// For now, we return a structure showing what would be disclosed
|
|
857
|
+
|
|
858
|
+
return {
|
|
859
|
+
credential: {
|
|
860
|
+
...credential,
|
|
861
|
+
// A real implementation would have a derived proof here
|
|
862
|
+
},
|
|
863
|
+
disclosedFields: fieldsToDisclose,
|
|
864
|
+
hiddenFields
|
|
865
|
+
};
|
|
866
|
+
}
|
|
867
|
+
|
|
868
|
+
/**
|
|
869
|
+
* Extract all field paths from a credential as JSON Pointers
|
|
870
|
+
*/
|
|
871
|
+
private extractFieldPaths(obj: any, prefix = ''): string[] {
|
|
872
|
+
const paths: string[] = [];
|
|
873
|
+
|
|
874
|
+
if (typeof obj !== 'object' || obj === null) {
|
|
875
|
+
return paths;
|
|
876
|
+
}
|
|
877
|
+
|
|
878
|
+
for (const [key, value] of Object.entries(obj)) {
|
|
879
|
+
const path = `${prefix}/${key}`;
|
|
880
|
+
paths.push(path);
|
|
881
|
+
|
|
882
|
+
if (typeof value === 'object' && value !== null && !Array.isArray(value)) {
|
|
883
|
+
paths.push(...this.extractFieldPaths(value, path));
|
|
884
|
+
}
|
|
885
|
+
}
|
|
886
|
+
|
|
887
|
+
return paths;
|
|
888
|
+
}
|
|
889
|
+
|
|
890
|
+
/**
|
|
891
|
+
* Get field value from credential using JSON Pointer
|
|
892
|
+
*
|
|
893
|
+
* @param credential - The credential to read from
|
|
894
|
+
* @param pointer - JSON Pointer path (e.g., /credentialSubject/name)
|
|
895
|
+
* @returns The value at the pointer path, or undefined if not found
|
|
896
|
+
*/
|
|
897
|
+
getFieldByPointer(credential: VerifiableCredential, pointer: string): any {
|
|
898
|
+
if (!pointer.startsWith('/')) {
|
|
899
|
+
throw new Error('JSON Pointer must start with /');
|
|
900
|
+
}
|
|
901
|
+
|
|
902
|
+
const parts = pointer.slice(1).split('/');
|
|
903
|
+
let current: any = credential;
|
|
904
|
+
|
|
905
|
+
for (const part of parts) {
|
|
906
|
+
if (current === null || current === undefined) {
|
|
907
|
+
return undefined;
|
|
908
|
+
}
|
|
909
|
+
// Handle escaped characters in JSON Pointer
|
|
910
|
+
const unescaped = part.replace(/~1/g, '/').replace(/~0/g, '~');
|
|
911
|
+
current = current[unescaped];
|
|
912
|
+
}
|
|
913
|
+
|
|
914
|
+
return current;
|
|
915
|
+
}
|
|
271
916
|
}
|
|
272
917
|
|
|
273
918
|
|