@originals/sdk 1.8.2 → 1.8.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 (144) hide show
  1. package/package.json +5 -6
  2. package/src/adapters/FeeOracleMock.ts +0 -9
  3. package/src/adapters/index.ts +0 -5
  4. package/src/adapters/providers/OrdHttpProvider.ts +0 -126
  5. package/src/adapters/providers/OrdMockProvider.ts +0 -101
  6. package/src/adapters/types.ts +0 -66
  7. package/src/bitcoin/BitcoinManager.ts +0 -329
  8. package/src/bitcoin/BroadcastClient.ts +0 -54
  9. package/src/bitcoin/OrdinalsClient.ts +0 -120
  10. package/src/bitcoin/PSBTBuilder.ts +0 -106
  11. package/src/bitcoin/fee-calculation.ts +0 -38
  12. package/src/bitcoin/providers/OrdNodeProvider.ts +0 -92
  13. package/src/bitcoin/providers/OrdinalsProvider.ts +0 -56
  14. package/src/bitcoin/providers/types.ts +0 -59
  15. package/src/bitcoin/transactions/commit.ts +0 -465
  16. package/src/bitcoin/transactions/index.ts +0 -13
  17. package/src/bitcoin/transfer.ts +0 -43
  18. package/src/bitcoin/utxo-selection.ts +0 -322
  19. package/src/bitcoin/utxo.ts +0 -113
  20. package/src/cel/ExternalReferenceManager.ts +0 -87
  21. package/src/cel/OriginalsCel.ts +0 -460
  22. package/src/cel/algorithms/createEventLog.ts +0 -68
  23. package/src/cel/algorithms/deactivateEventLog.ts +0 -109
  24. package/src/cel/algorithms/index.ts +0 -11
  25. package/src/cel/algorithms/updateEventLog.ts +0 -99
  26. package/src/cel/algorithms/verifyEventLog.ts +0 -306
  27. package/src/cel/algorithms/witnessEvent.ts +0 -87
  28. package/src/cel/cli/create.ts +0 -330
  29. package/src/cel/cli/index.ts +0 -383
  30. package/src/cel/cli/inspect.ts +0 -549
  31. package/src/cel/cli/migrate.ts +0 -473
  32. package/src/cel/cli/verify.ts +0 -249
  33. package/src/cel/hash.ts +0 -71
  34. package/src/cel/index.ts +0 -16
  35. package/src/cel/layers/BtcoCelManager.ts +0 -408
  36. package/src/cel/layers/PeerCelManager.ts +0 -371
  37. package/src/cel/layers/WebVHCelManager.ts +0 -361
  38. package/src/cel/layers/index.ts +0 -27
  39. package/src/cel/serialization/cbor.ts +0 -189
  40. package/src/cel/serialization/index.ts +0 -10
  41. package/src/cel/serialization/json.ts +0 -209
  42. package/src/cel/types.ts +0 -160
  43. package/src/cel/witnesses/BitcoinWitness.ts +0 -184
  44. package/src/cel/witnesses/HttpWitness.ts +0 -241
  45. package/src/cel/witnesses/WitnessService.ts +0 -51
  46. package/src/cel/witnesses/index.ts +0 -11
  47. package/src/contexts/credentials-v1.json +0 -237
  48. package/src/contexts/credentials-v2-examples.json +0 -5
  49. package/src/contexts/credentials-v2.json +0 -340
  50. package/src/contexts/credentials.json +0 -237
  51. package/src/contexts/data-integrity-v2.json +0 -81
  52. package/src/contexts/dids.json +0 -58
  53. package/src/contexts/ed255192020.json +0 -93
  54. package/src/contexts/ordinals-plus.json +0 -23
  55. package/src/contexts/originals.json +0 -22
  56. package/src/core/OriginalsSDK.ts +0 -420
  57. package/src/crypto/Multikey.ts +0 -194
  58. package/src/crypto/Signer.ts +0 -262
  59. package/src/crypto/noble-init.ts +0 -138
  60. package/src/did/BtcoDidResolver.ts +0 -231
  61. package/src/did/DIDManager.ts +0 -705
  62. package/src/did/Ed25519Verifier.ts +0 -68
  63. package/src/did/KeyManager.ts +0 -239
  64. package/src/did/WebVHManager.ts +0 -499
  65. package/src/did/createBtcoDidDocument.ts +0 -60
  66. package/src/did/providers/OrdinalsClientProviderAdapter.ts +0 -68
  67. package/src/events/EventEmitter.ts +0 -222
  68. package/src/events/index.ts +0 -19
  69. package/src/events/types.ts +0 -331
  70. package/src/examples/basic-usage.ts +0 -78
  71. package/src/examples/create-module-original.ts +0 -435
  72. package/src/examples/full-lifecycle-flow.ts +0 -514
  73. package/src/examples/run.ts +0 -60
  74. package/src/index.ts +0 -204
  75. package/src/kinds/KindRegistry.ts +0 -320
  76. package/src/kinds/index.ts +0 -74
  77. package/src/kinds/types.ts +0 -470
  78. package/src/kinds/validators/AgentValidator.ts +0 -257
  79. package/src/kinds/validators/AppValidator.ts +0 -211
  80. package/src/kinds/validators/DatasetValidator.ts +0 -242
  81. package/src/kinds/validators/DocumentValidator.ts +0 -311
  82. package/src/kinds/validators/MediaValidator.ts +0 -269
  83. package/src/kinds/validators/ModuleValidator.ts +0 -225
  84. package/src/kinds/validators/base.ts +0 -276
  85. package/src/kinds/validators/index.ts +0 -12
  86. package/src/lifecycle/BatchOperations.ts +0 -381
  87. package/src/lifecycle/LifecycleManager.ts +0 -2156
  88. package/src/lifecycle/OriginalsAsset.ts +0 -524
  89. package/src/lifecycle/ProvenanceQuery.ts +0 -280
  90. package/src/lifecycle/ResourceVersioning.ts +0 -163
  91. package/src/migration/MigrationManager.ts +0 -587
  92. package/src/migration/audit/AuditLogger.ts +0 -176
  93. package/src/migration/checkpoint/CheckpointManager.ts +0 -112
  94. package/src/migration/checkpoint/CheckpointStorage.ts +0 -101
  95. package/src/migration/index.ts +0 -33
  96. package/src/migration/operations/BaseMigration.ts +0 -126
  97. package/src/migration/operations/PeerToBtcoMigration.ts +0 -105
  98. package/src/migration/operations/PeerToWebvhMigration.ts +0 -62
  99. package/src/migration/operations/WebvhToBtcoMigration.ts +0 -105
  100. package/src/migration/rollback/RollbackManager.ts +0 -170
  101. package/src/migration/state/StateMachine.ts +0 -92
  102. package/src/migration/state/StateTracker.ts +0 -156
  103. package/src/migration/types.ts +0 -356
  104. package/src/migration/validation/BitcoinValidator.ts +0 -107
  105. package/src/migration/validation/CredentialValidator.ts +0 -62
  106. package/src/migration/validation/DIDCompatibilityValidator.ts +0 -151
  107. package/src/migration/validation/LifecycleValidator.ts +0 -64
  108. package/src/migration/validation/StorageValidator.ts +0 -79
  109. package/src/migration/validation/ValidationPipeline.ts +0 -213
  110. package/src/resources/ResourceManager.ts +0 -655
  111. package/src/resources/index.ts +0 -21
  112. package/src/resources/types.ts +0 -202
  113. package/src/storage/LocalStorageAdapter.ts +0 -64
  114. package/src/storage/MemoryStorageAdapter.ts +0 -29
  115. package/src/storage/StorageAdapter.ts +0 -25
  116. package/src/storage/index.ts +0 -3
  117. package/src/types/bitcoin.ts +0 -98
  118. package/src/types/common.ts +0 -92
  119. package/src/types/credentials.ts +0 -89
  120. package/src/types/did.ts +0 -31
  121. package/src/types/external-shims.d.ts +0 -53
  122. package/src/types/index.ts +0 -7
  123. package/src/types/network.ts +0 -178
  124. package/src/utils/EventLogger.ts +0 -298
  125. package/src/utils/Logger.ts +0 -324
  126. package/src/utils/MetricsCollector.ts +0 -358
  127. package/src/utils/bitcoin-address.ts +0 -132
  128. package/src/utils/cbor.ts +0 -31
  129. package/src/utils/encoding.ts +0 -135
  130. package/src/utils/hash.ts +0 -12
  131. package/src/utils/retry.ts +0 -46
  132. package/src/utils/satoshi-validation.ts +0 -196
  133. package/src/utils/serialization.ts +0 -102
  134. package/src/utils/telemetry.ts +0 -44
  135. package/src/utils/validation.ts +0 -123
  136. package/src/vc/CredentialManager.ts +0 -955
  137. package/src/vc/Issuer.ts +0 -105
  138. package/src/vc/Verifier.ts +0 -54
  139. package/src/vc/cryptosuites/bbs.ts +0 -253
  140. package/src/vc/cryptosuites/bbsSimple.ts +0 -21
  141. package/src/vc/cryptosuites/eddsa.ts +0 -99
  142. package/src/vc/documentLoader.ts +0 -81
  143. package/src/vc/proofs/data-integrity.ts +0 -33
  144. package/src/vc/utils/jsonld.ts +0 -18
@@ -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
-