@originals/sdk 1.2.0 → 1.4.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (244) hide show
  1. package/package.json +1 -1
  2. package/src/did/DIDManager.ts +1 -1
  3. package/src/did/WebVHManager.ts +11 -2
  4. package/src/examples/create-module-original.ts +435 -0
  5. package/src/examples/full-lifecycle-flow.ts +514 -0
  6. package/src/examples/run.ts +59 -4
  7. package/src/index.ts +69 -3
  8. package/src/kinds/KindRegistry.ts +290 -0
  9. package/src/kinds/index.ts +74 -0
  10. package/src/kinds/types.ts +470 -0
  11. package/src/kinds/validators/AgentValidator.ts +257 -0
  12. package/src/kinds/validators/AppValidator.ts +211 -0
  13. package/src/kinds/validators/DatasetValidator.ts +242 -0
  14. package/src/kinds/validators/DocumentValidator.ts +311 -0
  15. package/src/kinds/validators/MediaValidator.ts +269 -0
  16. package/src/kinds/validators/ModuleValidator.ts +225 -0
  17. package/src/kinds/validators/base.ts +276 -0
  18. package/src/kinds/validators/index.ts +12 -0
  19. package/src/lifecycle/LifecycleManager.ts +909 -1
  20. package/src/resources/ResourceManager.ts +655 -0
  21. package/src/resources/index.ts +21 -0
  22. package/src/resources/types.ts +202 -0
  23. package/src/types/common.ts +1 -1
  24. package/src/vc/CredentialManager.ts +647 -2
  25. package/tests/integration/createTypedOriginal.test.ts +379 -0
  26. package/tests/performance/BatchOperations.perf.test.ts +2 -2
  27. package/tests/unit/kinds/KindRegistry.test.ts +329 -0
  28. package/tests/unit/kinds/types.test.ts +409 -0
  29. package/tests/unit/kinds/validators.test.ts +651 -0
  30. package/tests/unit/lifecycle/LifecycleManager.cleanapi.test.ts +441 -0
  31. package/tests/unit/resources/ResourceManager.test.ts +740 -0
  32. package/tests/unit/vc/CredentialManager.helpers.test.ts +527 -0
  33. package/.turbo/turbo-build.log +0 -1
  34. package/dist/adapters/FeeOracleMock.d.ts +0 -6
  35. package/dist/adapters/FeeOracleMock.js +0 -8
  36. package/dist/adapters/index.d.ts +0 -4
  37. package/dist/adapters/index.js +0 -4
  38. package/dist/adapters/providers/OrdHttpProvider.d.ts +0 -56
  39. package/dist/adapters/providers/OrdHttpProvider.js +0 -110
  40. package/dist/adapters/providers/OrdMockProvider.d.ts +0 -70
  41. package/dist/adapters/providers/OrdMockProvider.js +0 -75
  42. package/dist/adapters/types.d.ts +0 -71
  43. package/dist/adapters/types.js +0 -1
  44. package/dist/bitcoin/BitcoinManager.d.ts +0 -15
  45. package/dist/bitcoin/BitcoinManager.js +0 -262
  46. package/dist/bitcoin/BroadcastClient.d.ts +0 -30
  47. package/dist/bitcoin/BroadcastClient.js +0 -35
  48. package/dist/bitcoin/OrdinalsClient.d.ts +0 -21
  49. package/dist/bitcoin/OrdinalsClient.js +0 -105
  50. package/dist/bitcoin/PSBTBuilder.d.ts +0 -24
  51. package/dist/bitcoin/PSBTBuilder.js +0 -80
  52. package/dist/bitcoin/fee-calculation.d.ts +0 -14
  53. package/dist/bitcoin/fee-calculation.js +0 -31
  54. package/dist/bitcoin/providers/OrdNodeProvider.d.ts +0 -38
  55. package/dist/bitcoin/providers/OrdNodeProvider.js +0 -67
  56. package/dist/bitcoin/providers/OrdinalsProvider.d.ts +0 -33
  57. package/dist/bitcoin/providers/OrdinalsProvider.js +0 -50
  58. package/dist/bitcoin/providers/types.d.ts +0 -63
  59. package/dist/bitcoin/providers/types.js +0 -1
  60. package/dist/bitcoin/transactions/commit.d.ts +0 -89
  61. package/dist/bitcoin/transactions/commit.js +0 -311
  62. package/dist/bitcoin/transactions/index.d.ts +0 -7
  63. package/dist/bitcoin/transactions/index.js +0 -8
  64. package/dist/bitcoin/transfer.d.ts +0 -9
  65. package/dist/bitcoin/transfer.js +0 -26
  66. package/dist/bitcoin/utxo-selection.d.ts +0 -78
  67. package/dist/bitcoin/utxo-selection.js +0 -237
  68. package/dist/bitcoin/utxo.d.ts +0 -26
  69. package/dist/bitcoin/utxo.js +0 -78
  70. package/dist/contexts/credentials-v1.json +0 -195
  71. package/dist/contexts/credentials-v2-examples.json +0 -5
  72. package/dist/contexts/credentials-v2.json +0 -301
  73. package/dist/contexts/credentials.json +0 -195
  74. package/dist/contexts/data-integrity-v2.json +0 -81
  75. package/dist/contexts/dids.json +0 -57
  76. package/dist/contexts/ed255192020.json +0 -93
  77. package/dist/contexts/ordinals-plus.json +0 -23
  78. package/dist/contexts/originals.json +0 -22
  79. package/dist/core/OriginalsSDK.d.ts +0 -158
  80. package/dist/core/OriginalsSDK.js +0 -274
  81. package/dist/crypto/Multikey.d.ts +0 -30
  82. package/dist/crypto/Multikey.js +0 -149
  83. package/dist/crypto/Signer.d.ts +0 -21
  84. package/dist/crypto/Signer.js +0 -196
  85. package/dist/crypto/noble-init.d.ts +0 -18
  86. package/dist/crypto/noble-init.js +0 -106
  87. package/dist/did/BtcoDidResolver.d.ts +0 -57
  88. package/dist/did/BtcoDidResolver.js +0 -166
  89. package/dist/did/DIDManager.d.ts +0 -101
  90. package/dist/did/DIDManager.js +0 -493
  91. package/dist/did/Ed25519Verifier.d.ts +0 -30
  92. package/dist/did/Ed25519Verifier.js +0 -59
  93. package/dist/did/KeyManager.d.ts +0 -17
  94. package/dist/did/KeyManager.js +0 -207
  95. package/dist/did/WebVHManager.d.ts +0 -100
  96. package/dist/did/WebVHManager.js +0 -304
  97. package/dist/did/createBtcoDidDocument.d.ts +0 -10
  98. package/dist/did/createBtcoDidDocument.js +0 -42
  99. package/dist/did/providers/OrdinalsClientProviderAdapter.d.ts +0 -23
  100. package/dist/did/providers/OrdinalsClientProviderAdapter.js +0 -51
  101. package/dist/events/EventEmitter.d.ts +0 -115
  102. package/dist/events/EventEmitter.js +0 -198
  103. package/dist/events/index.d.ts +0 -7
  104. package/dist/events/index.js +0 -6
  105. package/dist/events/types.d.ts +0 -286
  106. package/dist/events/types.js +0 -9
  107. package/dist/examples/basic-usage.d.ts +0 -3
  108. package/dist/examples/basic-usage.js +0 -62
  109. package/dist/examples/run.d.ts +0 -1
  110. package/dist/examples/run.js +0 -4
  111. package/dist/index.d.ts +0 -39
  112. package/dist/index.js +0 -47
  113. package/dist/lifecycle/BatchOperations.d.ts +0 -147
  114. package/dist/lifecycle/BatchOperations.js +0 -251
  115. package/dist/lifecycle/LifecycleManager.d.ts +0 -116
  116. package/dist/lifecycle/LifecycleManager.js +0 -971
  117. package/dist/lifecycle/OriginalsAsset.d.ts +0 -164
  118. package/dist/lifecycle/OriginalsAsset.js +0 -380
  119. package/dist/lifecycle/ProvenanceQuery.d.ts +0 -126
  120. package/dist/lifecycle/ProvenanceQuery.js +0 -220
  121. package/dist/lifecycle/ResourceVersioning.d.ts +0 -73
  122. package/dist/lifecycle/ResourceVersioning.js +0 -127
  123. package/dist/migration/MigrationManager.d.ts +0 -86
  124. package/dist/migration/MigrationManager.js +0 -412
  125. package/dist/migration/audit/AuditLogger.d.ts +0 -51
  126. package/dist/migration/audit/AuditLogger.js +0 -156
  127. package/dist/migration/checkpoint/CheckpointManager.d.ts +0 -31
  128. package/dist/migration/checkpoint/CheckpointManager.js +0 -96
  129. package/dist/migration/checkpoint/CheckpointStorage.d.ts +0 -26
  130. package/dist/migration/checkpoint/CheckpointStorage.js +0 -89
  131. package/dist/migration/index.d.ts +0 -22
  132. package/dist/migration/index.js +0 -27
  133. package/dist/migration/operations/BaseMigration.d.ts +0 -48
  134. package/dist/migration/operations/BaseMigration.js +0 -83
  135. package/dist/migration/operations/PeerToBtcoMigration.d.ts +0 -25
  136. package/dist/migration/operations/PeerToBtcoMigration.js +0 -67
  137. package/dist/migration/operations/PeerToWebvhMigration.d.ts +0 -19
  138. package/dist/migration/operations/PeerToWebvhMigration.js +0 -46
  139. package/dist/migration/operations/WebvhToBtcoMigration.d.ts +0 -25
  140. package/dist/migration/operations/WebvhToBtcoMigration.js +0 -67
  141. package/dist/migration/rollback/RollbackManager.d.ts +0 -29
  142. package/dist/migration/rollback/RollbackManager.js +0 -146
  143. package/dist/migration/state/StateMachine.d.ts +0 -25
  144. package/dist/migration/state/StateMachine.js +0 -76
  145. package/dist/migration/state/StateTracker.d.ts +0 -36
  146. package/dist/migration/state/StateTracker.js +0 -123
  147. package/dist/migration/types.d.ts +0 -306
  148. package/dist/migration/types.js +0 -33
  149. package/dist/migration/validation/BitcoinValidator.d.ts +0 -13
  150. package/dist/migration/validation/BitcoinValidator.js +0 -83
  151. package/dist/migration/validation/CredentialValidator.d.ts +0 -13
  152. package/dist/migration/validation/CredentialValidator.js +0 -46
  153. package/dist/migration/validation/DIDCompatibilityValidator.d.ts +0 -16
  154. package/dist/migration/validation/DIDCompatibilityValidator.js +0 -127
  155. package/dist/migration/validation/LifecycleValidator.d.ts +0 -10
  156. package/dist/migration/validation/LifecycleValidator.js +0 -52
  157. package/dist/migration/validation/StorageValidator.d.ts +0 -10
  158. package/dist/migration/validation/StorageValidator.js +0 -65
  159. package/dist/migration/validation/ValidationPipeline.d.ts +0 -29
  160. package/dist/migration/validation/ValidationPipeline.js +0 -180
  161. package/dist/storage/LocalStorageAdapter.d.ts +0 -11
  162. package/dist/storage/LocalStorageAdapter.js +0 -53
  163. package/dist/storage/MemoryStorageAdapter.d.ts +0 -6
  164. package/dist/storage/MemoryStorageAdapter.js +0 -21
  165. package/dist/storage/StorageAdapter.d.ts +0 -16
  166. package/dist/storage/StorageAdapter.js +0 -1
  167. package/dist/storage/index.d.ts +0 -2
  168. package/dist/storage/index.js +0 -2
  169. package/dist/types/bitcoin.d.ts +0 -84
  170. package/dist/types/bitcoin.js +0 -1
  171. package/dist/types/common.d.ts +0 -82
  172. package/dist/types/common.js +0 -1
  173. package/dist/types/credentials.d.ts +0 -75
  174. package/dist/types/credentials.js +0 -1
  175. package/dist/types/did.d.ts +0 -26
  176. package/dist/types/did.js +0 -1
  177. package/dist/types/index.d.ts +0 -5
  178. package/dist/types/index.js +0 -5
  179. package/dist/types/network.d.ts +0 -78
  180. package/dist/types/network.js +0 -145
  181. package/dist/utils/EventLogger.d.ts +0 -71
  182. package/dist/utils/EventLogger.js +0 -232
  183. package/dist/utils/Logger.d.ts +0 -106
  184. package/dist/utils/Logger.js +0 -257
  185. package/dist/utils/MetricsCollector.d.ts +0 -110
  186. package/dist/utils/MetricsCollector.js +0 -264
  187. package/dist/utils/bitcoin-address.d.ts +0 -38
  188. package/dist/utils/bitcoin-address.js +0 -113
  189. package/dist/utils/cbor.d.ts +0 -2
  190. package/dist/utils/cbor.js +0 -9
  191. package/dist/utils/encoding.d.ts +0 -37
  192. package/dist/utils/encoding.js +0 -120
  193. package/dist/utils/hash.d.ts +0 -1
  194. package/dist/utils/hash.js +0 -5
  195. package/dist/utils/retry.d.ts +0 -10
  196. package/dist/utils/retry.js +0 -35
  197. package/dist/utils/satoshi-validation.d.ts +0 -60
  198. package/dist/utils/satoshi-validation.js +0 -156
  199. package/dist/utils/serialization.d.ts +0 -14
  200. package/dist/utils/serialization.js +0 -76
  201. package/dist/utils/telemetry.d.ts +0 -17
  202. package/dist/utils/telemetry.js +0 -24
  203. package/dist/utils/validation.d.ts +0 -5
  204. package/dist/utils/validation.js +0 -98
  205. package/dist/vc/CredentialManager.d.ts +0 -22
  206. package/dist/vc/CredentialManager.js +0 -227
  207. package/dist/vc/Issuer.d.ts +0 -27
  208. package/dist/vc/Issuer.js +0 -70
  209. package/dist/vc/Verifier.d.ts +0 -16
  210. package/dist/vc/Verifier.js +0 -50
  211. package/dist/vc/cryptosuites/bbs.d.ts +0 -44
  212. package/dist/vc/cryptosuites/bbs.js +0 -213
  213. package/dist/vc/cryptosuites/bbsSimple.d.ts +0 -9
  214. package/dist/vc/cryptosuites/bbsSimple.js +0 -12
  215. package/dist/vc/cryptosuites/eddsa.d.ts +0 -30
  216. package/dist/vc/cryptosuites/eddsa.js +0 -81
  217. package/dist/vc/documentLoader.d.ts +0 -16
  218. package/dist/vc/documentLoader.js +0 -59
  219. package/dist/vc/proofs/data-integrity.d.ts +0 -21
  220. package/dist/vc/proofs/data-integrity.js +0 -15
  221. package/dist/vc/utils/jsonld.d.ts +0 -2
  222. package/dist/vc/utils/jsonld.js +0 -15
  223. package/test/logs/did_webvh_QmNTn9Kkp8dQ75WrF9xqJ2kuDp9QhKc3aPiERRMj8XoTBN_example_com.jsonl +0 -1
  224. package/test/logs/did_webvh_QmNu4MNr8Lr5txx5gYNhuhZDchXsZEu3hJXKYuphpWTPDp_example_com_users_etc_passwd.jsonl +0 -1
  225. package/test/logs/did_webvh_QmR9MrGZACzjKETA8SBRNCKG11HxU85c4bVR2qN5eDCfsD_example_com.jsonl +0 -1
  226. package/test/logs/did_webvh_QmUc5suaqRM2P4nrXxZwqYMfqzhdMqjuL7oJaJbEpCQVCd_example_com_users_etc_passwd.jsonl +0 -1
  227. package/test/logs/did_webvh_QmUkiB2RCV2VZ1RTXsCebWN25Eiy9TLvpzDWAJNjhgvB4X_example_com_etc_passwd.jsonl +0 -1
  228. package/test/logs/did_webvh_QmUoRTe8UMwpAQXZSAW7pjAgZK1tq2X3C6Kfxq3UXGcaGy_example_com_secret.jsonl +0 -1
  229. package/test/logs/did_webvh_QmWWot3chx1t6KwTmcE5i2FeDZ5JMkQw3qXycsKDVmJ9Be_example_com_users_alice.jsonl +0 -1
  230. package/test/logs/did_webvh_QmWvVgALL5kjZdpgR7KZay7J8UiiUr834kkRmWeFAxjAuC_example_com_users_etc_passwd.jsonl +0 -1
  231. package/test/logs/did_webvh_QmWwaRQHUZAFcKihFC6xR6tRTTrQhHPTku6azf1egWbpy1_example_com_users_alice.jsonl +0 -1
  232. package/test/logs/did_webvh_QmXJLtkz23r7AozbtXsZMKWnVU6rd38CkVtjdWuATU3Yp6_example_com_users_alice123_profile.jsonl +0 -1
  233. package/test/logs/did_webvh_QmYsce448po14oDE1wXbyaP6wY9HQgHSKLwdezn1k577SF_example_com_my_org_user_name_test_123.jsonl +0 -1
  234. package/test/logs/did_webvh_QmZBeNzzqajxdfwcDUPZ4P8C5YSXyRztrAwmPiKuKUxmAK_example_com.jsonl +0 -1
  235. package/test/logs/did_webvh_QmZhJsqxizwVbRtqCUkmE6XQunSxtxMt3gbTYadVBNAaEq_example_com.jsonl +0 -1
  236. package/test/logs/did_webvh_QmZk7NHU2D57RzzbMq4tWW9gBa9AqtVTWfiRM6RFdwGVj2_example_com.jsonl +0 -1
  237. package/test/logs/did_webvh_QmZshSXp9w8ovH62zGGBS1b5pGGPsuYiu1VQ935sga2hWF_example_com_level1_level2.jsonl +0 -1
  238. package/test/logs/did_webvh_QmbWAmw7HQL7vKJyCsctZihXf1rmT4sGvggKCPKWcUWjw1_example_com.jsonl +0 -1
  239. package/test/logs/did_webvh_QmbdLUMbYs3juR39TLB6hhrFWLcNg45ybUzeBJCS1MhCh1_example_com_C_Windows_System32.jsonl +0 -1
  240. package/test/logs/did_webvh_QmcaQ1Ma4gkSbae85aCm8Mv4rvdT2Sb2RR3JzYwrm5XBq8_example_com_etc_passwd.jsonl +0 -1
  241. package/test/logs/did_webvh_QmcbA7WQhsBqZSoDpKJHjV8Q5o53h8vmgJhQfo6rqTY5ho_example_com.jsonl +0 -1
  242. package/test/logs/did_webvh_Qmdy8uWr2gkUJrXsThynAug3DASTWwb3onEj89LKmMGZYB_example_com.jsonl +0 -1
  243. package/tests/e2e/README.md +0 -97
  244. package/tests/e2e/example.spec.ts +0 -78
@@ -0,0 +1,655 @@
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
+
@@ -0,0 +1,21 @@
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
+