@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.
Files changed (212) hide show
  1. package/package.json +4 -1
  2. package/.eslintrc.json +0 -33
  3. package/src/adapters/FeeOracleMock.ts +0 -9
  4. package/src/adapters/index.ts +0 -5
  5. package/src/adapters/providers/OrdHttpProvider.ts +0 -126
  6. package/src/adapters/providers/OrdMockProvider.ts +0 -101
  7. package/src/adapters/types.ts +0 -66
  8. package/src/bitcoin/BitcoinManager.ts +0 -330
  9. package/src/bitcoin/BroadcastClient.ts +0 -54
  10. package/src/bitcoin/OrdinalsClient.ts +0 -119
  11. package/src/bitcoin/PSBTBuilder.ts +0 -106
  12. package/src/bitcoin/fee-calculation.ts +0 -38
  13. package/src/bitcoin/providers/OrdNodeProvider.ts +0 -92
  14. package/src/bitcoin/providers/OrdinalsProvider.ts +0 -56
  15. package/src/bitcoin/providers/types.ts +0 -59
  16. package/src/bitcoin/transactions/commit.ts +0 -465
  17. package/src/bitcoin/transactions/index.ts +0 -13
  18. package/src/bitcoin/transfer.ts +0 -43
  19. package/src/bitcoin/utxo-selection.ts +0 -322
  20. package/src/bitcoin/utxo.ts +0 -113
  21. package/src/contexts/credentials-v1.json +0 -237
  22. package/src/contexts/credentials-v2-examples.json +0 -5
  23. package/src/contexts/credentials-v2.json +0 -340
  24. package/src/contexts/credentials.json +0 -237
  25. package/src/contexts/data-integrity-v2.json +0 -81
  26. package/src/contexts/dids.json +0 -58
  27. package/src/contexts/ed255192020.json +0 -93
  28. package/src/contexts/ordinals-plus.json +0 -23
  29. package/src/contexts/originals.json +0 -22
  30. package/src/core/OriginalsSDK.ts +0 -416
  31. package/src/crypto/Multikey.ts +0 -194
  32. package/src/crypto/Signer.ts +0 -254
  33. package/src/crypto/noble-init.ts +0 -121
  34. package/src/did/BtcoDidResolver.ts +0 -227
  35. package/src/did/DIDManager.ts +0 -694
  36. package/src/did/Ed25519Verifier.ts +0 -68
  37. package/src/did/KeyManager.ts +0 -236
  38. package/src/did/WebVHManager.ts +0 -498
  39. package/src/did/createBtcoDidDocument.ts +0 -59
  40. package/src/did/providers/OrdinalsClientProviderAdapter.ts +0 -68
  41. package/src/events/EventEmitter.ts +0 -222
  42. package/src/events/index.ts +0 -19
  43. package/src/events/types.ts +0 -331
  44. package/src/examples/basic-usage.ts +0 -78
  45. package/src/examples/create-module-original.ts +0 -435
  46. package/src/examples/full-lifecycle-flow.ts +0 -514
  47. package/src/examples/run.ts +0 -60
  48. package/src/index.ts +0 -150
  49. package/src/kinds/KindRegistry.ts +0 -290
  50. package/src/kinds/index.ts +0 -74
  51. package/src/kinds/types.ts +0 -470
  52. package/src/kinds/validators/AgentValidator.ts +0 -257
  53. package/src/kinds/validators/AppValidator.ts +0 -211
  54. package/src/kinds/validators/DatasetValidator.ts +0 -242
  55. package/src/kinds/validators/DocumentValidator.ts +0 -311
  56. package/src/kinds/validators/MediaValidator.ts +0 -269
  57. package/src/kinds/validators/ModuleValidator.ts +0 -225
  58. package/src/kinds/validators/base.ts +0 -276
  59. package/src/kinds/validators/index.ts +0 -12
  60. package/src/lifecycle/BatchOperations.ts +0 -373
  61. package/src/lifecycle/LifecycleManager.ts +0 -2126
  62. package/src/lifecycle/OriginalsAsset.ts +0 -524
  63. package/src/lifecycle/ProvenanceQuery.ts +0 -280
  64. package/src/lifecycle/ResourceVersioning.ts +0 -163
  65. package/src/migration/MigrationManager.ts +0 -527
  66. package/src/migration/audit/AuditLogger.ts +0 -176
  67. package/src/migration/checkpoint/CheckpointManager.ts +0 -112
  68. package/src/migration/checkpoint/CheckpointStorage.ts +0 -101
  69. package/src/migration/index.ts +0 -33
  70. package/src/migration/operations/BaseMigration.ts +0 -126
  71. package/src/migration/operations/PeerToBtcoMigration.ts +0 -105
  72. package/src/migration/operations/PeerToWebvhMigration.ts +0 -62
  73. package/src/migration/operations/WebvhToBtcoMigration.ts +0 -105
  74. package/src/migration/rollback/RollbackManager.ts +0 -170
  75. package/src/migration/state/StateMachine.ts +0 -92
  76. package/src/migration/state/StateTracker.ts +0 -156
  77. package/src/migration/types.ts +0 -344
  78. package/src/migration/validation/BitcoinValidator.ts +0 -107
  79. package/src/migration/validation/CredentialValidator.ts +0 -62
  80. package/src/migration/validation/DIDCompatibilityValidator.ts +0 -151
  81. package/src/migration/validation/LifecycleValidator.ts +0 -64
  82. package/src/migration/validation/StorageValidator.ts +0 -79
  83. package/src/migration/validation/ValidationPipeline.ts +0 -213
  84. package/src/resources/ResourceManager.ts +0 -655
  85. package/src/resources/index.ts +0 -21
  86. package/src/resources/types.ts +0 -202
  87. package/src/storage/LocalStorageAdapter.ts +0 -61
  88. package/src/storage/MemoryStorageAdapter.ts +0 -29
  89. package/src/storage/StorageAdapter.ts +0 -25
  90. package/src/storage/index.ts +0 -3
  91. package/src/types/bitcoin.ts +0 -98
  92. package/src/types/common.ts +0 -92
  93. package/src/types/credentials.ts +0 -88
  94. package/src/types/did.ts +0 -31
  95. package/src/types/external-shims.d.ts +0 -53
  96. package/src/types/index.ts +0 -7
  97. package/src/types/network.ts +0 -175
  98. package/src/utils/EventLogger.ts +0 -298
  99. package/src/utils/Logger.ts +0 -322
  100. package/src/utils/MetricsCollector.ts +0 -358
  101. package/src/utils/bitcoin-address.ts +0 -130
  102. package/src/utils/cbor.ts +0 -12
  103. package/src/utils/encoding.ts +0 -127
  104. package/src/utils/hash.ts +0 -6
  105. package/src/utils/retry.ts +0 -46
  106. package/src/utils/satoshi-validation.ts +0 -196
  107. package/src/utils/serialization.ts +0 -96
  108. package/src/utils/telemetry.ts +0 -40
  109. package/src/utils/validation.ts +0 -119
  110. package/src/vc/CredentialManager.ts +0 -918
  111. package/src/vc/Issuer.ts +0 -100
  112. package/src/vc/Verifier.ts +0 -47
  113. package/src/vc/cryptosuites/bbs.ts +0 -253
  114. package/src/vc/cryptosuites/bbsSimple.ts +0 -21
  115. package/src/vc/cryptosuites/eddsa.ts +0 -99
  116. package/src/vc/documentLoader.ts +0 -67
  117. package/src/vc/proofs/data-integrity.ts +0 -33
  118. package/src/vc/utils/jsonld.ts +0 -18
  119. package/tests/__mocks__/bbs-signatures.js +0 -17
  120. package/tests/__mocks__/mf-base58.js +0 -24
  121. package/tests/fixtures/did-documents.ts +0 -247
  122. package/tests/index.test.ts +0 -21
  123. package/tests/integration/BatchOperations.test.ts +0 -531
  124. package/tests/integration/CompleteLifecycle.e2e.test.ts +0 -735
  125. package/tests/integration/CredentialManager.test.ts +0 -42
  126. package/tests/integration/DIDManager.test.ts +0 -41
  127. package/tests/integration/DidPeerToWebVhFlow.test.ts +0 -351
  128. package/tests/integration/Events.test.ts +0 -435
  129. package/tests/integration/Lifecycle.transfer.btco.integration.test.ts +0 -25
  130. package/tests/integration/LifecycleManager.test.ts +0 -21
  131. package/tests/integration/MultikeyFlow.test.ts +0 -52
  132. package/tests/integration/TelemetryIntegration.test.ts +0 -395
  133. package/tests/integration/WebVhPublish.test.ts +0 -48
  134. package/tests/integration/createTypedOriginal.test.ts +0 -379
  135. package/tests/integration/migration/peer-to-webvh.test.ts +0 -172
  136. package/tests/manual/test-commit-creation.ts +0 -323
  137. package/tests/mocks/MockKeyStore.ts +0 -38
  138. package/tests/mocks/adapters/MemoryStorageAdapter.ts +0 -24
  139. package/tests/mocks/adapters/MockFeeOracle.ts +0 -11
  140. package/tests/mocks/adapters/MockOrdinalsProvider.ts +0 -76
  141. package/tests/mocks/adapters/OrdMockProvider.test.ts +0 -176
  142. package/tests/mocks/adapters/index.ts +0 -6
  143. package/tests/performance/BatchOperations.perf.test.ts +0 -403
  144. package/tests/performance/logging.perf.test.ts +0 -336
  145. package/tests/sdk.test.ts +0 -43
  146. package/tests/security/bitcoin-penetration-tests.test.ts +0 -622
  147. package/tests/setup.bun.ts +0 -69
  148. package/tests/setup.jest.ts +0 -23
  149. package/tests/stress/batch-operations-stress.test.ts +0 -571
  150. package/tests/unit/adapters/FeeOracleMock.test.ts +0 -40
  151. package/tests/unit/bitcoin/BitcoinManager.test.ts +0 -293
  152. package/tests/unit/bitcoin/BroadcastClient.test.ts +0 -52
  153. package/tests/unit/bitcoin/OrdNodeProvider.test.ts +0 -53
  154. package/tests/unit/bitcoin/OrdinalsClient.test.ts +0 -381
  155. package/tests/unit/bitcoin/OrdinalsClientProvider.test.ts +0 -102
  156. package/tests/unit/bitcoin/PSBTBuilder.test.ts +0 -84
  157. package/tests/unit/bitcoin/fee-calculation.test.ts +0 -261
  158. package/tests/unit/bitcoin/transactions/commit.test.ts +0 -649
  159. package/tests/unit/bitcoin/transfer.test.ts +0 -31
  160. package/tests/unit/bitcoin/utxo-selection-new.test.ts +0 -502
  161. package/tests/unit/bitcoin/utxo.more.test.ts +0 -39
  162. package/tests/unit/bitcoin/utxo.selection.test.ts +0 -38
  163. package/tests/unit/core/OriginalsSDK.test.ts +0 -152
  164. package/tests/unit/crypto/Multikey.test.ts +0 -206
  165. package/tests/unit/crypto/Signer.test.ts +0 -408
  166. package/tests/unit/did/BtcoDidResolver.test.ts +0 -611
  167. package/tests/unit/did/DIDManager.more.test.ts +0 -43
  168. package/tests/unit/did/DIDManager.test.ts +0 -185
  169. package/tests/unit/did/Ed25519Verifier.test.ts +0 -160
  170. package/tests/unit/did/KeyManager.test.ts +0 -452
  171. package/tests/unit/did/OrdinalsClientProviderAdapter.test.ts +0 -45
  172. package/tests/unit/did/WebVHManager.test.ts +0 -435
  173. package/tests/unit/did/createBtcoDidDocument.test.ts +0 -67
  174. package/tests/unit/did/providers/OrdinalsClientProviderAdapter.test.ts +0 -159
  175. package/tests/unit/events/EventEmitter.test.ts +0 -407
  176. package/tests/unit/kinds/KindRegistry.test.ts +0 -329
  177. package/tests/unit/kinds/types.test.ts +0 -409
  178. package/tests/unit/kinds/validators.test.ts +0 -651
  179. package/tests/unit/lifecycle/BatchOperations.test.ts +0 -527
  180. package/tests/unit/lifecycle/LifecycleManager.cleanapi.test.ts +0 -441
  181. package/tests/unit/lifecycle/LifecycleManager.keymanagement.test.ts +0 -312
  182. package/tests/unit/lifecycle/LifecycleManager.prov.test.ts +0 -18
  183. package/tests/unit/lifecycle/LifecycleManager.test.ts +0 -213
  184. package/tests/unit/lifecycle/LifecycleManager.transfer.unit.test.ts +0 -30
  185. package/tests/unit/lifecycle/OriginalsAsset.test.ts +0 -176
  186. package/tests/unit/lifecycle/ProvenanceQuery.test.ts +0 -577
  187. package/tests/unit/lifecycle/ResourceVersioning.test.ts +0 -651
  188. package/tests/unit/resources/ResourceManager.test.ts +0 -740
  189. package/tests/unit/storage/MemoryStorageAdapter.test.ts +0 -93
  190. package/tests/unit/types/network.test.ts +0 -255
  191. package/tests/unit/utils/EventIntegration.test.ts +0 -384
  192. package/tests/unit/utils/Logger.test.ts +0 -473
  193. package/tests/unit/utils/MetricsCollector.test.ts +0 -358
  194. package/tests/unit/utils/bitcoin-address.test.ts +0 -250
  195. package/tests/unit/utils/cbor.test.ts +0 -35
  196. package/tests/unit/utils/encoding.test.ts +0 -318
  197. package/tests/unit/utils/hash.test.ts +0 -12
  198. package/tests/unit/utils/retry.test.ts +0 -100
  199. package/tests/unit/utils/satoshi-validation.test.ts +0 -354
  200. package/tests/unit/utils/serialization.test.ts +0 -124
  201. package/tests/unit/utils/telemetry.test.ts +0 -52
  202. package/tests/unit/utils/validation.test.ts +0 -141
  203. package/tests/unit/vc/CredentialManager.helpers.test.ts +0 -527
  204. package/tests/unit/vc/CredentialManager.test.ts +0 -487
  205. package/tests/unit/vc/Issuer.test.ts +0 -107
  206. package/tests/unit/vc/Verifier.test.ts +0 -525
  207. package/tests/unit/vc/bbs.test.ts +0 -282
  208. package/tests/unit/vc/cryptosuites/eddsa.test.ts +0 -398
  209. package/tests/unit/vc/documentLoader.test.ts +0 -121
  210. package/tests/unit/vc/proofs/data-integrity.test.ts +0 -24
  211. package/tsconfig.json +0 -31
  212. package/tsconfig.test.json +0 -15
@@ -1,655 +0,0 @@
1
- /**
2
- * ResourceManager - CRUD operations for immutable, versioned resources.
3
- *
4
- * Resources in the Originals SDK are content-addressed and immutable. Each "update"
5
- * creates a new version with a new content hash, linked to the previous version
6
- * via previousVersionHash. This creates a verifiable provenance chain.
7
- *
8
- * @example
9
- * ```typescript
10
- * const manager = new ResourceManager();
11
- *
12
- * // Create a new resource
13
- * const resource = manager.createResource('Hello, World!', {
14
- * type: 'text',
15
- * contentType: 'text/plain'
16
- * });
17
- *
18
- * // Update creates a new version
19
- * const updatedResource = manager.updateResource(resource, 'Hello, Updated World!', {
20
- * changes: 'Updated greeting'
21
- * });
22
- *
23
- * // Get version history
24
- * const history = manager.getResourceHistory(resource.id);
25
- * ```
26
- */
27
-
28
- import { sha256 } from '@noble/hashes/sha2.js';
29
- import { bytesToHex } from '@noble/hashes/utils.js';
30
- import { v4 as uuidv4 } from 'uuid';
31
- import type {
32
- Resource,
33
- ResourceOptions,
34
- ResourceUpdateOptions,
35
- ResourceVersionHistory,
36
- ResourceManagerConfig,
37
- ResourceValidationResult,
38
- ResourceType,
39
- } from './types.js';
40
- import { MIME_TYPE_MAP, DEFAULT_RESOURCE_CONFIG } from './types.js';
41
-
42
- /**
43
- * Regular expression for validating MIME types according to RFC 6838.
44
- * Format: type/subtype where type and subtype are restricted character sets.
45
- */
46
- const MIME_TYPE_REGEX = /^[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}\/[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}$/;
47
-
48
- /**
49
- * ResourceManager provides CRUD operations for immutable, content-addressed resources
50
- * with versioning support and validation.
51
- */
52
- export class ResourceManager {
53
- private resources: Map<string, Resource[]>;
54
- private config: Required<ResourceManagerConfig>;
55
-
56
- /**
57
- * Create a new ResourceManager instance.
58
- *
59
- * @param config - Optional configuration for the manager
60
- */
61
- constructor(config?: ResourceManagerConfig) {
62
- this.resources = new Map();
63
- this.config = { ...DEFAULT_RESOURCE_CONFIG, ...config };
64
- }
65
-
66
- /**
67
- * Create a new resource from content.
68
- *
69
- * @param content - The resource content (string for text, Buffer for binary)
70
- * @param options - Creation options including type and contentType
71
- * @returns The created Resource
72
- * @throws Error if content or options are invalid
73
- *
74
- * @example
75
- * ```typescript
76
- * // Create a text resource
77
- * const textResource = manager.createResource('# README\nHello', {
78
- * type: 'text',
79
- * contentType: 'text/markdown'
80
- * });
81
- *
82
- * // Create a binary resource (image)
83
- * const imageBuffer = fs.readFileSync('image.png');
84
- * const imageResource = manager.createResource(imageBuffer, {
85
- * type: 'image',
86
- * contentType: 'image/png'
87
- * });
88
- * ```
89
- */
90
- createResource(content: Buffer | string, options: ResourceOptions): Resource {
91
- // Validate inputs
92
- if (content === null || content === undefined) {
93
- throw new Error('Content is required');
94
- }
95
- if (!options) {
96
- throw new Error('Options are required');
97
- }
98
- if (!options.type) {
99
- throw new Error('Resource type is required');
100
- }
101
- if (!options.contentType) {
102
- throw new Error('Content type is required');
103
- }
104
-
105
- // Validate MIME type format
106
- if (!this.isValidMimeType(options.contentType)) {
107
- throw new Error(`Invalid MIME type format: ${options.contentType}`);
108
- }
109
-
110
- // Check allowed content types
111
- if (this.config.allowedContentTypes.length > 0 &&
112
- !this.config.allowedContentTypes.includes(options.contentType)) {
113
- throw new Error(`Content type not allowed: ${options.contentType}. Allowed types: ${this.config.allowedContentTypes.join(', ')}`);
114
- }
115
-
116
- // Convert content to buffer for consistent handling
117
- const contentBuffer = this.toBuffer(content);
118
-
119
- // Validate size
120
- const maxSize = options.maxSize || this.config.defaultMaxSize;
121
- if (contentBuffer.length > maxSize) {
122
- throw new Error(`Resource size (${contentBuffer.length} bytes) exceeds maximum allowed size (${maxSize} bytes)`);
123
- }
124
-
125
- // Generate hash
126
- const hash = this.hashContent(contentBuffer);
127
-
128
- // Generate or use provided ID
129
- const id = options.id || uuidv4();
130
-
131
- // Create resource object
132
- const resource: Resource = {
133
- id,
134
- type: options.type,
135
- contentType: options.contentType,
136
- hash,
137
- size: contentBuffer.length,
138
- version: 1,
139
- createdAt: new Date().toISOString(),
140
- url: options.url,
141
- description: options.description,
142
- };
143
-
144
- // Store content if configured to do so
145
- if (this.config.storeContent) {
146
- if (this.isBinaryContent(content)) {
147
- resource.contentBase64 = contentBuffer.toString('base64');
148
- } else {
149
- resource.content = typeof content === 'string' ? content : contentBuffer.toString('utf-8');
150
- }
151
- }
152
-
153
- // Store in version history
154
- this.resources.set(id, [resource]);
155
-
156
- return resource;
157
- }
158
-
159
- /**
160
- * Update a resource by creating a new version.
161
- * The original resource remains unchanged (immutable versioning).
162
- *
163
- * @param resource - The resource to update (or its ID)
164
- * @param newContent - The new content
165
- * @param options - Optional update options including change description
166
- * @returns The new version of the resource
167
- * @throws Error if resource not found or content unchanged
168
- *
169
- * @example
170
- * ```typescript
171
- * const v2 = manager.updateResource(originalResource, 'Updated content', {
172
- * changes: 'Fixed typo in documentation'
173
- * });
174
- *
175
- * console.log(v2.version); // 2
176
- * console.log(v2.previousVersionHash); // hash of v1
177
- * ```
178
- */
179
- updateResource(
180
- resource: Resource | string,
181
- newContent: Buffer | string,
182
- options?: ResourceUpdateOptions
183
- ): Resource {
184
- const resourceId = typeof resource === 'string' ? resource : resource.id;
185
-
186
- // Get version history
187
- const versions = this.resources.get(resourceId);
188
- if (!versions || versions.length === 0) {
189
- throw new Error(`Resource not found: ${resourceId}`);
190
- }
191
-
192
- // Get current (latest) version
193
- const currentVersion = versions[versions.length - 1];
194
-
195
- // Convert content to buffer
196
- const contentBuffer = this.toBuffer(newContent);
197
-
198
- // Generate hash for new content
199
- const newHash = this.hashContent(contentBuffer);
200
-
201
- // Check if content has actually changed
202
- if (newHash === currentVersion.hash) {
203
- throw new Error('Content unchanged - new version would be identical to current version');
204
- }
205
-
206
- // Validate size
207
- if (contentBuffer.length > this.config.defaultMaxSize) {
208
- throw new Error(`Resource size (${contentBuffer.length} bytes) exceeds maximum allowed size (${this.config.defaultMaxSize} bytes)`);
209
- }
210
-
211
- // Determine content type (use provided or inherit from previous)
212
- const contentType = options?.contentType || currentVersion.contentType;
213
-
214
- // Validate new content type if changed
215
- if (options?.contentType && !this.isValidMimeType(options.contentType)) {
216
- throw new Error(`Invalid MIME type format: ${options.contentType}`);
217
- }
218
-
219
- // Create new version
220
- const newVersion: Resource = {
221
- id: resourceId,
222
- type: currentVersion.type,
223
- contentType,
224
- hash: newHash,
225
- size: contentBuffer.length,
226
- version: (currentVersion.version || 1) + 1,
227
- previousVersionHash: currentVersion.hash,
228
- createdAt: new Date().toISOString(),
229
- url: currentVersion.url,
230
- description: currentVersion.description,
231
- };
232
-
233
- // Store content if configured
234
- if (this.config.storeContent) {
235
- if (this.isBinaryContent(newContent)) {
236
- newVersion.contentBase64 = contentBuffer.toString('base64');
237
- } else {
238
- newVersion.content = typeof newContent === 'string' ? newContent : contentBuffer.toString('utf-8');
239
- }
240
- }
241
-
242
- // Add to version history
243
- versions.push(newVersion);
244
-
245
- return newVersion;
246
- }
247
-
248
- /**
249
- * Get the complete version history for a resource.
250
- *
251
- * @param resourceId - The logical resource ID
252
- * @returns Array of all versions (oldest to newest), or empty array if not found
253
- *
254
- * @example
255
- * ```typescript
256
- * const history = manager.getResourceHistory('my-resource-id');
257
- * console.log(`Found ${history.length} versions`);
258
- * history.forEach((v, i) => console.log(`v${i + 1}: ${v.hash}`));
259
- * ```
260
- */
261
- getResourceHistory(resourceId: string): Resource[] {
262
- const versions = this.resources.get(resourceId);
263
- if (!versions) {
264
- return [];
265
- }
266
- return [...versions]; // Return copy to prevent external mutation
267
- }
268
-
269
- /**
270
- * Get detailed version history with metadata.
271
- *
272
- * @param resourceId - The logical resource ID
273
- * @returns ResourceVersionHistory object or null if not found
274
- */
275
- getResourceVersionHistory(resourceId: string): ResourceVersionHistory | null {
276
- const versions = this.resources.get(resourceId);
277
- if (!versions || versions.length === 0) {
278
- return null;
279
- }
280
-
281
- return {
282
- resourceId,
283
- versions: [...versions],
284
- currentVersion: versions[versions.length - 1],
285
- versionCount: versions.length,
286
- };
287
- }
288
-
289
- /**
290
- * Get a specific version of a resource.
291
- *
292
- * @param resourceId - The logical resource ID
293
- * @param version - Version number (1-indexed)
294
- * @returns The resource at that version, or null if not found
295
- */
296
- getResourceVersion(resourceId: string, version: number): Resource | null {
297
- const versions = this.resources.get(resourceId);
298
- if (!versions || version < 1 || version > versions.length) {
299
- return null;
300
- }
301
- return versions[version - 1];
302
- }
303
-
304
- /**
305
- * Get the current (latest) version of a resource.
306
- *
307
- * @param resourceId - The logical resource ID
308
- * @returns The current version, or null if not found
309
- */
310
- getCurrentVersion(resourceId: string): Resource | null {
311
- const versions = this.resources.get(resourceId);
312
- if (!versions || versions.length === 0) {
313
- return null;
314
- }
315
- return versions[versions.length - 1];
316
- }
317
-
318
- /**
319
- * Get a resource by its content hash.
320
- *
321
- * @param hash - The content hash to search for
322
- * @returns The resource with that hash, or null if not found
323
- */
324
- getResourceByHash(hash: string): Resource | null {
325
- for (const versions of this.resources.values()) {
326
- const found = versions.find(r => r.hash === hash);
327
- if (found) {
328
- return found;
329
- }
330
- }
331
- return null;
332
- }
333
-
334
- /**
335
- * Validate a resource object.
336
- *
337
- * @param resource - The resource to validate
338
- * @returns ValidationResult with valid flag and any errors/warnings
339
- *
340
- * @example
341
- * ```typescript
342
- * const result = manager.validateResource(resource);
343
- * if (!result.valid) {
344
- * console.error('Validation errors:', result.errors);
345
- * }
346
- * if (result.warnings.length > 0) {
347
- * console.warn('Warnings:', result.warnings);
348
- * }
349
- * ```
350
- */
351
- validateResource(resource: Resource): ResourceValidationResult {
352
- const errors: string[] = [];
353
- const warnings: string[] = [];
354
-
355
- // Required field validation
356
- if (!resource) {
357
- return { valid: false, errors: ['Resource is null or undefined'], warnings: [] };
358
- }
359
-
360
- if (!resource.id || typeof resource.id !== 'string') {
361
- errors.push('Missing or invalid resource ID');
362
- }
363
-
364
- if (!resource.type || typeof resource.type !== 'string') {
365
- errors.push('Missing or invalid resource type');
366
- }
367
-
368
- if (!resource.contentType || typeof resource.contentType !== 'string') {
369
- errors.push('Missing or invalid content type');
370
- } else if (this.config.strictMimeValidation && !this.isValidMimeType(resource.contentType)) {
371
- errors.push(`Invalid MIME type format: ${resource.contentType}`);
372
- }
373
-
374
- if (!resource.hash || typeof resource.hash !== 'string') {
375
- errors.push('Missing or invalid hash');
376
- } else if (!/^[0-9a-fA-F]{64}$/.test(resource.hash)) {
377
- errors.push('Invalid hash format (must be 64 character hex string)');
378
- }
379
-
380
- // Version chain validation
381
- if (resource.version !== undefined) {
382
- if (typeof resource.version !== 'number' || resource.version < 1) {
383
- errors.push('Invalid version number (must be positive integer)');
384
- }
385
-
386
- // v1 should not have previousVersionHash
387
- if (resource.version === 1 && resource.previousVersionHash) {
388
- warnings.push('First version should not have previousVersionHash');
389
- }
390
-
391
- // v2+ should have previousVersionHash
392
- if (resource.version > 1 && !resource.previousVersionHash) {
393
- errors.push('Versions greater than 1 must have previousVersionHash');
394
- }
395
- }
396
-
397
- // Size validation
398
- if (resource.size !== undefined) {
399
- if (typeof resource.size !== 'number' || resource.size < 0) {
400
- errors.push('Invalid size (must be non-negative number)');
401
- }
402
-
403
- if (resource.size > this.config.defaultMaxSize) {
404
- warnings.push(`Resource size (${resource.size} bytes) exceeds default maximum (${this.config.defaultMaxSize} bytes)`);
405
- }
406
- }
407
-
408
- // Timestamp validation
409
- if (resource.createdAt) {
410
- const date = new Date(resource.createdAt);
411
- if (isNaN(date.getTime())) {
412
- errors.push('Invalid createdAt timestamp');
413
- }
414
- }
415
-
416
- // Content hash verification (if content is present)
417
- if (resource.content || resource.contentBase64) {
418
- const content = resource.content
419
- ? Buffer.from(resource.content, 'utf-8')
420
- : Buffer.from(resource.contentBase64 || '', 'base64');
421
- const computedHash = this.hashContent(content);
422
-
423
- if (computedHash !== resource.hash) {
424
- errors.push(`Content hash mismatch: expected ${resource.hash}, computed ${computedHash}`);
425
- }
426
- }
427
-
428
- // Check allowed content types if configured
429
- if (this.config.allowedContentTypes.length > 0 &&
430
- resource.contentType &&
431
- !this.config.allowedContentTypes.includes(resource.contentType)) {
432
- errors.push(`Content type not allowed: ${resource.contentType}`);
433
- }
434
-
435
- return {
436
- valid: errors.length === 0,
437
- errors,
438
- warnings,
439
- };
440
- }
441
-
442
- /**
443
- * Verify the integrity of a resource's version chain.
444
- * Ensures that previousVersionHash references form a valid chain.
445
- *
446
- * @param resourceId - The logical resource ID to verify
447
- * @returns ResourceValidationResult indicating chain integrity
448
- */
449
- verifyVersionChain(resourceId: string): ResourceValidationResult {
450
- const errors: string[] = [];
451
- const warnings: string[] = [];
452
-
453
- const versions = this.resources.get(resourceId);
454
- if (!versions || versions.length === 0) {
455
- return { valid: false, errors: ['Resource not found'], warnings: [] };
456
- }
457
-
458
- for (let i = 0; i < versions.length; i++) {
459
- const version = versions[i];
460
-
461
- // Check version numbers are sequential
462
- if ((version.version || i + 1) !== i + 1) {
463
- errors.push(`Version number mismatch at index ${i}: expected ${i + 1}, got ${version.version}`);
464
- }
465
-
466
- // First version should not have previousVersionHash
467
- if (i === 0) {
468
- if (version.previousVersionHash) {
469
- warnings.push('First version has previousVersionHash (should not exist for v1)');
470
- }
471
- } else {
472
- // Subsequent versions must link to previous
473
- const prevVersion = versions[i - 1];
474
- if (!version.previousVersionHash) {
475
- errors.push(`Version ${i + 1} missing previousVersionHash`);
476
- } else if (version.previousVersionHash !== prevVersion.hash) {
477
- errors.push(`Version ${i + 1} previousVersionHash mismatch: expected ${prevVersion.hash}, got ${version.previousVersionHash}`);
478
- }
479
- }
480
- }
481
-
482
- return {
483
- valid: errors.length === 0,
484
- errors,
485
- warnings,
486
- };
487
- }
488
-
489
- /**
490
- * Hash content using SHA-256.
491
- *
492
- * @param content - Content to hash (string or Buffer)
493
- * @returns Hex-encoded SHA-256 hash
494
- *
495
- * @example
496
- * ```typescript
497
- * const hash = manager.hashContent('Hello, World!');
498
- * console.log(hash); // 64-character hex string
499
- * ```
500
- */
501
- hashContent(content: Buffer | string): string {
502
- const buffer = this.toBuffer(content);
503
- const hash = sha256(buffer);
504
- return bytesToHex(hash);
505
- }
506
-
507
- /**
508
- * Delete a resource and all its versions.
509
- *
510
- * @param resourceId - The resource ID to delete
511
- * @returns true if deleted, false if not found
512
- */
513
- deleteResource(resourceId: string): boolean {
514
- return this.resources.delete(resourceId);
515
- }
516
-
517
- /**
518
- * List all resource IDs managed by this instance.
519
- *
520
- * @returns Array of resource IDs
521
- */
522
- listResourceIds(): string[] {
523
- return Array.from(this.resources.keys());
524
- }
525
-
526
- /**
527
- * Get the total number of resources (unique IDs) managed.
528
- *
529
- * @returns Number of resources
530
- */
531
- getResourceCount(): number {
532
- return this.resources.size;
533
- }
534
-
535
- /**
536
- * Get the total number of versions across all resources.
537
- *
538
- * @returns Total version count
539
- */
540
- getTotalVersionCount(): number {
541
- let count = 0;
542
- for (const versions of this.resources.values()) {
543
- count += versions.length;
544
- }
545
- return count;
546
- }
547
-
548
- /**
549
- * Import a resource from an existing AssetResource.
550
- * Useful for loading resources from storage or external sources.
551
- *
552
- * @param assetResource - The AssetResource to import
553
- * @returns The imported Resource
554
- */
555
- importResource(assetResource: Resource): Resource {
556
- const resourceId = assetResource.id;
557
-
558
- // Get or create version array
559
- let versions = this.resources.get(resourceId);
560
- if (!versions) {
561
- versions = [];
562
- this.resources.set(resourceId, versions);
563
- }
564
-
565
- // Check if this version already exists (by hash)
566
- const existingVersion = versions.find(v => v.hash === assetResource.hash);
567
- if (existingVersion) {
568
- return existingVersion;
569
- }
570
-
571
- // Add to version array (maintain order by version number)
572
- const version = assetResource.version || 1;
573
- const insertIndex = versions.findIndex(v => (v.version || 1) > version);
574
- if (insertIndex === -1) {
575
- versions.push(assetResource);
576
- } else {
577
- versions.splice(insertIndex, 0, assetResource);
578
- }
579
-
580
- return assetResource;
581
- }
582
-
583
- /**
584
- * Export all resources as an array (for serialization).
585
- *
586
- * @returns Array of all resources (all versions)
587
- */
588
- exportResources(): Resource[] {
589
- const allResources: Resource[] = [];
590
- for (const versions of this.resources.values()) {
591
- allResources.push(...versions);
592
- }
593
- return allResources;
594
- }
595
-
596
- /**
597
- * Clear all resources from this manager.
598
- */
599
- clear(): void {
600
- this.resources.clear();
601
- }
602
-
603
- /**
604
- * Infer resource type from MIME type.
605
- *
606
- * @param contentType - The MIME content type
607
- * @returns Inferred ResourceType
608
- */
609
- static inferResourceType(contentType: string): ResourceType {
610
- // Check exact match first
611
- if (contentType in MIME_TYPE_MAP) {
612
- return MIME_TYPE_MAP[contentType];
613
- }
614
-
615
- // Check by prefix
616
- const prefix = contentType.split('/')[0];
617
- switch (prefix) {
618
- case 'image':
619
- return 'image';
620
- case 'audio':
621
- return 'audio';
622
- case 'video':
623
- return 'video';
624
- case 'text':
625
- return 'text';
626
- default:
627
- return 'other';
628
- }
629
- }
630
-
631
- /**
632
- * Check if a string is a valid MIME type format.
633
- */
634
- private isValidMimeType(mimeType: string): boolean {
635
- return MIME_TYPE_REGEX.test(mimeType);
636
- }
637
-
638
- /**
639
- * Convert content to Buffer.
640
- */
641
- private toBuffer(content: Buffer | string): Buffer {
642
- if (Buffer.isBuffer(content)) {
643
- return content;
644
- }
645
- return Buffer.from(content, 'utf-8');
646
- }
647
-
648
- /**
649
- * Check if content is binary (Buffer) rather than text.
650
- */
651
- private isBinaryContent(content: Buffer | string): boolean {
652
- return Buffer.isBuffer(content);
653
- }
654
- }
655
-
@@ -1,21 +0,0 @@
1
- /**
2
- * Resources module for the Originals SDK.
3
- *
4
- * This module provides resource management with immutable versioning,
5
- * content hashing, and validation.
6
- *
7
- * @module resources
8
- */
9
-
10
- export { ResourceManager } from './ResourceManager.js';
11
- export type {
12
- Resource,
13
- ResourceOptions,
14
- ResourceUpdateOptions,
15
- ResourceVersionHistory,
16
- ResourceManagerConfig,
17
- ResourceValidationResult,
18
- ResourceType,
19
- } from './types.js';
20
- export { MIME_TYPE_MAP, DEFAULT_RESOURCE_CONFIG } from './types.js';
21
-