@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,225 @@
1
+ /**
2
+ * Module Kind Validator
3
+ *
4
+ * Validates manifests for reusable code modules with exports and dependencies.
5
+ */
6
+
7
+ import { OriginalKind, type OriginalManifest, type ValidationResult, type ModuleMetadata } from '../types';
8
+ import { BaseKindValidator, ValidationUtils } from './base';
9
+
10
+ /**
11
+ * Valid module formats
12
+ */
13
+ const VALID_FORMATS = ['esm', 'commonjs', 'umd', 'amd', 'iife'];
14
+
15
+ /**
16
+ * Validator for Module Originals
17
+ */
18
+ export class ModuleValidator extends BaseKindValidator<OriginalKind.Module> {
19
+ readonly kind = OriginalKind.Module;
20
+
21
+ protected validateKind(manifest: OriginalManifest<OriginalKind.Module>): ValidationResult {
22
+ const errors: ValidationResult['errors'] = [];
23
+ const warnings: ValidationResult['warnings'] = [];
24
+ const metadata = manifest.metadata as ModuleMetadata;
25
+
26
+ // Validate metadata exists
27
+ if (!metadata || typeof metadata !== 'object') {
28
+ return ValidationUtils.failure([
29
+ ValidationUtils.error('MISSING_METADATA', 'Module manifest must have metadata', 'metadata'),
30
+ ]);
31
+ }
32
+
33
+ // Validate format (required)
34
+ if (!metadata.format) {
35
+ errors.push(ValidationUtils.error(
36
+ 'MISSING_FORMAT',
37
+ 'Module must specify a format',
38
+ 'metadata.format',
39
+ ));
40
+ } else if (!VALID_FORMATS.includes(metadata.format)) {
41
+ errors.push(ValidationUtils.error(
42
+ 'INVALID_FORMAT',
43
+ `Invalid module format: "${metadata.format}". Must be one of: ${VALID_FORMATS.join(', ')}`,
44
+ 'metadata.format',
45
+ metadata.format,
46
+ ));
47
+ }
48
+
49
+ // Validate main (required)
50
+ if (!metadata.main || typeof metadata.main !== 'string') {
51
+ errors.push(ValidationUtils.error(
52
+ 'MISSING_MAIN',
53
+ 'Module must specify a main entrypoint',
54
+ 'metadata.main',
55
+ ));
56
+ } else {
57
+ // Check if main references an existing resource
58
+ if (!ValidationUtils.resourceExists(metadata.main, manifest.resources)) {
59
+ warnings.push(ValidationUtils.warning(
60
+ 'MAIN_NOT_RESOURCE',
61
+ `Main entrypoint "${metadata.main}" does not match a resource ID`,
62
+ 'metadata.main',
63
+ 'Ensure the main entrypoint is a valid resource ID',
64
+ ));
65
+ }
66
+ }
67
+
68
+ // Validate types if specified
69
+ if (metadata.types) {
70
+ if (typeof metadata.types !== 'string') {
71
+ errors.push(ValidationUtils.error(
72
+ 'INVALID_TYPES',
73
+ 'Types must be a string (resource ID)',
74
+ 'metadata.types',
75
+ ));
76
+ } else if (!ValidationUtils.resourceExists(metadata.types, manifest.resources)) {
77
+ warnings.push(ValidationUtils.warning(
78
+ 'TYPES_NOT_RESOURCE',
79
+ `Types file "${metadata.types}" does not match a resource ID`,
80
+ 'metadata.types',
81
+ ));
82
+ }
83
+ } else {
84
+ warnings.push(ValidationUtils.warning(
85
+ 'MISSING_TYPES',
86
+ 'Consider adding TypeScript type definitions',
87
+ 'metadata.types',
88
+ 'Add a .d.ts file for better TypeScript support',
89
+ ));
90
+ }
91
+
92
+ // Validate exports if specified
93
+ if (metadata.exports) {
94
+ if (typeof metadata.exports !== 'object' || Array.isArray(metadata.exports)) {
95
+ errors.push(ValidationUtils.error(
96
+ 'INVALID_EXPORTS',
97
+ 'Exports must be an object',
98
+ 'metadata.exports',
99
+ ));
100
+ } else {
101
+ for (const [key, value] of Object.entries(metadata.exports)) {
102
+ if (typeof value === 'string') {
103
+ // Simple string export is valid
104
+ continue;
105
+ } else if (typeof value === 'object' && value !== null) {
106
+ // Conditional exports object
107
+ const exportObj = value as { import?: string; require?: string; types?: string };
108
+ if (!exportObj.import && !exportObj.require) {
109
+ warnings.push(ValidationUtils.warning(
110
+ 'INCOMPLETE_EXPORT',
111
+ `Export "${key}" should have at least 'import' or 'require' field`,
112
+ `metadata.exports.${key}`,
113
+ ));
114
+ }
115
+ } else {
116
+ errors.push(ValidationUtils.error(
117
+ 'INVALID_EXPORT',
118
+ `Export "${key}" must be a string or object`,
119
+ `metadata.exports.${key}`,
120
+ ));
121
+ }
122
+ }
123
+
124
+ // Check for "." export (main export)
125
+ if (!('.' in metadata.exports)) {
126
+ warnings.push(ValidationUtils.warning(
127
+ 'MISSING_MAIN_EXPORT',
128
+ 'Consider adding a "." export for the main module entry',
129
+ 'metadata.exports',
130
+ ));
131
+ }
132
+ }
133
+ }
134
+
135
+ // Validate peer dependencies if specified
136
+ if (metadata.peerDependencies) {
137
+ if (typeof metadata.peerDependencies !== 'object' || Array.isArray(metadata.peerDependencies)) {
138
+ errors.push(ValidationUtils.error(
139
+ 'INVALID_PEER_DEPS',
140
+ 'Peer dependencies must be an object',
141
+ 'metadata.peerDependencies',
142
+ ));
143
+ } else {
144
+ for (const [name, version] of Object.entries(metadata.peerDependencies)) {
145
+ if (typeof version !== 'string') {
146
+ errors.push(ValidationUtils.error(
147
+ 'INVALID_PEER_DEP_VERSION',
148
+ `Peer dependency "${name}" must have a string version`,
149
+ `metadata.peerDependencies.${name}`,
150
+ ));
151
+ }
152
+ }
153
+ }
154
+ }
155
+
156
+ // Validate browser field if specified
157
+ if (metadata.browser) {
158
+ if (typeof metadata.browser !== 'string') {
159
+ errors.push(ValidationUtils.error(
160
+ 'INVALID_BROWSER',
161
+ 'Browser field must be a string (resource ID)',
162
+ 'metadata.browser',
163
+ ));
164
+ } else if (!ValidationUtils.resourceExists(metadata.browser, manifest.resources)) {
165
+ warnings.push(ValidationUtils.warning(
166
+ 'BROWSER_NOT_RESOURCE',
167
+ `Browser entrypoint "${metadata.browser}" does not match a resource ID`,
168
+ 'metadata.browser',
169
+ ));
170
+ }
171
+ }
172
+
173
+ // Validate sideEffects if specified
174
+ if (metadata.sideEffects !== undefined) {
175
+ if (typeof metadata.sideEffects !== 'boolean' && !Array.isArray(metadata.sideEffects)) {
176
+ errors.push(ValidationUtils.error(
177
+ 'INVALID_SIDE_EFFECTS',
178
+ 'sideEffects must be a boolean or array of strings',
179
+ 'metadata.sideEffects',
180
+ ));
181
+ } else if (Array.isArray(metadata.sideEffects)) {
182
+ for (let i = 0; i < metadata.sideEffects.length; i++) {
183
+ if (typeof metadata.sideEffects[i] !== 'string') {
184
+ errors.push(ValidationUtils.error(
185
+ 'INVALID_SIDE_EFFECT_ENTRY',
186
+ `sideEffects entry at index ${i} must be a string`,
187
+ `metadata.sideEffects[${i}]`,
188
+ ));
189
+ }
190
+ }
191
+ }
192
+ }
193
+
194
+ // Validate TypeScript config if specified
195
+ if (metadata.typescript) {
196
+ if (typeof metadata.typescript !== 'object') {
197
+ errors.push(ValidationUtils.error(
198
+ 'INVALID_TYPESCRIPT',
199
+ 'TypeScript config must be an object',
200
+ 'metadata.typescript',
201
+ ));
202
+ }
203
+ }
204
+
205
+ // Check that at least one code resource exists
206
+ const codeResources = manifest.resources.filter(r =>
207
+ r.contentType.includes('javascript') ||
208
+ r.contentType.includes('typescript') ||
209
+ r.contentType.includes('json') ||
210
+ r.type === 'code'
211
+ );
212
+ if (codeResources.length === 0) {
213
+ warnings.push(ValidationUtils.warning(
214
+ 'NO_CODE_RESOURCES',
215
+ 'No code resources found. Ensure resources have appropriate content types',
216
+ 'resources',
217
+ ));
218
+ }
219
+
220
+ return errors.length > 0
221
+ ? ValidationUtils.failure(errors, warnings)
222
+ : ValidationUtils.success(warnings);
223
+ }
224
+ }
225
+
@@ -0,0 +1,276 @@
1
+ /**
2
+ * Base validator interface and common validation utilities
3
+ */
4
+
5
+ import type {
6
+ OriginalKind,
7
+ OriginalManifest,
8
+ ValidationResult,
9
+ ValidationError,
10
+ ValidationWarning,
11
+ BaseManifest,
12
+ } from '../types';
13
+ import type { AssetResource } from '../../types/common';
14
+
15
+ /**
16
+ * Interface for kind-specific validators
17
+ */
18
+ export interface KindValidator<K extends OriginalKind = OriginalKind> {
19
+ /** The kind this validator handles */
20
+ readonly kind: K;
21
+
22
+ /**
23
+ * Validate a manifest for this kind
24
+ * @param manifest - The manifest to validate
25
+ * @returns Validation result with errors and warnings
26
+ */
27
+ validate(manifest: OriginalManifest<K>): ValidationResult;
28
+ }
29
+
30
+ /**
31
+ * Common validation utilities
32
+ */
33
+ export class ValidationUtils {
34
+ /**
35
+ * Check if a string is a valid semantic version
36
+ */
37
+ static isValidSemver(version: string): boolean {
38
+ const semverRegex = /^(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$/;
39
+ return semverRegex.test(version);
40
+ }
41
+
42
+ /**
43
+ * Check if a string is a valid DID
44
+ */
45
+ static isValidDID(did: string): boolean {
46
+ // DID format: did:method:identifier (identifier can contain : for path segments)
47
+ // Examples: did:peer:123, did:webvh:example.com:user, did:btco:12345
48
+ const didRegex = /^did:[a-z0-9]+:[a-zA-Z0-9._:%-]+$/;
49
+ return didRegex.test(did);
50
+ }
51
+
52
+ /**
53
+ * Check if a string is a valid SPDX license identifier
54
+ */
55
+ static isValidSPDXLicense(license: string): boolean {
56
+ // Common SPDX identifiers (not exhaustive)
57
+ const commonLicenses = [
58
+ 'MIT', 'Apache-2.0', 'GPL-3.0', 'GPL-2.0', 'BSD-2-Clause', 'BSD-3-Clause',
59
+ 'ISC', 'MPL-2.0', 'LGPL-3.0', 'AGPL-3.0', 'Unlicense', 'CC0-1.0',
60
+ 'CC-BY-4.0', 'CC-BY-SA-4.0', 'CC-BY-NC-4.0', 'WTFPL', 'Zlib', 'BSL-1.0',
61
+ 'MIT-0', 'Apache-1.0', 'Apache-1.1', 'EPL-2.0', 'EUPL-1.2',
62
+ ];
63
+ return commonLicenses.includes(license) || /^[A-Z0-9][A-Z0-9._-]*$/i.test(license);
64
+ }
65
+
66
+ /**
67
+ * Check if a string is a valid MIME type
68
+ */
69
+ static isValidMimeType(mimeType: string): boolean {
70
+ const mimeRegex = /^[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}\/[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}$/;
71
+ return mimeRegex.test(mimeType);
72
+ }
73
+
74
+ /**
75
+ * Check if a string is a valid URL
76
+ */
77
+ static isValidURL(url: string): boolean {
78
+ try {
79
+ new URL(url);
80
+ return true;
81
+ } catch {
82
+ return false;
83
+ }
84
+ }
85
+
86
+ /**
87
+ * Check if a string is a valid ISO 639-1 language code
88
+ */
89
+ static isValidLanguageCode(code: string): boolean {
90
+ // ISO 639-1 is 2 lowercase letters
91
+ return /^[a-z]{2}$/.test(code);
92
+ }
93
+
94
+ /**
95
+ * Check if a resource ID exists in the resources array
96
+ */
97
+ static resourceExists(resourceId: string, resources: AssetResource[]): boolean {
98
+ return resources.some(r => r.id === resourceId);
99
+ }
100
+
101
+ /**
102
+ * Get a resource by ID
103
+ */
104
+ static getResource(resourceId: string, resources: AssetResource[]): AssetResource | undefined {
105
+ return resources.find(r => r.id === resourceId);
106
+ }
107
+
108
+ /**
109
+ * Create a validation error
110
+ */
111
+ static error(code: string, message: string, path?: string, value?: unknown): ValidationError {
112
+ return { code, message, path, value };
113
+ }
114
+
115
+ /**
116
+ * Create a validation warning
117
+ */
118
+ static warning(code: string, message: string, path?: string, suggestion?: string): ValidationWarning {
119
+ return { code, message, path, suggestion };
120
+ }
121
+
122
+ /**
123
+ * Create a successful validation result
124
+ */
125
+ static success(warnings: ValidationWarning[] = []): ValidationResult {
126
+ return { isValid: true, errors: [], warnings };
127
+ }
128
+
129
+ /**
130
+ * Create a failed validation result
131
+ */
132
+ static failure(errors: ValidationError[], warnings: ValidationWarning[] = []): ValidationResult {
133
+ return { isValid: false, errors, warnings };
134
+ }
135
+
136
+ /**
137
+ * Merge multiple validation results
138
+ */
139
+ static merge(...results: ValidationResult[]): ValidationResult {
140
+ const errors: ValidationError[] = [];
141
+ const warnings: ValidationWarning[] = [];
142
+
143
+ for (const result of results) {
144
+ errors.push(...result.errors);
145
+ warnings.push(...result.warnings);
146
+ }
147
+
148
+ return {
149
+ isValid: errors.length === 0,
150
+ errors,
151
+ warnings,
152
+ };
153
+ }
154
+ }
155
+
156
+ /**
157
+ * Base validator class with common validation logic
158
+ */
159
+ export abstract class BaseKindValidator<K extends OriginalKind> implements KindValidator<K> {
160
+ abstract readonly kind: K;
161
+
162
+ /**
163
+ * Validate a manifest
164
+ * Combines base validation with kind-specific validation
165
+ */
166
+ validate(manifest: OriginalManifest<K>): ValidationResult {
167
+ const baseResult = this.validateBase(manifest);
168
+ const kindResult = this.validateKind(manifest);
169
+
170
+ return ValidationUtils.merge(baseResult, kindResult);
171
+ }
172
+
173
+ /**
174
+ * Validate base manifest fields common to all kinds
175
+ */
176
+ protected validateBase(manifest: BaseManifest & { kind: K }): ValidationResult {
177
+ const errors: ValidationError[] = [];
178
+ const warnings: ValidationWarning[] = [];
179
+
180
+ // Validate kind
181
+ if (!manifest.kind) {
182
+ errors.push(ValidationUtils.error('MISSING_KIND', 'Manifest must specify a kind', 'kind'));
183
+ }
184
+
185
+ // Validate name
186
+ if (!manifest.name || typeof manifest.name !== 'string') {
187
+ errors.push(ValidationUtils.error('MISSING_NAME', 'Manifest must have a name', 'name'));
188
+ } else if (manifest.name.length < 1 || manifest.name.length > 200) {
189
+ errors.push(ValidationUtils.error('INVALID_NAME_LENGTH', 'Name must be between 1 and 200 characters', 'name', manifest.name));
190
+ }
191
+
192
+ // Validate version
193
+ if (!manifest.version || typeof manifest.version !== 'string') {
194
+ errors.push(ValidationUtils.error('MISSING_VERSION', 'Manifest must have a version', 'version'));
195
+ } else if (!ValidationUtils.isValidSemver(manifest.version)) {
196
+ errors.push(ValidationUtils.error('INVALID_VERSION', 'Version must be a valid semantic version (e.g., 1.0.0)', 'version', manifest.version));
197
+ }
198
+
199
+ // Validate resources
200
+ if (!manifest.resources || !Array.isArray(manifest.resources)) {
201
+ errors.push(ValidationUtils.error('MISSING_RESOURCES', 'Manifest must have resources array', 'resources'));
202
+ } else if (manifest.resources.length === 0) {
203
+ errors.push(ValidationUtils.error('EMPTY_RESOURCES', 'Manifest must have at least one resource', 'resources'));
204
+ } else {
205
+ // Validate each resource
206
+ for (let i = 0; i < manifest.resources.length; i++) {
207
+ const resource = manifest.resources[i];
208
+ const resourcePath = `resources[${i}]`;
209
+
210
+ if (!resource.id || typeof resource.id !== 'string') {
211
+ errors.push(ValidationUtils.error('INVALID_RESOURCE_ID', `Resource at index ${i} must have an id`, `${resourcePath}.id`));
212
+ }
213
+ if (!resource.type || typeof resource.type !== 'string') {
214
+ errors.push(ValidationUtils.error('INVALID_RESOURCE_TYPE', `Resource at index ${i} must have a type`, `${resourcePath}.type`));
215
+ }
216
+ if (!resource.contentType || !ValidationUtils.isValidMimeType(resource.contentType)) {
217
+ errors.push(ValidationUtils.error('INVALID_CONTENT_TYPE', `Resource at index ${i} must have a valid MIME contentType`, `${resourcePath}.contentType`, resource.contentType));
218
+ }
219
+ if (!resource.hash || typeof resource.hash !== 'string' || !/^[0-9a-fA-F]+$/.test(resource.hash)) {
220
+ errors.push(ValidationUtils.error('INVALID_RESOURCE_HASH', `Resource at index ${i} must have a valid hex hash`, `${resourcePath}.hash`));
221
+ }
222
+ }
223
+
224
+ // Check for duplicate resource IDs
225
+ const resourceIds = manifest.resources.map(r => r.id);
226
+ const duplicates = resourceIds.filter((id, index) => resourceIds.indexOf(id) !== index);
227
+ if (duplicates.length > 0) {
228
+ errors.push(ValidationUtils.error('DUPLICATE_RESOURCE_IDS', `Duplicate resource IDs found: ${[...new Set(duplicates)].join(', ')}`, 'resources'));
229
+ }
230
+ }
231
+
232
+ // Validate dependencies if present
233
+ if (manifest.dependencies) {
234
+ if (!Array.isArray(manifest.dependencies)) {
235
+ errors.push(ValidationUtils.error('INVALID_DEPENDENCIES', 'Dependencies must be an array', 'dependencies'));
236
+ } else {
237
+ for (let i = 0; i < manifest.dependencies.length; i++) {
238
+ const dep = manifest.dependencies[i];
239
+ const depPath = `dependencies[${i}]`;
240
+
241
+ if (!dep.did || !ValidationUtils.isValidDID(dep.did)) {
242
+ errors.push(ValidationUtils.error('INVALID_DEPENDENCY_DID', `Dependency at index ${i} must have a valid DID`, `${depPath}.did`, dep.did));
243
+ }
244
+ }
245
+ }
246
+ }
247
+
248
+ // Validate optional fields
249
+ if (manifest.license && !ValidationUtils.isValidSPDXLicense(manifest.license)) {
250
+ warnings.push(ValidationUtils.warning('UNKNOWN_LICENSE', `License "${manifest.license}" is not a recognized SPDX identifier`, 'license', 'Use a valid SPDX license identifier like MIT, Apache-2.0, etc.'));
251
+ }
252
+
253
+ if (manifest.homepage && !ValidationUtils.isValidURL(manifest.homepage)) {
254
+ errors.push(ValidationUtils.error('INVALID_HOMEPAGE', 'Homepage must be a valid URL', 'homepage', manifest.homepage));
255
+ }
256
+
257
+ if (manifest.repository && !ValidationUtils.isValidURL(manifest.repository)) {
258
+ errors.push(ValidationUtils.error('INVALID_REPOSITORY', 'Repository must be a valid URL', 'repository', manifest.repository));
259
+ }
260
+
261
+ // Suggest adding description if missing
262
+ if (!manifest.description) {
263
+ warnings.push(ValidationUtils.warning('MISSING_DESCRIPTION', 'Consider adding a description for better discoverability', 'description', 'Add a brief description of this Original'));
264
+ }
265
+
266
+ return errors.length > 0
267
+ ? ValidationUtils.failure(errors, warnings)
268
+ : ValidationUtils.success(warnings);
269
+ }
270
+
271
+ /**
272
+ * Kind-specific validation to be implemented by subclasses
273
+ */
274
+ protected abstract validateKind(manifest: OriginalManifest<K>): ValidationResult;
275
+ }
276
+
@@ -0,0 +1,12 @@
1
+ /**
2
+ * Kind validators exports
3
+ */
4
+
5
+ export { BaseKindValidator, ValidationUtils, type KindValidator } from './base';
6
+ export { AppValidator } from './AppValidator';
7
+ export { AgentValidator } from './AgentValidator';
8
+ export { ModuleValidator } from './ModuleValidator';
9
+ export { DatasetValidator } from './DatasetValidator';
10
+ export { MediaValidator } from './MediaValidator';
11
+ export { DocumentValidator } from './DocumentValidator';
12
+