@originals/sdk 1.4.2 → 1.4.3
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 +4 -1
- package/.eslintrc.json +0 -33
- package/src/adapters/FeeOracleMock.ts +0 -9
- package/src/adapters/index.ts +0 -5
- package/src/adapters/providers/OrdHttpProvider.ts +0 -126
- package/src/adapters/providers/OrdMockProvider.ts +0 -101
- package/src/adapters/types.ts +0 -66
- package/src/bitcoin/BitcoinManager.ts +0 -330
- package/src/bitcoin/BroadcastClient.ts +0 -54
- package/src/bitcoin/OrdinalsClient.ts +0 -119
- package/src/bitcoin/PSBTBuilder.ts +0 -106
- package/src/bitcoin/fee-calculation.ts +0 -38
- package/src/bitcoin/providers/OrdNodeProvider.ts +0 -92
- package/src/bitcoin/providers/OrdinalsProvider.ts +0 -56
- package/src/bitcoin/providers/types.ts +0 -59
- package/src/bitcoin/transactions/commit.ts +0 -465
- package/src/bitcoin/transactions/index.ts +0 -13
- package/src/bitcoin/transfer.ts +0 -43
- package/src/bitcoin/utxo-selection.ts +0 -322
- package/src/bitcoin/utxo.ts +0 -113
- package/src/contexts/credentials-v1.json +0 -237
- package/src/contexts/credentials-v2-examples.json +0 -5
- package/src/contexts/credentials-v2.json +0 -340
- package/src/contexts/credentials.json +0 -237
- package/src/contexts/data-integrity-v2.json +0 -81
- package/src/contexts/dids.json +0 -58
- package/src/contexts/ed255192020.json +0 -93
- package/src/contexts/ordinals-plus.json +0 -23
- package/src/contexts/originals.json +0 -22
- package/src/core/OriginalsSDK.ts +0 -416
- package/src/crypto/Multikey.ts +0 -194
- package/src/crypto/Signer.ts +0 -254
- package/src/crypto/noble-init.ts +0 -121
- package/src/did/BtcoDidResolver.ts +0 -227
- package/src/did/DIDManager.ts +0 -694
- package/src/did/Ed25519Verifier.ts +0 -68
- package/src/did/KeyManager.ts +0 -236
- package/src/did/WebVHManager.ts +0 -498
- package/src/did/createBtcoDidDocument.ts +0 -59
- package/src/did/providers/OrdinalsClientProviderAdapter.ts +0 -68
- package/src/events/EventEmitter.ts +0 -222
- package/src/events/index.ts +0 -19
- package/src/events/types.ts +0 -331
- package/src/examples/basic-usage.ts +0 -78
- package/src/examples/create-module-original.ts +0 -435
- package/src/examples/full-lifecycle-flow.ts +0 -514
- package/src/examples/run.ts +0 -60
- package/src/index.ts +0 -150
- package/src/kinds/KindRegistry.ts +0 -290
- package/src/kinds/index.ts +0 -74
- package/src/kinds/types.ts +0 -470
- package/src/kinds/validators/AgentValidator.ts +0 -257
- package/src/kinds/validators/AppValidator.ts +0 -211
- package/src/kinds/validators/DatasetValidator.ts +0 -242
- package/src/kinds/validators/DocumentValidator.ts +0 -311
- package/src/kinds/validators/MediaValidator.ts +0 -269
- package/src/kinds/validators/ModuleValidator.ts +0 -225
- package/src/kinds/validators/base.ts +0 -276
- package/src/kinds/validators/index.ts +0 -12
- package/src/lifecycle/BatchOperations.ts +0 -373
- package/src/lifecycle/LifecycleManager.ts +0 -2126
- package/src/lifecycle/OriginalsAsset.ts +0 -524
- package/src/lifecycle/ProvenanceQuery.ts +0 -280
- package/src/lifecycle/ResourceVersioning.ts +0 -163
- package/src/migration/MigrationManager.ts +0 -527
- package/src/migration/audit/AuditLogger.ts +0 -176
- package/src/migration/checkpoint/CheckpointManager.ts +0 -112
- package/src/migration/checkpoint/CheckpointStorage.ts +0 -101
- package/src/migration/index.ts +0 -33
- package/src/migration/operations/BaseMigration.ts +0 -126
- package/src/migration/operations/PeerToBtcoMigration.ts +0 -105
- package/src/migration/operations/PeerToWebvhMigration.ts +0 -62
- package/src/migration/operations/WebvhToBtcoMigration.ts +0 -105
- package/src/migration/rollback/RollbackManager.ts +0 -170
- package/src/migration/state/StateMachine.ts +0 -92
- package/src/migration/state/StateTracker.ts +0 -156
- package/src/migration/types.ts +0 -344
- package/src/migration/validation/BitcoinValidator.ts +0 -107
- package/src/migration/validation/CredentialValidator.ts +0 -62
- package/src/migration/validation/DIDCompatibilityValidator.ts +0 -151
- package/src/migration/validation/LifecycleValidator.ts +0 -64
- package/src/migration/validation/StorageValidator.ts +0 -79
- package/src/migration/validation/ValidationPipeline.ts +0 -213
- package/src/resources/ResourceManager.ts +0 -655
- package/src/resources/index.ts +0 -21
- package/src/resources/types.ts +0 -202
- package/src/storage/LocalStorageAdapter.ts +0 -61
- package/src/storage/MemoryStorageAdapter.ts +0 -29
- package/src/storage/StorageAdapter.ts +0 -25
- package/src/storage/index.ts +0 -3
- package/src/types/bitcoin.ts +0 -98
- package/src/types/common.ts +0 -92
- package/src/types/credentials.ts +0 -88
- package/src/types/did.ts +0 -31
- package/src/types/external-shims.d.ts +0 -53
- package/src/types/index.ts +0 -7
- package/src/types/network.ts +0 -175
- package/src/utils/EventLogger.ts +0 -298
- package/src/utils/Logger.ts +0 -322
- package/src/utils/MetricsCollector.ts +0 -358
- package/src/utils/bitcoin-address.ts +0 -130
- package/src/utils/cbor.ts +0 -12
- package/src/utils/encoding.ts +0 -127
- package/src/utils/hash.ts +0 -6
- package/src/utils/retry.ts +0 -46
- package/src/utils/satoshi-validation.ts +0 -196
- package/src/utils/serialization.ts +0 -96
- package/src/utils/telemetry.ts +0 -40
- package/src/utils/validation.ts +0 -119
- package/src/vc/CredentialManager.ts +0 -918
- package/src/vc/Issuer.ts +0 -100
- package/src/vc/Verifier.ts +0 -47
- package/src/vc/cryptosuites/bbs.ts +0 -253
- package/src/vc/cryptosuites/bbsSimple.ts +0 -21
- package/src/vc/cryptosuites/eddsa.ts +0 -99
- package/src/vc/documentLoader.ts +0 -67
- package/src/vc/proofs/data-integrity.ts +0 -33
- package/src/vc/utils/jsonld.ts +0 -18
- package/tests/__mocks__/bbs-signatures.js +0 -17
- package/tests/__mocks__/mf-base58.js +0 -24
- package/tests/fixtures/did-documents.ts +0 -247
- package/tests/index.test.ts +0 -21
- package/tests/integration/BatchOperations.test.ts +0 -531
- package/tests/integration/CompleteLifecycle.e2e.test.ts +0 -735
- package/tests/integration/CredentialManager.test.ts +0 -42
- package/tests/integration/DIDManager.test.ts +0 -41
- package/tests/integration/DidPeerToWebVhFlow.test.ts +0 -351
- package/tests/integration/Events.test.ts +0 -435
- package/tests/integration/Lifecycle.transfer.btco.integration.test.ts +0 -25
- package/tests/integration/LifecycleManager.test.ts +0 -21
- package/tests/integration/MultikeyFlow.test.ts +0 -52
- package/tests/integration/TelemetryIntegration.test.ts +0 -395
- package/tests/integration/WebVhPublish.test.ts +0 -48
- package/tests/integration/createTypedOriginal.test.ts +0 -379
- package/tests/integration/migration/peer-to-webvh.test.ts +0 -172
- package/tests/manual/test-commit-creation.ts +0 -323
- package/tests/mocks/MockKeyStore.ts +0 -38
- package/tests/mocks/adapters/MemoryStorageAdapter.ts +0 -24
- package/tests/mocks/adapters/MockFeeOracle.ts +0 -11
- package/tests/mocks/adapters/MockOrdinalsProvider.ts +0 -76
- package/tests/mocks/adapters/OrdMockProvider.test.ts +0 -176
- package/tests/mocks/adapters/index.ts +0 -6
- package/tests/performance/BatchOperations.perf.test.ts +0 -403
- package/tests/performance/logging.perf.test.ts +0 -336
- package/tests/sdk.test.ts +0 -43
- package/tests/security/bitcoin-penetration-tests.test.ts +0 -622
- package/tests/setup.bun.ts +0 -69
- package/tests/setup.jest.ts +0 -23
- package/tests/stress/batch-operations-stress.test.ts +0 -571
- package/tests/unit/adapters/FeeOracleMock.test.ts +0 -40
- package/tests/unit/bitcoin/BitcoinManager.test.ts +0 -293
- package/tests/unit/bitcoin/BroadcastClient.test.ts +0 -52
- package/tests/unit/bitcoin/OrdNodeProvider.test.ts +0 -53
- package/tests/unit/bitcoin/OrdinalsClient.test.ts +0 -381
- package/tests/unit/bitcoin/OrdinalsClientProvider.test.ts +0 -102
- package/tests/unit/bitcoin/PSBTBuilder.test.ts +0 -84
- package/tests/unit/bitcoin/fee-calculation.test.ts +0 -261
- package/tests/unit/bitcoin/transactions/commit.test.ts +0 -649
- package/tests/unit/bitcoin/transfer.test.ts +0 -31
- package/tests/unit/bitcoin/utxo-selection-new.test.ts +0 -502
- package/tests/unit/bitcoin/utxo.more.test.ts +0 -39
- package/tests/unit/bitcoin/utxo.selection.test.ts +0 -38
- package/tests/unit/core/OriginalsSDK.test.ts +0 -152
- package/tests/unit/crypto/Multikey.test.ts +0 -206
- package/tests/unit/crypto/Signer.test.ts +0 -408
- package/tests/unit/did/BtcoDidResolver.test.ts +0 -611
- package/tests/unit/did/DIDManager.more.test.ts +0 -43
- package/tests/unit/did/DIDManager.test.ts +0 -185
- package/tests/unit/did/Ed25519Verifier.test.ts +0 -160
- package/tests/unit/did/KeyManager.test.ts +0 -452
- package/tests/unit/did/OrdinalsClientProviderAdapter.test.ts +0 -45
- package/tests/unit/did/WebVHManager.test.ts +0 -435
- package/tests/unit/did/createBtcoDidDocument.test.ts +0 -67
- package/tests/unit/did/providers/OrdinalsClientProviderAdapter.test.ts +0 -159
- package/tests/unit/events/EventEmitter.test.ts +0 -407
- package/tests/unit/kinds/KindRegistry.test.ts +0 -329
- package/tests/unit/kinds/types.test.ts +0 -409
- package/tests/unit/kinds/validators.test.ts +0 -651
- package/tests/unit/lifecycle/BatchOperations.test.ts +0 -527
- package/tests/unit/lifecycle/LifecycleManager.cleanapi.test.ts +0 -441
- package/tests/unit/lifecycle/LifecycleManager.keymanagement.test.ts +0 -312
- package/tests/unit/lifecycle/LifecycleManager.prov.test.ts +0 -18
- package/tests/unit/lifecycle/LifecycleManager.test.ts +0 -213
- package/tests/unit/lifecycle/LifecycleManager.transfer.unit.test.ts +0 -30
- package/tests/unit/lifecycle/OriginalsAsset.test.ts +0 -176
- package/tests/unit/lifecycle/ProvenanceQuery.test.ts +0 -577
- package/tests/unit/lifecycle/ResourceVersioning.test.ts +0 -651
- package/tests/unit/resources/ResourceManager.test.ts +0 -740
- package/tests/unit/storage/MemoryStorageAdapter.test.ts +0 -93
- package/tests/unit/types/network.test.ts +0 -255
- package/tests/unit/utils/EventIntegration.test.ts +0 -384
- package/tests/unit/utils/Logger.test.ts +0 -473
- package/tests/unit/utils/MetricsCollector.test.ts +0 -358
- package/tests/unit/utils/bitcoin-address.test.ts +0 -250
- package/tests/unit/utils/cbor.test.ts +0 -35
- package/tests/unit/utils/encoding.test.ts +0 -318
- package/tests/unit/utils/hash.test.ts +0 -12
- package/tests/unit/utils/retry.test.ts +0 -100
- package/tests/unit/utils/satoshi-validation.test.ts +0 -354
- package/tests/unit/utils/serialization.test.ts +0 -124
- package/tests/unit/utils/telemetry.test.ts +0 -52
- package/tests/unit/utils/validation.test.ts +0 -141
- package/tests/unit/vc/CredentialManager.helpers.test.ts +0 -527
- package/tests/unit/vc/CredentialManager.test.ts +0 -487
- package/tests/unit/vc/Issuer.test.ts +0 -107
- package/tests/unit/vc/Verifier.test.ts +0 -525
- package/tests/unit/vc/bbs.test.ts +0 -282
- package/tests/unit/vc/cryptosuites/eddsa.test.ts +0 -398
- package/tests/unit/vc/documentLoader.test.ts +0 -121
- package/tests/unit/vc/proofs/data-integrity.test.ts +0 -24
- package/tsconfig.json +0 -31
- package/tsconfig.test.json +0 -15
|
@@ -1,651 +0,0 @@
|
|
|
1
|
-
import { describe, test, expect, beforeEach, spyOn } from 'bun:test';
|
|
2
|
-
import { OriginalsAsset } from '../../../src/lifecycle/OriginalsAsset';
|
|
3
|
-
import { ResourceVersionManager } from '../../../src/lifecycle/ResourceVersioning';
|
|
4
|
-
import { AssetResource, DIDDocument, VerifiableCredential, OriginalsConfig } from '../../../src/types';
|
|
5
|
-
import { hashResource } from '../../../src/utils/validation';
|
|
6
|
-
import { CredentialManager } from '../../../src/vc/CredentialManager';
|
|
7
|
-
import { DIDManager } from '../../../src/did/DIDManager';
|
|
8
|
-
|
|
9
|
-
function buildDid(id: string): DIDDocument {
|
|
10
|
-
return {
|
|
11
|
-
'@context': ['https://www.w3.org/ns/did/v1'],
|
|
12
|
-
id,
|
|
13
|
-
verificationMethod: [{
|
|
14
|
-
id: `${id}#key-0`,
|
|
15
|
-
type: 'Multikey',
|
|
16
|
-
controller: id,
|
|
17
|
-
publicKeyMultibase: 'z6MkhaXgBZDvotDkL5257faiztiGiC2QtKLGpbnnEGta2doK'
|
|
18
|
-
}]
|
|
19
|
-
};
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
const emptyCreds: VerifiableCredential[] = [];
|
|
23
|
-
|
|
24
|
-
describe('ResourceVersionManager', () => {
|
|
25
|
-
let versionManager: ResourceVersionManager;
|
|
26
|
-
|
|
27
|
-
beforeEach(() => {
|
|
28
|
-
versionManager = new ResourceVersionManager();
|
|
29
|
-
});
|
|
30
|
-
|
|
31
|
-
test('adds first version of a resource', () => {
|
|
32
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
33
|
-
|
|
34
|
-
const history = versionManager.getHistory('res1');
|
|
35
|
-
expect(history).not.toBeNull();
|
|
36
|
-
expect(history!.versions.length).toBe(1);
|
|
37
|
-
expect(history!.currentVersion.version).toBe(1);
|
|
38
|
-
expect(history!.currentVersion.hash).toBe('hash1');
|
|
39
|
-
expect(history!.currentVersion.previousVersionHash).toBeUndefined();
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
test('adds second version with previous hash link', () => {
|
|
43
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
44
|
-
versionManager.addVersion('res1', 'hash2', 'text/plain', 'hash1', 'Updated content');
|
|
45
|
-
|
|
46
|
-
const history = versionManager.getHistory('res1');
|
|
47
|
-
expect(history!.versions.length).toBe(2);
|
|
48
|
-
expect(history!.currentVersion.version).toBe(2);
|
|
49
|
-
expect(history!.currentVersion.hash).toBe('hash2');
|
|
50
|
-
expect(history!.currentVersion.previousVersionHash).toBe('hash1');
|
|
51
|
-
expect(history!.currentVersion.changes).toBe('Updated content');
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
test('getVersion retrieves specific version', () => {
|
|
55
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
56
|
-
versionManager.addVersion('res1', 'hash2', 'text/plain', 'hash1');
|
|
57
|
-
|
|
58
|
-
const v1 = versionManager.getVersion('res1', 1);
|
|
59
|
-
const v2 = versionManager.getVersion('res1', 2);
|
|
60
|
-
|
|
61
|
-
expect(v1!.hash).toBe('hash1');
|
|
62
|
-
expect(v2!.hash).toBe('hash2');
|
|
63
|
-
});
|
|
64
|
-
|
|
65
|
-
test('getVersion returns null for invalid version', () => {
|
|
66
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
67
|
-
|
|
68
|
-
expect(versionManager.getVersion('res1', 0)).toBeNull();
|
|
69
|
-
expect(versionManager.getVersion('res1', 2)).toBeNull();
|
|
70
|
-
expect(versionManager.getVersion('nonexistent', 1)).toBeNull();
|
|
71
|
-
});
|
|
72
|
-
|
|
73
|
-
test('getCurrentVersion returns latest version', () => {
|
|
74
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
75
|
-
versionManager.addVersion('res1', 'hash2', 'text/plain', 'hash1');
|
|
76
|
-
versionManager.addVersion('res1', 'hash3', 'text/plain', 'hash2');
|
|
77
|
-
|
|
78
|
-
const current = versionManager.getCurrentVersion('res1');
|
|
79
|
-
expect(current!.version).toBe(3);
|
|
80
|
-
expect(current!.hash).toBe('hash3');
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
test('verifyChain validates correct version chain', () => {
|
|
84
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
85
|
-
versionManager.addVersion('res1', 'hash2', 'text/plain', 'hash1');
|
|
86
|
-
versionManager.addVersion('res1', 'hash3', 'text/plain', 'hash2');
|
|
87
|
-
|
|
88
|
-
expect(versionManager.verifyChain('res1')).toBe(true);
|
|
89
|
-
});
|
|
90
|
-
|
|
91
|
-
test('verifyChain fails for broken chain', () => {
|
|
92
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
93
|
-
versionManager.addVersion('res1', 'hash2', 'text/plain', 'wronghash');
|
|
94
|
-
|
|
95
|
-
expect(versionManager.verifyChain('res1')).toBe(false);
|
|
96
|
-
});
|
|
97
|
-
|
|
98
|
-
test('verifyChain fails if first version has previousVersionHash', () => {
|
|
99
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain', 'somehash');
|
|
100
|
-
|
|
101
|
-
expect(versionManager.verifyChain('res1')).toBe(false);
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
test('toJSON serializes version data', () => {
|
|
105
|
-
versionManager.addVersion('res1', 'hash1', 'text/plain');
|
|
106
|
-
versionManager.addVersion('res2', 'hash2', 'image/png');
|
|
107
|
-
|
|
108
|
-
const json = versionManager.toJSON();
|
|
109
|
-
expect(json).toHaveProperty('res1');
|
|
110
|
-
expect(json).toHaveProperty('res2');
|
|
111
|
-
});
|
|
112
|
-
});
|
|
113
|
-
|
|
114
|
-
describe('OriginalsAsset - Resource Versioning', () => {
|
|
115
|
-
test('creates asset with initial resource version 1', () => {
|
|
116
|
-
const resources: AssetResource[] = [
|
|
117
|
-
{
|
|
118
|
-
id: 'res1',
|
|
119
|
-
type: 'text',
|
|
120
|
-
content: 'hello',
|
|
121
|
-
contentType: 'text/plain',
|
|
122
|
-
hash: hashResource(Buffer.from('hello', 'utf-8')),
|
|
123
|
-
version: 1,
|
|
124
|
-
createdAt: new Date().toISOString()
|
|
125
|
-
}
|
|
126
|
-
];
|
|
127
|
-
|
|
128
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
129
|
-
|
|
130
|
-
expect(asset.resources.length).toBe(1);
|
|
131
|
-
expect(asset.resources[0].version).toBe(1);
|
|
132
|
-
expect(asset.resources[0].previousVersionHash).toBeUndefined();
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
test('addResourceVersion creates new version and preserves old', () => {
|
|
136
|
-
const resources: AssetResource[] = [
|
|
137
|
-
{
|
|
138
|
-
id: 'res1',
|
|
139
|
-
type: 'text',
|
|
140
|
-
content: 'hello',
|
|
141
|
-
contentType: 'text/plain',
|
|
142
|
-
hash: hashResource(Buffer.from('hello', 'utf-8'))
|
|
143
|
-
}
|
|
144
|
-
];
|
|
145
|
-
|
|
146
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
147
|
-
const newResource = asset.addResourceVersion('res1', 'hello world', 'text/plain', 'Added world');
|
|
148
|
-
|
|
149
|
-
expect(asset.resources.length).toBe(2);
|
|
150
|
-
expect(newResource.version).toBe(2);
|
|
151
|
-
expect(newResource.previousVersionHash).toBe(resources[0].hash);
|
|
152
|
-
expect(newResource.hash).not.toBe(resources[0].hash);
|
|
153
|
-
|
|
154
|
-
// Old version should still be accessible
|
|
155
|
-
const v1 = asset.getResourceVersion('res1', 1);
|
|
156
|
-
expect(v1).not.toBeNull();
|
|
157
|
-
expect(v1!.content).toBe('hello');
|
|
158
|
-
});
|
|
159
|
-
|
|
160
|
-
test('addResourceVersion throws error if content unchanged', () => {
|
|
161
|
-
const resources: AssetResource[] = [
|
|
162
|
-
{
|
|
163
|
-
id: 'res1',
|
|
164
|
-
type: 'text',
|
|
165
|
-
content: 'hello',
|
|
166
|
-
contentType: 'text/plain',
|
|
167
|
-
hash: hashResource(Buffer.from('hello', 'utf-8'))
|
|
168
|
-
}
|
|
169
|
-
];
|
|
170
|
-
|
|
171
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
172
|
-
|
|
173
|
-
expect(() => {
|
|
174
|
-
asset.addResourceVersion('res1', 'hello', 'text/plain');
|
|
175
|
-
}).toThrow('Content unchanged');
|
|
176
|
-
});
|
|
177
|
-
|
|
178
|
-
test('addResourceVersion throws error if resource not found', () => {
|
|
179
|
-
const resources: AssetResource[] = [
|
|
180
|
-
{
|
|
181
|
-
id: 'res1',
|
|
182
|
-
type: 'text',
|
|
183
|
-
content: 'hello',
|
|
184
|
-
contentType: 'text/plain',
|
|
185
|
-
hash: hashResource(Buffer.from('hello', 'utf-8'))
|
|
186
|
-
}
|
|
187
|
-
];
|
|
188
|
-
|
|
189
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
190
|
-
|
|
191
|
-
expect(() => {
|
|
192
|
-
asset.addResourceVersion('nonexistent', 'content', 'text/plain');
|
|
193
|
-
}).toThrow('Resource with id nonexistent not found');
|
|
194
|
-
});
|
|
195
|
-
|
|
196
|
-
test('getAllVersions returns all versions sorted', () => {
|
|
197
|
-
const resources: AssetResource[] = [
|
|
198
|
-
{
|
|
199
|
-
id: 'res1',
|
|
200
|
-
type: 'text',
|
|
201
|
-
content: 'v1',
|
|
202
|
-
contentType: 'text/plain',
|
|
203
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
204
|
-
}
|
|
205
|
-
];
|
|
206
|
-
|
|
207
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
208
|
-
asset.addResourceVersion('res1', 'v2', 'text/plain');
|
|
209
|
-
asset.addResourceVersion('res1', 'v3', 'text/plain');
|
|
210
|
-
|
|
211
|
-
const versions = asset.getAllVersions('res1');
|
|
212
|
-
expect(versions.length).toBe(3);
|
|
213
|
-
expect(versions[0].version || 1).toBe(1);
|
|
214
|
-
expect(versions[1].version).toBe(2);
|
|
215
|
-
expect(versions[2].version).toBe(3);
|
|
216
|
-
});
|
|
217
|
-
|
|
218
|
-
test('getResourceHistory returns complete history', () => {
|
|
219
|
-
const resources: AssetResource[] = [
|
|
220
|
-
{
|
|
221
|
-
id: 'res1',
|
|
222
|
-
type: 'text',
|
|
223
|
-
content: 'v1',
|
|
224
|
-
contentType: 'text/plain',
|
|
225
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
226
|
-
}
|
|
227
|
-
];
|
|
228
|
-
|
|
229
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
230
|
-
asset.addResourceVersion('res1', 'v2', 'text/plain');
|
|
231
|
-
|
|
232
|
-
const history = asset.getResourceHistory('res1');
|
|
233
|
-
expect(history).not.toBeNull();
|
|
234
|
-
expect(history!.resourceId).toBe('res1');
|
|
235
|
-
expect(history!.versions.length).toBe(2);
|
|
236
|
-
expect(history!.currentVersion.version).toBe(2);
|
|
237
|
-
});
|
|
238
|
-
|
|
239
|
-
test('version chain integrity is verifiable', () => {
|
|
240
|
-
const resources: AssetResource[] = [
|
|
241
|
-
{
|
|
242
|
-
id: 'res1',
|
|
243
|
-
type: 'text',
|
|
244
|
-
content: 'v1',
|
|
245
|
-
contentType: 'text/plain',
|
|
246
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
247
|
-
}
|
|
248
|
-
];
|
|
249
|
-
|
|
250
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
251
|
-
asset.addResourceVersion('res1', 'v2', 'text/plain');
|
|
252
|
-
asset.addResourceVersion('res1', 'v3', 'text/plain');
|
|
253
|
-
|
|
254
|
-
// Access internal version manager for testing
|
|
255
|
-
const history = asset.getResourceHistory('res1');
|
|
256
|
-
expect(history).not.toBeNull();
|
|
257
|
-
|
|
258
|
-
// Verify chain manually
|
|
259
|
-
const versions = history!.versions;
|
|
260
|
-
expect(versions[0].previousVersionHash).toBeUndefined();
|
|
261
|
-
expect(versions[1].previousVersionHash).toBe(versions[0].hash);
|
|
262
|
-
expect(versions[2].previousVersionHash).toBe(versions[1].hash);
|
|
263
|
-
});
|
|
264
|
-
|
|
265
|
-
test('emits resource:version:created event', async () => {
|
|
266
|
-
const resources: AssetResource[] = [
|
|
267
|
-
{
|
|
268
|
-
id: 'res1',
|
|
269
|
-
type: 'text',
|
|
270
|
-
content: 'v1',
|
|
271
|
-
contentType: 'text/plain',
|
|
272
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
273
|
-
}
|
|
274
|
-
];
|
|
275
|
-
|
|
276
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
277
|
-
|
|
278
|
-
let eventEmitted = false;
|
|
279
|
-
let capturedEvent: any = null;
|
|
280
|
-
|
|
281
|
-
asset.on('resource:version:created', (event) => {
|
|
282
|
-
eventEmitted = true;
|
|
283
|
-
capturedEvent = event;
|
|
284
|
-
});
|
|
285
|
-
|
|
286
|
-
asset.addResourceVersion('res1', 'v2', 'text/plain', 'Test changes');
|
|
287
|
-
|
|
288
|
-
// Wait for microtask to complete
|
|
289
|
-
await new Promise(resolve => setImmediate(resolve));
|
|
290
|
-
|
|
291
|
-
expect(eventEmitted).toBe(true);
|
|
292
|
-
expect(capturedEvent.type).toBe('resource:version:created');
|
|
293
|
-
expect(capturedEvent.asset.id).toBe(asset.id);
|
|
294
|
-
expect(capturedEvent.resource.id).toBe('res1');
|
|
295
|
-
expect(capturedEvent.resource.fromVersion).toBe(1);
|
|
296
|
-
expect(capturedEvent.resource.toVersion).toBe(2);
|
|
297
|
-
expect(capturedEvent.changes).toBe('Test changes');
|
|
298
|
-
});
|
|
299
|
-
|
|
300
|
-
test('provenance tracks resource updates', () => {
|
|
301
|
-
const resources: AssetResource[] = [
|
|
302
|
-
{
|
|
303
|
-
id: 'res1',
|
|
304
|
-
type: 'text',
|
|
305
|
-
content: 'v1',
|
|
306
|
-
contentType: 'text/plain',
|
|
307
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
308
|
-
}
|
|
309
|
-
];
|
|
310
|
-
|
|
311
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
312
|
-
const v1Hash = resources[0].hash;
|
|
313
|
-
|
|
314
|
-
asset.addResourceVersion('res1', 'v2', 'text/plain', 'First update');
|
|
315
|
-
asset.addResourceVersion('res1', 'v3', 'text/plain', 'Second update');
|
|
316
|
-
|
|
317
|
-
const provenance = asset.getProvenance();
|
|
318
|
-
expect(provenance.resourceUpdates.length).toBe(2);
|
|
319
|
-
|
|
320
|
-
const update1 = provenance.resourceUpdates[0];
|
|
321
|
-
expect(update1.resourceId).toBe('res1');
|
|
322
|
-
expect(update1.fromVersion).toBe(1);
|
|
323
|
-
expect(update1.toVersion).toBe(2);
|
|
324
|
-
expect(update1.fromHash).toBe(v1Hash);
|
|
325
|
-
expect(update1.changes).toBe('First update');
|
|
326
|
-
|
|
327
|
-
const update2 = provenance.resourceUpdates[1];
|
|
328
|
-
expect(update2.fromVersion).toBe(2);
|
|
329
|
-
expect(update2.toVersion).toBe(3);
|
|
330
|
-
expect(update2.changes).toBe('Second update');
|
|
331
|
-
});
|
|
332
|
-
|
|
333
|
-
test('hash-based content addressing validated', () => {
|
|
334
|
-
const content1 = 'content 1';
|
|
335
|
-
const content2 = 'content 2';
|
|
336
|
-
const hash1 = hashResource(Buffer.from(content1, 'utf-8'));
|
|
337
|
-
const hash2 = hashResource(Buffer.from(content2, 'utf-8'));
|
|
338
|
-
|
|
339
|
-
expect(hash1).not.toBe(hash2);
|
|
340
|
-
|
|
341
|
-
const resources: AssetResource[] = [
|
|
342
|
-
{
|
|
343
|
-
id: 'res1',
|
|
344
|
-
type: 'text',
|
|
345
|
-
content: content1,
|
|
346
|
-
contentType: 'text/plain',
|
|
347
|
-
hash: hash1
|
|
348
|
-
}
|
|
349
|
-
];
|
|
350
|
-
|
|
351
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
352
|
-
const newResource = asset.addResourceVersion('res1', content2, 'text/plain');
|
|
353
|
-
|
|
354
|
-
expect(newResource.hash).toBe(hash2);
|
|
355
|
-
expect(newResource.hash).not.toBe(hash1);
|
|
356
|
-
});
|
|
357
|
-
|
|
358
|
-
test('versioning works with Buffer content', () => {
|
|
359
|
-
const buffer1 = Buffer.from('binary content 1', 'utf-8');
|
|
360
|
-
const resources: AssetResource[] = [
|
|
361
|
-
{
|
|
362
|
-
id: 'res1',
|
|
363
|
-
type: 'data',
|
|
364
|
-
contentType: 'application/octet-stream',
|
|
365
|
-
hash: hashResource(buffer1)
|
|
366
|
-
}
|
|
367
|
-
];
|
|
368
|
-
|
|
369
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
370
|
-
const buffer2 = Buffer.from('binary content 2', 'utf-8');
|
|
371
|
-
const newResource = asset.addResourceVersion('res1', buffer2, 'application/octet-stream');
|
|
372
|
-
|
|
373
|
-
expect(newResource.version).toBe(2);
|
|
374
|
-
expect(newResource.hash).toBe(hashResource(buffer2));
|
|
375
|
-
});
|
|
376
|
-
|
|
377
|
-
test('versioning works across all layers (did:peer)', () => {
|
|
378
|
-
const resources: AssetResource[] = [
|
|
379
|
-
{
|
|
380
|
-
id: 'res1',
|
|
381
|
-
type: 'text',
|
|
382
|
-
content: 'v1',
|
|
383
|
-
contentType: 'text/plain',
|
|
384
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
385
|
-
}
|
|
386
|
-
];
|
|
387
|
-
|
|
388
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
389
|
-
expect(asset.currentLayer).toBe('did:peer');
|
|
390
|
-
|
|
391
|
-
const newResource = asset.addResourceVersion('res1', 'v2', 'text/plain');
|
|
392
|
-
expect(newResource.version).toBe(2);
|
|
393
|
-
});
|
|
394
|
-
|
|
395
|
-
test('versioning works across all layers (did:webvh)', () => {
|
|
396
|
-
const resources: AssetResource[] = [
|
|
397
|
-
{
|
|
398
|
-
id: 'res1',
|
|
399
|
-
type: 'text',
|
|
400
|
-
content: 'v1',
|
|
401
|
-
contentType: 'text/plain',
|
|
402
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
403
|
-
}
|
|
404
|
-
];
|
|
405
|
-
|
|
406
|
-
const asset = new OriginalsAsset(resources, buildDid('did:webvh:example.com:xyz'), emptyCreds);
|
|
407
|
-
expect(asset.currentLayer).toBe('did:webvh');
|
|
408
|
-
|
|
409
|
-
const newResource = asset.addResourceVersion('res1', 'v2', 'text/plain');
|
|
410
|
-
expect(newResource.version).toBe(2);
|
|
411
|
-
});
|
|
412
|
-
|
|
413
|
-
test('versioning works across all layers (did:btco)', () => {
|
|
414
|
-
const resources: AssetResource[] = [
|
|
415
|
-
{
|
|
416
|
-
id: 'res1',
|
|
417
|
-
type: 'text',
|
|
418
|
-
content: 'v1',
|
|
419
|
-
contentType: 'text/plain',
|
|
420
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
421
|
-
}
|
|
422
|
-
];
|
|
423
|
-
|
|
424
|
-
const asset = new OriginalsAsset(resources, buildDid('did:btco:123'), emptyCreds);
|
|
425
|
-
expect(asset.currentLayer).toBe('did:btco');
|
|
426
|
-
|
|
427
|
-
const newResource = asset.addResourceVersion('res1', 'v2', 'text/plain');
|
|
428
|
-
expect(newResource.version).toBe(2);
|
|
429
|
-
});
|
|
430
|
-
|
|
431
|
-
test('multiple resources can be versioned independently', () => {
|
|
432
|
-
const resources: AssetResource[] = [
|
|
433
|
-
{
|
|
434
|
-
id: 'res1',
|
|
435
|
-
type: 'text',
|
|
436
|
-
content: 'res1-v1',
|
|
437
|
-
contentType: 'text/plain',
|
|
438
|
-
hash: hashResource(Buffer.from('res1-v1', 'utf-8'))
|
|
439
|
-
},
|
|
440
|
-
{
|
|
441
|
-
id: 'res2',
|
|
442
|
-
type: 'text',
|
|
443
|
-
content: 'res2-v1',
|
|
444
|
-
contentType: 'text/plain',
|
|
445
|
-
hash: hashResource(Buffer.from('res2-v1', 'utf-8'))
|
|
446
|
-
}
|
|
447
|
-
];
|
|
448
|
-
|
|
449
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
450
|
-
|
|
451
|
-
asset.addResourceVersion('res1', 'res1-v2', 'text/plain');
|
|
452
|
-
asset.addResourceVersion('res2', 'res2-v2', 'text/plain');
|
|
453
|
-
asset.addResourceVersion('res1', 'res1-v3', 'text/plain');
|
|
454
|
-
|
|
455
|
-
const res1Versions = asset.getAllVersions('res1');
|
|
456
|
-
const res2Versions = asset.getAllVersions('res2');
|
|
457
|
-
|
|
458
|
-
expect(res1Versions.length).toBe(3);
|
|
459
|
-
expect(res2Versions.length).toBe(2);
|
|
460
|
-
});
|
|
461
|
-
|
|
462
|
-
test('timestamp is recorded for each version', () => {
|
|
463
|
-
const resources: AssetResource[] = [
|
|
464
|
-
{
|
|
465
|
-
id: 'res1',
|
|
466
|
-
type: 'text',
|
|
467
|
-
content: 'v1',
|
|
468
|
-
contentType: 'text/plain',
|
|
469
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
470
|
-
}
|
|
471
|
-
];
|
|
472
|
-
|
|
473
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
474
|
-
const beforeTime = new Date().toISOString();
|
|
475
|
-
|
|
476
|
-
const newResource = asset.addResourceVersion('res1', 'v2', 'text/plain');
|
|
477
|
-
const afterTime = new Date().toISOString();
|
|
478
|
-
|
|
479
|
-
expect(newResource.createdAt).toBeDefined();
|
|
480
|
-
expect(newResource.createdAt! >= beforeTime).toBe(true);
|
|
481
|
-
expect(newResource.createdAt! <= afterTime).toBe(true);
|
|
482
|
-
});
|
|
483
|
-
});
|
|
484
|
-
|
|
485
|
-
describe('OriginalsAsset - Unsorted Resource Loading', () => {
|
|
486
|
-
test('handles unsorted resources correctly when loading persisted data', () => {
|
|
487
|
-
const hash1 = hashResource(Buffer.from('v1', 'utf-8'));
|
|
488
|
-
const hash2 = hashResource(Buffer.from('v2', 'utf-8'));
|
|
489
|
-
const hash3 = hashResource(Buffer.from('v3', 'utf-8'));
|
|
490
|
-
|
|
491
|
-
// Simulate persisted resources loaded in wrong order [v3, v1, v2]
|
|
492
|
-
const unsortedResources: AssetResource[] = [
|
|
493
|
-
{
|
|
494
|
-
id: 'res1',
|
|
495
|
-
type: 'text',
|
|
496
|
-
content: 'v3',
|
|
497
|
-
contentType: 'text/plain',
|
|
498
|
-
hash: hash3,
|
|
499
|
-
version: 3,
|
|
500
|
-
previousVersionHash: hash2,
|
|
501
|
-
createdAt: '2025-01-03T00:00:00Z'
|
|
502
|
-
},
|
|
503
|
-
{
|
|
504
|
-
id: 'res1',
|
|
505
|
-
type: 'text',
|
|
506
|
-
content: 'v1',
|
|
507
|
-
contentType: 'text/plain',
|
|
508
|
-
hash: hash1,
|
|
509
|
-
version: 1,
|
|
510
|
-
createdAt: '2025-01-01T00:00:00Z'
|
|
511
|
-
},
|
|
512
|
-
{
|
|
513
|
-
id: 'res1',
|
|
514
|
-
type: 'text',
|
|
515
|
-
content: 'v2',
|
|
516
|
-
contentType: 'text/plain',
|
|
517
|
-
hash: hash2,
|
|
518
|
-
version: 2,
|
|
519
|
-
previousVersionHash: hash1,
|
|
520
|
-
createdAt: '2025-01-02T00:00:00Z'
|
|
521
|
-
}
|
|
522
|
-
];
|
|
523
|
-
|
|
524
|
-
const asset = new OriginalsAsset(unsortedResources, buildDid('did:peer:xyz'), emptyCreds);
|
|
525
|
-
|
|
526
|
-
// Verify version history is correct despite unsorted input
|
|
527
|
-
const history = asset.getResourceHistory('res1');
|
|
528
|
-
expect(history).not.toBeNull();
|
|
529
|
-
expect(history!.versions.length).toBe(3);
|
|
530
|
-
|
|
531
|
-
// Check versions are in correct order
|
|
532
|
-
expect(history!.versions[0].version).toBe(1);
|
|
533
|
-
expect(history!.versions[0].hash).toBe(hash1);
|
|
534
|
-
expect(history!.versions[0].previousVersionHash).toBeUndefined();
|
|
535
|
-
|
|
536
|
-
expect(history!.versions[1].version).toBe(2);
|
|
537
|
-
expect(history!.versions[1].hash).toBe(hash2);
|
|
538
|
-
expect(history!.versions[1].previousVersionHash).toBe(hash1);
|
|
539
|
-
|
|
540
|
-
expect(history!.versions[2].version).toBe(3);
|
|
541
|
-
expect(history!.versions[2].hash).toBe(hash3);
|
|
542
|
-
expect(history!.versions[2].previousVersionHash).toBe(hash2);
|
|
543
|
-
|
|
544
|
-
// Current version should be v3
|
|
545
|
-
expect(history!.currentVersion.version).toBe(3);
|
|
546
|
-
expect(history!.currentVersion.hash).toBe(hash3);
|
|
547
|
-
});
|
|
548
|
-
|
|
549
|
-
test('handles multiple resources with unsorted versions', () => {
|
|
550
|
-
const res1_v1_hash = hashResource(Buffer.from('res1-v1', 'utf-8'));
|
|
551
|
-
const res1_v2_hash = hashResource(Buffer.from('res1-v2', 'utf-8'));
|
|
552
|
-
const res2_v1_hash = hashResource(Buffer.from('res2-v1', 'utf-8'));
|
|
553
|
-
const res2_v2_hash = hashResource(Buffer.from('res2-v2', 'utf-8'));
|
|
554
|
-
|
|
555
|
-
// Multiple resources, each with unsorted versions
|
|
556
|
-
const resources: AssetResource[] = [
|
|
557
|
-
// res2 v2 (out of order)
|
|
558
|
-
{
|
|
559
|
-
id: 'res2',
|
|
560
|
-
type: 'text',
|
|
561
|
-
content: 'res2-v2',
|
|
562
|
-
contentType: 'text/plain',
|
|
563
|
-
hash: res2_v2_hash,
|
|
564
|
-
version: 2,
|
|
565
|
-
previousVersionHash: res2_v1_hash
|
|
566
|
-
},
|
|
567
|
-
// res1 v2 (out of order)
|
|
568
|
-
{
|
|
569
|
-
id: 'res1',
|
|
570
|
-
type: 'text',
|
|
571
|
-
content: 'res1-v2',
|
|
572
|
-
contentType: 'text/plain',
|
|
573
|
-
hash: res1_v2_hash,
|
|
574
|
-
version: 2,
|
|
575
|
-
previousVersionHash: res1_v1_hash
|
|
576
|
-
},
|
|
577
|
-
// res2 v1
|
|
578
|
-
{
|
|
579
|
-
id: 'res2',
|
|
580
|
-
type: 'text',
|
|
581
|
-
content: 'res2-v1',
|
|
582
|
-
contentType: 'text/plain',
|
|
583
|
-
hash: res2_v1_hash,
|
|
584
|
-
version: 1
|
|
585
|
-
},
|
|
586
|
-
// res1 v1
|
|
587
|
-
{
|
|
588
|
-
id: 'res1',
|
|
589
|
-
type: 'text',
|
|
590
|
-
content: 'res1-v1',
|
|
591
|
-
contentType: 'text/plain',
|
|
592
|
-
hash: res1_v1_hash,
|
|
593
|
-
version: 1
|
|
594
|
-
}
|
|
595
|
-
];
|
|
596
|
-
|
|
597
|
-
const asset = new OriginalsAsset(resources, buildDid('did:peer:xyz'), emptyCreds);
|
|
598
|
-
|
|
599
|
-
// Verify both resources have correct version chains
|
|
600
|
-
const res1History = asset.getResourceHistory('res1');
|
|
601
|
-
expect(res1History!.versions[0].version).toBe(1);
|
|
602
|
-
expect(res1History!.versions[1].version).toBe(2);
|
|
603
|
-
expect(res1History!.versions[1].previousVersionHash).toBe(res1_v1_hash);
|
|
604
|
-
|
|
605
|
-
const res2History = asset.getResourceHistory('res2');
|
|
606
|
-
expect(res2History!.versions[0].version).toBe(1);
|
|
607
|
-
expect(res2History!.versions[1].version).toBe(2);
|
|
608
|
-
expect(res2History!.versions[1].previousVersionHash).toBe(res2_v1_hash);
|
|
609
|
-
});
|
|
610
|
-
});
|
|
611
|
-
|
|
612
|
-
describe('OriginalsAsset - Credential Integration', () => {
|
|
613
|
-
test('credential can be issued for version creation (integration check)', async () => {
|
|
614
|
-
const resources: AssetResource[] = [
|
|
615
|
-
{
|
|
616
|
-
id: 'res1',
|
|
617
|
-
type: 'text',
|
|
618
|
-
content: 'v1',
|
|
619
|
-
contentType: 'text/plain',
|
|
620
|
-
hash: hashResource(Buffer.from('v1', 'utf-8'))
|
|
621
|
-
}
|
|
622
|
-
];
|
|
623
|
-
|
|
624
|
-
const didDoc = buildDid('did:peer:xyz');
|
|
625
|
-
const asset = new OriginalsAsset(resources, didDoc, emptyCreds);
|
|
626
|
-
|
|
627
|
-
// Create credential manager
|
|
628
|
-
const config: OriginalsConfig = {
|
|
629
|
-
network: 'regtest',
|
|
630
|
-
defaultKeyType: 'ES256K'
|
|
631
|
-
};
|
|
632
|
-
const didManager = new DIDManager(config);
|
|
633
|
-
const credentialManager = new CredentialManager(config, didManager);
|
|
634
|
-
|
|
635
|
-
// Issue a credential for resource update
|
|
636
|
-
const credential = await credentialManager.createResourceCredential(
|
|
637
|
-
'ResourceUpdated',
|
|
638
|
-
{
|
|
639
|
-
id: 'res1',
|
|
640
|
-
fromVersion: 1,
|
|
641
|
-
toVersion: 2,
|
|
642
|
-
timestamp: new Date().toISOString()
|
|
643
|
-
},
|
|
644
|
-
asset.id
|
|
645
|
-
);
|
|
646
|
-
|
|
647
|
-
expect(credential.type).toContain('ResourceUpdated');
|
|
648
|
-
expect(credential.credentialSubject).toHaveProperty('fromVersion');
|
|
649
|
-
expect(credential.credentialSubject).toHaveProperty('toVersion');
|
|
650
|
-
});
|
|
651
|
-
});
|