@originals/sdk 1.1.0

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 (403) hide show
  1. package/.eslintrc.json +33 -0
  2. package/.turbo/turbo-build.log +1 -0
  3. package/.turbo/turbo-test.log +68353 -0
  4. package/dist/adapters/FeeOracleMock.d.ts +6 -0
  5. package/dist/adapters/FeeOracleMock.js +8 -0
  6. package/dist/adapters/index.d.ts +4 -0
  7. package/dist/adapters/index.js +4 -0
  8. package/dist/adapters/providers/OrdHttpProvider.d.ts +56 -0
  9. package/dist/adapters/providers/OrdHttpProvider.js +110 -0
  10. package/dist/adapters/providers/OrdMockProvider.d.ts +70 -0
  11. package/dist/adapters/providers/OrdMockProvider.js +75 -0
  12. package/dist/adapters/types.d.ts +71 -0
  13. package/dist/adapters/types.js +1 -0
  14. package/dist/bitcoin/BitcoinManager.d.ts +15 -0
  15. package/dist/bitcoin/BitcoinManager.js +262 -0
  16. package/dist/bitcoin/BroadcastClient.d.ts +30 -0
  17. package/dist/bitcoin/BroadcastClient.js +35 -0
  18. package/dist/bitcoin/OrdinalsClient.d.ts +21 -0
  19. package/dist/bitcoin/OrdinalsClient.js +105 -0
  20. package/dist/bitcoin/PSBTBuilder.d.ts +24 -0
  21. package/dist/bitcoin/PSBTBuilder.js +80 -0
  22. package/dist/bitcoin/fee-calculation.d.ts +14 -0
  23. package/dist/bitcoin/fee-calculation.js +31 -0
  24. package/dist/bitcoin/providers/OrdNodeProvider.d.ts +38 -0
  25. package/dist/bitcoin/providers/OrdNodeProvider.js +67 -0
  26. package/dist/bitcoin/providers/OrdinalsProvider.d.ts +33 -0
  27. package/dist/bitcoin/providers/OrdinalsProvider.js +50 -0
  28. package/dist/bitcoin/providers/types.d.ts +63 -0
  29. package/dist/bitcoin/providers/types.js +1 -0
  30. package/dist/bitcoin/transactions/commit.d.ts +89 -0
  31. package/dist/bitcoin/transactions/commit.js +311 -0
  32. package/dist/bitcoin/transactions/index.d.ts +7 -0
  33. package/dist/bitcoin/transactions/index.js +8 -0
  34. package/dist/bitcoin/transfer.d.ts +9 -0
  35. package/dist/bitcoin/transfer.js +26 -0
  36. package/dist/bitcoin/utxo-selection.d.ts +78 -0
  37. package/dist/bitcoin/utxo-selection.js +237 -0
  38. package/dist/bitcoin/utxo.d.ts +26 -0
  39. package/dist/bitcoin/utxo.js +78 -0
  40. package/dist/contexts/credentials-v1.json +195 -0
  41. package/dist/contexts/credentials-v2-examples.json +5 -0
  42. package/dist/contexts/credentials-v2.json +301 -0
  43. package/dist/contexts/credentials.json +195 -0
  44. package/dist/contexts/data-integrity-v2.json +81 -0
  45. package/dist/contexts/dids.json +57 -0
  46. package/dist/contexts/ed255192020.json +93 -0
  47. package/dist/contexts/ordinals-plus.json +23 -0
  48. package/dist/contexts/originals.json +22 -0
  49. package/dist/core/OriginalsSDK.d.ts +158 -0
  50. package/dist/core/OriginalsSDK.js +274 -0
  51. package/dist/crypto/Multikey.d.ts +30 -0
  52. package/dist/crypto/Multikey.js +149 -0
  53. package/dist/crypto/Signer.d.ts +21 -0
  54. package/dist/crypto/Signer.js +196 -0
  55. package/dist/crypto/noble-init.d.ts +18 -0
  56. package/dist/crypto/noble-init.js +106 -0
  57. package/dist/did/BtcoDidResolver.d.ts +57 -0
  58. package/dist/did/BtcoDidResolver.js +166 -0
  59. package/dist/did/DIDManager.d.ts +101 -0
  60. package/dist/did/DIDManager.js +493 -0
  61. package/dist/did/Ed25519Verifier.d.ts +30 -0
  62. package/dist/did/Ed25519Verifier.js +59 -0
  63. package/dist/did/KeyManager.d.ts +17 -0
  64. package/dist/did/KeyManager.js +207 -0
  65. package/dist/did/WebVHManager.d.ts +100 -0
  66. package/dist/did/WebVHManager.js +304 -0
  67. package/dist/did/createBtcoDidDocument.d.ts +10 -0
  68. package/dist/did/createBtcoDidDocument.js +42 -0
  69. package/dist/did/providers/OrdinalsClientProviderAdapter.d.ts +23 -0
  70. package/dist/did/providers/OrdinalsClientProviderAdapter.js +51 -0
  71. package/dist/events/EventEmitter.d.ts +115 -0
  72. package/dist/events/EventEmitter.js +198 -0
  73. package/dist/events/index.d.ts +7 -0
  74. package/dist/events/index.js +6 -0
  75. package/dist/events/types.d.ts +286 -0
  76. package/dist/events/types.js +9 -0
  77. package/dist/examples/basic-usage.d.ts +3 -0
  78. package/dist/examples/basic-usage.js +62 -0
  79. package/dist/examples/run.d.ts +1 -0
  80. package/dist/examples/run.js +4 -0
  81. package/dist/index.d.ts +39 -0
  82. package/dist/index.js +47 -0
  83. package/dist/lifecycle/BatchOperations.d.ts +147 -0
  84. package/dist/lifecycle/BatchOperations.js +251 -0
  85. package/dist/lifecycle/LifecycleManager.d.ts +116 -0
  86. package/dist/lifecycle/LifecycleManager.js +971 -0
  87. package/dist/lifecycle/OriginalsAsset.d.ts +164 -0
  88. package/dist/lifecycle/OriginalsAsset.js +380 -0
  89. package/dist/lifecycle/ProvenanceQuery.d.ts +126 -0
  90. package/dist/lifecycle/ProvenanceQuery.js +220 -0
  91. package/dist/lifecycle/ResourceVersioning.d.ts +73 -0
  92. package/dist/lifecycle/ResourceVersioning.js +127 -0
  93. package/dist/migration/MigrationManager.d.ts +86 -0
  94. package/dist/migration/MigrationManager.js +412 -0
  95. package/dist/migration/audit/AuditLogger.d.ts +51 -0
  96. package/dist/migration/audit/AuditLogger.js +156 -0
  97. package/dist/migration/checkpoint/CheckpointManager.d.ts +31 -0
  98. package/dist/migration/checkpoint/CheckpointManager.js +96 -0
  99. package/dist/migration/checkpoint/CheckpointStorage.d.ts +26 -0
  100. package/dist/migration/checkpoint/CheckpointStorage.js +89 -0
  101. package/dist/migration/index.d.ts +22 -0
  102. package/dist/migration/index.js +27 -0
  103. package/dist/migration/operations/BaseMigration.d.ts +48 -0
  104. package/dist/migration/operations/BaseMigration.js +83 -0
  105. package/dist/migration/operations/PeerToBtcoMigration.d.ts +25 -0
  106. package/dist/migration/operations/PeerToBtcoMigration.js +67 -0
  107. package/dist/migration/operations/PeerToWebvhMigration.d.ts +19 -0
  108. package/dist/migration/operations/PeerToWebvhMigration.js +46 -0
  109. package/dist/migration/operations/WebvhToBtcoMigration.d.ts +25 -0
  110. package/dist/migration/operations/WebvhToBtcoMigration.js +67 -0
  111. package/dist/migration/rollback/RollbackManager.d.ts +29 -0
  112. package/dist/migration/rollback/RollbackManager.js +146 -0
  113. package/dist/migration/state/StateMachine.d.ts +25 -0
  114. package/dist/migration/state/StateMachine.js +76 -0
  115. package/dist/migration/state/StateTracker.d.ts +36 -0
  116. package/dist/migration/state/StateTracker.js +123 -0
  117. package/dist/migration/types.d.ts +306 -0
  118. package/dist/migration/types.js +33 -0
  119. package/dist/migration/validation/BitcoinValidator.d.ts +13 -0
  120. package/dist/migration/validation/BitcoinValidator.js +83 -0
  121. package/dist/migration/validation/CredentialValidator.d.ts +13 -0
  122. package/dist/migration/validation/CredentialValidator.js +46 -0
  123. package/dist/migration/validation/DIDCompatibilityValidator.d.ts +16 -0
  124. package/dist/migration/validation/DIDCompatibilityValidator.js +127 -0
  125. package/dist/migration/validation/LifecycleValidator.d.ts +10 -0
  126. package/dist/migration/validation/LifecycleValidator.js +52 -0
  127. package/dist/migration/validation/StorageValidator.d.ts +10 -0
  128. package/dist/migration/validation/StorageValidator.js +65 -0
  129. package/dist/migration/validation/ValidationPipeline.d.ts +29 -0
  130. package/dist/migration/validation/ValidationPipeline.js +180 -0
  131. package/dist/storage/LocalStorageAdapter.d.ts +11 -0
  132. package/dist/storage/LocalStorageAdapter.js +53 -0
  133. package/dist/storage/MemoryStorageAdapter.d.ts +6 -0
  134. package/dist/storage/MemoryStorageAdapter.js +21 -0
  135. package/dist/storage/StorageAdapter.d.ts +16 -0
  136. package/dist/storage/StorageAdapter.js +1 -0
  137. package/dist/storage/index.d.ts +2 -0
  138. package/dist/storage/index.js +2 -0
  139. package/dist/types/bitcoin.d.ts +84 -0
  140. package/dist/types/bitcoin.js +1 -0
  141. package/dist/types/common.d.ts +82 -0
  142. package/dist/types/common.js +1 -0
  143. package/dist/types/credentials.d.ts +75 -0
  144. package/dist/types/credentials.js +1 -0
  145. package/dist/types/did.d.ts +26 -0
  146. package/dist/types/did.js +1 -0
  147. package/dist/types/index.d.ts +5 -0
  148. package/dist/types/index.js +5 -0
  149. package/dist/types/network.d.ts +78 -0
  150. package/dist/types/network.js +145 -0
  151. package/dist/utils/EventLogger.d.ts +71 -0
  152. package/dist/utils/EventLogger.js +232 -0
  153. package/dist/utils/Logger.d.ts +106 -0
  154. package/dist/utils/Logger.js +257 -0
  155. package/dist/utils/MetricsCollector.d.ts +110 -0
  156. package/dist/utils/MetricsCollector.js +264 -0
  157. package/dist/utils/bitcoin-address.d.ts +38 -0
  158. package/dist/utils/bitcoin-address.js +113 -0
  159. package/dist/utils/cbor.d.ts +2 -0
  160. package/dist/utils/cbor.js +9 -0
  161. package/dist/utils/encoding.d.ts +37 -0
  162. package/dist/utils/encoding.js +120 -0
  163. package/dist/utils/hash.d.ts +1 -0
  164. package/dist/utils/hash.js +5 -0
  165. package/dist/utils/retry.d.ts +10 -0
  166. package/dist/utils/retry.js +35 -0
  167. package/dist/utils/satoshi-validation.d.ts +60 -0
  168. package/dist/utils/satoshi-validation.js +156 -0
  169. package/dist/utils/serialization.d.ts +14 -0
  170. package/dist/utils/serialization.js +76 -0
  171. package/dist/utils/telemetry.d.ts +17 -0
  172. package/dist/utils/telemetry.js +24 -0
  173. package/dist/utils/validation.d.ts +5 -0
  174. package/dist/utils/validation.js +98 -0
  175. package/dist/vc/CredentialManager.d.ts +22 -0
  176. package/dist/vc/CredentialManager.js +227 -0
  177. package/dist/vc/Issuer.d.ts +27 -0
  178. package/dist/vc/Issuer.js +70 -0
  179. package/dist/vc/Verifier.d.ts +16 -0
  180. package/dist/vc/Verifier.js +50 -0
  181. package/dist/vc/cryptosuites/bbs.d.ts +44 -0
  182. package/dist/vc/cryptosuites/bbs.js +213 -0
  183. package/dist/vc/cryptosuites/bbsSimple.d.ts +9 -0
  184. package/dist/vc/cryptosuites/bbsSimple.js +12 -0
  185. package/dist/vc/cryptosuites/eddsa.d.ts +30 -0
  186. package/dist/vc/cryptosuites/eddsa.js +81 -0
  187. package/dist/vc/documentLoader.d.ts +16 -0
  188. package/dist/vc/documentLoader.js +59 -0
  189. package/dist/vc/proofs/data-integrity.d.ts +21 -0
  190. package/dist/vc/proofs/data-integrity.js +15 -0
  191. package/dist/vc/utils/jsonld.d.ts +2 -0
  192. package/dist/vc/utils/jsonld.js +15 -0
  193. package/package.json +79 -0
  194. package/src/adapters/FeeOracleMock.ts +9 -0
  195. package/src/adapters/index.ts +5 -0
  196. package/src/adapters/providers/OrdHttpProvider.ts +126 -0
  197. package/src/adapters/providers/OrdMockProvider.ts +101 -0
  198. package/src/adapters/types.ts +66 -0
  199. package/src/bitcoin/BitcoinManager.ts +330 -0
  200. package/src/bitcoin/BroadcastClient.ts +54 -0
  201. package/src/bitcoin/OrdinalsClient.ts +119 -0
  202. package/src/bitcoin/PSBTBuilder.ts +106 -0
  203. package/src/bitcoin/fee-calculation.ts +38 -0
  204. package/src/bitcoin/providers/OrdNodeProvider.ts +92 -0
  205. package/src/bitcoin/providers/OrdinalsProvider.ts +56 -0
  206. package/src/bitcoin/providers/types.ts +59 -0
  207. package/src/bitcoin/transactions/commit.ts +465 -0
  208. package/src/bitcoin/transactions/index.ts +13 -0
  209. package/src/bitcoin/transfer.ts +43 -0
  210. package/src/bitcoin/utxo-selection.ts +322 -0
  211. package/src/bitcoin/utxo.ts +113 -0
  212. package/src/contexts/credentials-v1.json +237 -0
  213. package/src/contexts/credentials-v2-examples.json +5 -0
  214. package/src/contexts/credentials-v2.json +340 -0
  215. package/src/contexts/credentials.json +237 -0
  216. package/src/contexts/data-integrity-v2.json +81 -0
  217. package/src/contexts/dids.json +58 -0
  218. package/src/contexts/ed255192020.json +93 -0
  219. package/src/contexts/ordinals-plus.json +23 -0
  220. package/src/contexts/originals.json +22 -0
  221. package/src/core/OriginalsSDK.ts +416 -0
  222. package/src/crypto/Multikey.ts +194 -0
  223. package/src/crypto/Signer.ts +254 -0
  224. package/src/crypto/noble-init.ts +121 -0
  225. package/src/did/BtcoDidResolver.ts +227 -0
  226. package/src/did/DIDManager.ts +694 -0
  227. package/src/did/Ed25519Verifier.ts +68 -0
  228. package/src/did/KeyManager.ts +236 -0
  229. package/src/did/WebVHManager.ts +489 -0
  230. package/src/did/createBtcoDidDocument.ts +59 -0
  231. package/src/did/providers/OrdinalsClientProviderAdapter.ts +68 -0
  232. package/src/events/EventEmitter.ts +222 -0
  233. package/src/events/index.ts +19 -0
  234. package/src/events/types.ts +331 -0
  235. package/src/examples/basic-usage.ts +78 -0
  236. package/src/examples/run.ts +5 -0
  237. package/src/index.ts +84 -0
  238. package/src/lifecycle/BatchOperations.ts +373 -0
  239. package/src/lifecycle/LifecycleManager.ts +1218 -0
  240. package/src/lifecycle/OriginalsAsset.ts +524 -0
  241. package/src/lifecycle/ProvenanceQuery.ts +280 -0
  242. package/src/lifecycle/ResourceVersioning.ts +163 -0
  243. package/src/migration/MigrationManager.ts +527 -0
  244. package/src/migration/audit/AuditLogger.ts +176 -0
  245. package/src/migration/checkpoint/CheckpointManager.ts +112 -0
  246. package/src/migration/checkpoint/CheckpointStorage.ts +101 -0
  247. package/src/migration/index.ts +33 -0
  248. package/src/migration/operations/BaseMigration.ts +126 -0
  249. package/src/migration/operations/PeerToBtcoMigration.ts +105 -0
  250. package/src/migration/operations/PeerToWebvhMigration.ts +62 -0
  251. package/src/migration/operations/WebvhToBtcoMigration.ts +105 -0
  252. package/src/migration/rollback/RollbackManager.ts +170 -0
  253. package/src/migration/state/StateMachine.ts +92 -0
  254. package/src/migration/state/StateTracker.ts +156 -0
  255. package/src/migration/types.ts +344 -0
  256. package/src/migration/validation/BitcoinValidator.ts +107 -0
  257. package/src/migration/validation/CredentialValidator.ts +62 -0
  258. package/src/migration/validation/DIDCompatibilityValidator.ts +151 -0
  259. package/src/migration/validation/LifecycleValidator.ts +64 -0
  260. package/src/migration/validation/StorageValidator.ts +79 -0
  261. package/src/migration/validation/ValidationPipeline.ts +213 -0
  262. package/src/storage/LocalStorageAdapter.ts +61 -0
  263. package/src/storage/MemoryStorageAdapter.ts +29 -0
  264. package/src/storage/StorageAdapter.ts +25 -0
  265. package/src/storage/index.ts +3 -0
  266. package/src/types/bitcoin.ts +98 -0
  267. package/src/types/common.ts +92 -0
  268. package/src/types/credentials.ts +88 -0
  269. package/src/types/did.ts +31 -0
  270. package/src/types/external-shims.d.ts +53 -0
  271. package/src/types/index.ts +7 -0
  272. package/src/types/network.ts +175 -0
  273. package/src/utils/EventLogger.ts +298 -0
  274. package/src/utils/Logger.ts +322 -0
  275. package/src/utils/MetricsCollector.ts +358 -0
  276. package/src/utils/bitcoin-address.ts +130 -0
  277. package/src/utils/cbor.ts +12 -0
  278. package/src/utils/encoding.ts +127 -0
  279. package/src/utils/hash.ts +6 -0
  280. package/src/utils/retry.ts +46 -0
  281. package/src/utils/satoshi-validation.ts +196 -0
  282. package/src/utils/serialization.ts +96 -0
  283. package/src/utils/telemetry.ts +40 -0
  284. package/src/utils/validation.ts +119 -0
  285. package/src/vc/CredentialManager.ts +273 -0
  286. package/src/vc/Issuer.ts +100 -0
  287. package/src/vc/Verifier.ts +47 -0
  288. package/src/vc/cryptosuites/bbs.ts +253 -0
  289. package/src/vc/cryptosuites/bbsSimple.ts +21 -0
  290. package/src/vc/cryptosuites/eddsa.ts +99 -0
  291. package/src/vc/documentLoader.ts +67 -0
  292. package/src/vc/proofs/data-integrity.ts +33 -0
  293. package/src/vc/utils/jsonld.ts +18 -0
  294. package/test/logs/did_webvh_QmQsRNhXxPSCSeLjpbKYcNMZj8b1kBQAoC6cZmkFAgmpHt_example_com.jsonl +1 -0
  295. package/test/logs/did_webvh_QmSQkpD58qxcqMWHYcEmDUn3wk7hHvJwzYTrZmhh6zjPQ8_example_com_users_alice123_profile.jsonl +1 -0
  296. package/test/logs/did_webvh_QmTMda6VW3cUPdKk5Yc3onnv1vdgEumvWWdP2noAYFSjeG_example_com.jsonl +1 -0
  297. package/test/logs/did_webvh_QmTkb8KnCYcsnKKDCY4eUQuKQdKJLrCinvhw13v3zETxpE_example_com_users_etc_passwd.jsonl +1 -0
  298. package/test/logs/did_webvh_QmTn9FdCfpXFDrxHH52pwB4iNrDFVvNDjJ5FQTcDbmM3Fg_example_com.jsonl +1 -0
  299. package/test/logs/did_webvh_QmUCQUi1xjtJjnSQ1XJZgKqcWgErx1v7E2dz4DAPraAyJP_example_com_etc_passwd.jsonl +1 -0
  300. package/test/logs/did_webvh_QmUENQJCDKBJVRS5BkL6zjaUvcRjkb9xHmy7foCgRjmv3W_example_com.jsonl +1 -0
  301. package/test/logs/did_webvh_QmUPdGyjYBEnQ3aQUkmqyyBKTyjvCP5RZQGiaEDeTtf6dc_example_com.jsonl +1 -0
  302. package/test/logs/did_webvh_QmUoHTuHMWzQM29ZFrE9VLtMxkZ5u869yqee8LwcCLN39M_example_com.jsonl +1 -0
  303. package/test/logs/did_webvh_QmUrnms8G65ggVKsr9oQeWrLUBuGChwQPPb2LCFvaoNxaw_example_com_users_alice.jsonl +1 -0
  304. package/test/logs/did_webvh_QmUwiw3eSXdHG1hPvoAGu3cuK5jF4aXRYDLBAjPXfv1qzb_example_com_level1_level2.jsonl +1 -0
  305. package/test/logs/did_webvh_QmW7bzKh6yFEKNAtmVsrPGvvsMHTUQdzJSNsTZkbuGFpbj_example_com_secret.jsonl +1 -0
  306. package/test/logs/did_webvh_QmXbFTFBBJ8zpjdz9WE1DNN44A2wprFmdvAubjSffeyoAG_example_com.jsonl +1 -0
  307. package/test/logs/did_webvh_QmXyVXFPCTffGb2mTUFDeMCsScjnpLWkyUkVkB6q6QoeBf_example_com_C_Windows_System32.jsonl +1 -0
  308. package/test/logs/did_webvh_QmZK9B81gxZtvo5fYHYKDtKt8zZfZZPhmCMhbujBJuRRzE_example_com_etc_passwd.jsonl +1 -0
  309. package/test/logs/did_webvh_QmbNLCVSdXSVLrwFBvCBQPAabjtRb1SGHjkGVyw3QUbfBL_example_com_users_etc_passwd.jsonl +1 -0
  310. package/test/logs/did_webvh_QmbeaicmGW3Q7Yzbqmftc8a9jLBngokveb5A2KVKfVGZRb_example_com_my_org_user_name_test_123.jsonl +1 -0
  311. package/test/logs/did_webvh_Qmdv7c7AjUreUfoKyvkN2UpAWTozxKsv99srQetPJMJEnp_example_com_users_etc_passwd.jsonl +1 -0
  312. package/test/logs/did_webvh_QmeioWY3uypYLkYpCXe9eCYnn4xBVruP9C1d79azMrTEHG_example_com.jsonl +1 -0
  313. package/test/logs/did_webvh_Qmf4QH5dsA6Ecr5HJ6KaJL9uJRyY8RxrQdqoRCM25DzvPi_example_com_users_alice.jsonl +1 -0
  314. package/tests/__mocks__/bbs-signatures.js +17 -0
  315. package/tests/__mocks__/mf-base58.js +24 -0
  316. package/tests/e2e/README.md +97 -0
  317. package/tests/e2e/example.spec.ts +78 -0
  318. package/tests/fixtures/did-documents.ts +247 -0
  319. package/tests/index.test.ts +21 -0
  320. package/tests/integration/BatchOperations.test.ts +531 -0
  321. package/tests/integration/CompleteLifecycle.e2e.test.ts +735 -0
  322. package/tests/integration/CredentialManager.test.ts +42 -0
  323. package/tests/integration/DIDManager.test.ts +41 -0
  324. package/tests/integration/DidPeerToWebVhFlow.test.ts +351 -0
  325. package/tests/integration/Events.test.ts +435 -0
  326. package/tests/integration/Lifecycle.transfer.btco.integration.test.ts +25 -0
  327. package/tests/integration/LifecycleManager.test.ts +21 -0
  328. package/tests/integration/MultikeyFlow.test.ts +52 -0
  329. package/tests/integration/TelemetryIntegration.test.ts +395 -0
  330. package/tests/integration/WebVhPublish.test.ts +48 -0
  331. package/tests/integration/migration/peer-to-webvh.test.ts +172 -0
  332. package/tests/manual/test-commit-creation.ts +323 -0
  333. package/tests/mocks/MockKeyStore.ts +38 -0
  334. package/tests/mocks/adapters/MemoryStorageAdapter.ts +24 -0
  335. package/tests/mocks/adapters/MockFeeOracle.ts +11 -0
  336. package/tests/mocks/adapters/MockOrdinalsProvider.ts +76 -0
  337. package/tests/mocks/adapters/OrdMockProvider.test.ts +176 -0
  338. package/tests/mocks/adapters/index.ts +6 -0
  339. package/tests/performance/BatchOperations.perf.test.ts +403 -0
  340. package/tests/performance/logging.perf.test.ts +336 -0
  341. package/tests/sdk.test.ts +43 -0
  342. package/tests/security/bitcoin-penetration-tests.test.ts +622 -0
  343. package/tests/setup.bun.ts +69 -0
  344. package/tests/setup.jest.ts +23 -0
  345. package/tests/stress/batch-operations-stress.test.ts +571 -0
  346. package/tests/unit/adapters/FeeOracleMock.test.ts +40 -0
  347. package/tests/unit/bitcoin/BitcoinManager.test.ts +293 -0
  348. package/tests/unit/bitcoin/BroadcastClient.test.ts +52 -0
  349. package/tests/unit/bitcoin/OrdNodeProvider.test.ts +53 -0
  350. package/tests/unit/bitcoin/OrdinalsClient.test.ts +381 -0
  351. package/tests/unit/bitcoin/OrdinalsClientProvider.test.ts +102 -0
  352. package/tests/unit/bitcoin/PSBTBuilder.test.ts +84 -0
  353. package/tests/unit/bitcoin/fee-calculation.test.ts +261 -0
  354. package/tests/unit/bitcoin/transactions/commit.test.ts +649 -0
  355. package/tests/unit/bitcoin/transfer.test.ts +31 -0
  356. package/tests/unit/bitcoin/utxo-selection-new.test.ts +502 -0
  357. package/tests/unit/bitcoin/utxo.more.test.ts +39 -0
  358. package/tests/unit/bitcoin/utxo.selection.test.ts +38 -0
  359. package/tests/unit/core/OriginalsSDK.test.ts +152 -0
  360. package/tests/unit/crypto/Multikey.test.ts +206 -0
  361. package/tests/unit/crypto/Signer.test.ts +408 -0
  362. package/tests/unit/did/BtcoDidResolver.test.ts +611 -0
  363. package/tests/unit/did/DIDManager.more.test.ts +43 -0
  364. package/tests/unit/did/DIDManager.test.ts +185 -0
  365. package/tests/unit/did/Ed25519Verifier.test.ts +160 -0
  366. package/tests/unit/did/KeyManager.test.ts +452 -0
  367. package/tests/unit/did/OrdinalsClientProviderAdapter.test.ts +45 -0
  368. package/tests/unit/did/WebVHManager.test.ts +435 -0
  369. package/tests/unit/did/createBtcoDidDocument.test.ts +67 -0
  370. package/tests/unit/did/providers/OrdinalsClientProviderAdapter.test.ts +159 -0
  371. package/tests/unit/events/EventEmitter.test.ts +407 -0
  372. package/tests/unit/lifecycle/BatchOperations.test.ts +527 -0
  373. package/tests/unit/lifecycle/LifecycleManager.keymanagement.test.ts +312 -0
  374. package/tests/unit/lifecycle/LifecycleManager.prov.test.ts +18 -0
  375. package/tests/unit/lifecycle/LifecycleManager.test.ts +213 -0
  376. package/tests/unit/lifecycle/LifecycleManager.transfer.unit.test.ts +30 -0
  377. package/tests/unit/lifecycle/OriginalsAsset.test.ts +176 -0
  378. package/tests/unit/lifecycle/ProvenanceQuery.test.ts +577 -0
  379. package/tests/unit/lifecycle/ResourceVersioning.test.ts +651 -0
  380. package/tests/unit/storage/MemoryStorageAdapter.test.ts +93 -0
  381. package/tests/unit/types/network.test.ts +255 -0
  382. package/tests/unit/utils/EventIntegration.test.ts +384 -0
  383. package/tests/unit/utils/Logger.test.ts +473 -0
  384. package/tests/unit/utils/MetricsCollector.test.ts +358 -0
  385. package/tests/unit/utils/bitcoin-address.test.ts +250 -0
  386. package/tests/unit/utils/cbor.test.ts +35 -0
  387. package/tests/unit/utils/encoding.test.ts +318 -0
  388. package/tests/unit/utils/hash.test.ts +12 -0
  389. package/tests/unit/utils/retry.test.ts +100 -0
  390. package/tests/unit/utils/satoshi-validation.test.ts +354 -0
  391. package/tests/unit/utils/serialization.test.ts +124 -0
  392. package/tests/unit/utils/telemetry.test.ts +52 -0
  393. package/tests/unit/utils/validation.test.ts +141 -0
  394. package/tests/unit/vc/CredentialManager.test.ts +487 -0
  395. package/tests/unit/vc/Issuer.test.ts +107 -0
  396. package/tests/unit/vc/Verifier.test.ts +525 -0
  397. package/tests/unit/vc/bbs.test.ts +282 -0
  398. package/tests/unit/vc/cryptosuites/eddsa.test.ts +398 -0
  399. package/tests/unit/vc/documentLoader.test.ts +121 -0
  400. package/tests/unit/vc/proofs/data-integrity.test.ts +24 -0
  401. package/tsconfig.json +32 -0
  402. package/tsconfig.test.json +15 -0
  403. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,354 @@
1
+ import { describe, test, expect } from 'bun:test';
2
+ import {
3
+ validateSatoshiNumber,
4
+ parseSatoshiIdentifier,
5
+ assertValidSatoshi,
6
+ MAX_SATOSHI_SUPPLY
7
+ } from '../../../src/utils/satoshi-validation';
8
+ import { StructuredError } from '../../../src/utils/telemetry';
9
+
10
+ describe('satoshi-validation', () => {
11
+ describe('validateSatoshiNumber', () => {
12
+ describe('valid satoshi numbers', () => {
13
+ test('accepts zero', () => {
14
+ const result = validateSatoshiNumber(0);
15
+ expect(result.valid).toBe(true);
16
+ expect(result.error).toBeUndefined();
17
+ });
18
+
19
+ test('accepts one', () => {
20
+ const result = validateSatoshiNumber(1);
21
+ expect(result.valid).toBe(true);
22
+ expect(result.error).toBeUndefined();
23
+ });
24
+
25
+ test('accepts valid positive number', () => {
26
+ const result = validateSatoshiNumber(123456789);
27
+ expect(result.valid).toBe(true);
28
+ expect(result.error).toBeUndefined();
29
+ });
30
+
31
+ test('accepts string representation of valid number', () => {
32
+ const result = validateSatoshiNumber('123456789');
33
+ expect(result.valid).toBe(true);
34
+ expect(result.error).toBeUndefined();
35
+ });
36
+
37
+ test('accepts maximum supply', () => {
38
+ const result = validateSatoshiNumber(MAX_SATOSHI_SUPPLY);
39
+ expect(result.valid).toBe(true);
40
+ expect(result.error).toBeUndefined();
41
+ });
42
+
43
+ test('accepts maximum supply as string', () => {
44
+ const result = validateSatoshiNumber(String(MAX_SATOSHI_SUPPLY));
45
+ expect(result.valid).toBe(true);
46
+ expect(result.error).toBeUndefined();
47
+ });
48
+
49
+ test('accepts large valid ordinal number', () => {
50
+ const result = validateSatoshiNumber('1066296127976657');
51
+ expect(result.valid).toBe(true);
52
+ expect(result.error).toBeUndefined();
53
+ });
54
+ });
55
+
56
+ describe('invalid formats', () => {
57
+ test('rejects null', () => {
58
+ const result = validateSatoshiNumber(null as any);
59
+ expect(result.valid).toBe(false);
60
+ expect(result.error).toContain('cannot be null, undefined, or empty string');
61
+ });
62
+
63
+ test('rejects undefined', () => {
64
+ const result = validateSatoshiNumber(undefined as any);
65
+ expect(result.valid).toBe(false);
66
+ expect(result.error).toContain('cannot be null, undefined, or empty string');
67
+ });
68
+
69
+ test('rejects empty string', () => {
70
+ const result = validateSatoshiNumber('');
71
+ expect(result.valid).toBe(false);
72
+ expect(result.error).toContain('cannot be null, undefined, or empty string');
73
+ });
74
+
75
+ test('rejects whitespace-only string', () => {
76
+ const result = validateSatoshiNumber(' ');
77
+ expect(result.valid).toBe(false);
78
+ expect(result.error).toContain('cannot be empty or whitespace-only string');
79
+ });
80
+
81
+ test('rejects negative number', () => {
82
+ const result = validateSatoshiNumber(-1);
83
+ expect(result.valid).toBe(false);
84
+ expect(result.error).toContain('non-negative integer');
85
+ });
86
+
87
+ test('rejects negative string', () => {
88
+ const result = validateSatoshiNumber('-123');
89
+ expect(result.valid).toBe(false);
90
+ expect(result.error).toContain('non-negative integer');
91
+ });
92
+
93
+ test('rejects decimal number', () => {
94
+ const result = validateSatoshiNumber(123.456);
95
+ expect(result.valid).toBe(false);
96
+ expect(result.error).toContain('non-negative integer');
97
+ });
98
+
99
+ test('rejects decimal string', () => {
100
+ const result = validateSatoshiNumber('123.456');
101
+ expect(result.valid).toBe(false);
102
+ expect(result.error).toContain('non-negative integer');
103
+ });
104
+
105
+ test('rejects scientific notation', () => {
106
+ const result = validateSatoshiNumber('1e5');
107
+ expect(result.valid).toBe(false);
108
+ expect(result.error).toContain('non-negative integer');
109
+ });
110
+
111
+ test('rejects non-numeric string', () => {
112
+ const result = validateSatoshiNumber('abc');
113
+ expect(result.valid).toBe(false);
114
+ expect(result.error).toContain('non-negative integer');
115
+ });
116
+
117
+ test('rejects alphanumeric string', () => {
118
+ const result = validateSatoshiNumber('123abc');
119
+ expect(result.valid).toBe(false);
120
+ expect(result.error).toContain('non-negative integer');
121
+ });
122
+
123
+ test('rejects string with spaces', () => {
124
+ const result = validateSatoshiNumber('123 456');
125
+ expect(result.valid).toBe(false);
126
+ expect(result.error).toContain('non-negative integer');
127
+ });
128
+
129
+ test('rejects infinity', () => {
130
+ const result = validateSatoshiNumber(Infinity);
131
+ expect(result.valid).toBe(false);
132
+ expect(result.error).toContain('non-negative integer');
133
+ });
134
+
135
+ test('rejects NaN', () => {
136
+ const result = validateSatoshiNumber(NaN);
137
+ expect(result.valid).toBe(false);
138
+ expect(result.error).toContain('non-negative integer');
139
+ });
140
+ });
141
+
142
+ describe('range validation', () => {
143
+ test('rejects value exceeding maximum supply', () => {
144
+ const result = validateSatoshiNumber(MAX_SATOSHI_SUPPLY + 1);
145
+ expect(result.valid).toBe(false);
146
+ expect(result.error).toContain("within Bitcoin's total supply");
147
+ });
148
+
149
+ test('rejects extremely large value', () => {
150
+ const result = validateSatoshiNumber('999999999999999999');
151
+ expect(result.valid).toBe(false);
152
+ expect(result.error).toContain("within Bitcoin's total supply");
153
+ });
154
+
155
+ test('accepts value at boundary', () => {
156
+ const result = validateSatoshiNumber(MAX_SATOSHI_SUPPLY);
157
+ expect(result.valid).toBe(true);
158
+ });
159
+
160
+ test('accepts value just below boundary', () => {
161
+ const result = validateSatoshiNumber(MAX_SATOSHI_SUPPLY - 1);
162
+ expect(result.valid).toBe(true);
163
+ });
164
+ });
165
+ });
166
+
167
+ describe('parseSatoshiIdentifier', () => {
168
+ describe('plain satoshi numbers', () => {
169
+ test('parses valid plain number string', () => {
170
+ const result = parseSatoshiIdentifier('123456');
171
+ expect(result).toBe(123456);
172
+ });
173
+
174
+ test('parses zero', () => {
175
+ const result = parseSatoshiIdentifier('0');
176
+ expect(result).toBe(0);
177
+ });
178
+
179
+ test('parses large ordinal', () => {
180
+ const result = parseSatoshiIdentifier('1066296127976657');
181
+ expect(result).toBe(1066296127976657);
182
+ });
183
+ });
184
+
185
+ describe('did:btco DIDs', () => {
186
+ test('extracts satoshi from mainnet DID', () => {
187
+ const result = parseSatoshiIdentifier('did:btco:123456');
188
+ expect(result).toBe(123456);
189
+ });
190
+
191
+ test('extracts satoshi from testnet DID', () => {
192
+ const result = parseSatoshiIdentifier('did:btco:test:123456');
193
+ expect(result).toBe(123456);
194
+ });
195
+
196
+ test('extracts satoshi from signet DID', () => {
197
+ const result = parseSatoshiIdentifier('did:btco:sig:789012');
198
+ expect(result).toBe(789012);
199
+ });
200
+
201
+ test('extracts large ordinal from DID', () => {
202
+ const result = parseSatoshiIdentifier('did:btco:1066296127976657');
203
+ expect(result).toBe(1066296127976657);
204
+ });
205
+
206
+ test('handles whitespace around DID', () => {
207
+ const result = parseSatoshiIdentifier(' did:btco:123456 ');
208
+ expect(result).toBe(123456);
209
+ });
210
+ });
211
+
212
+ describe('error handling', () => {
213
+ test('throws on empty string', () => {
214
+ expect(() => parseSatoshiIdentifier('')).toThrow(StructuredError);
215
+ expect(() => parseSatoshiIdentifier('')).toThrow('non-empty string');
216
+ });
217
+
218
+ test('throws on whitespace-only string', () => {
219
+ expect(() => parseSatoshiIdentifier(' ')).toThrow(StructuredError);
220
+ expect(() => parseSatoshiIdentifier(' ')).toThrow('cannot be empty');
221
+ });
222
+
223
+ test('throws on null', () => {
224
+ expect(() => parseSatoshiIdentifier(null as any)).toThrow(StructuredError);
225
+ expect(() => parseSatoshiIdentifier(null as any)).toThrow('must be a non-empty string');
226
+ });
227
+
228
+ test('throws on invalid satoshi in DID', () => {
229
+ expect(() => parseSatoshiIdentifier('did:btco:-123')).toThrow(StructuredError);
230
+ expect(() => parseSatoshiIdentifier('did:btco:-123')).toThrow('non-negative integer');
231
+ });
232
+
233
+ test('throws on invalid satoshi format', () => {
234
+ expect(() => parseSatoshiIdentifier('abc')).toThrow(StructuredError);
235
+ expect(() => parseSatoshiIdentifier('abc')).toThrow('non-negative integer');
236
+ });
237
+
238
+ test('throws on decimal satoshi', () => {
239
+ expect(() => parseSatoshiIdentifier('123.456')).toThrow(StructuredError);
240
+ expect(() => parseSatoshiIdentifier('123.456')).toThrow('non-negative integer');
241
+ });
242
+
243
+ test('throws on satoshi exceeding max supply', () => {
244
+ const tooLarge = String(MAX_SATOSHI_SUPPLY + 1);
245
+ expect(() => parseSatoshiIdentifier(tooLarge)).toThrow(StructuredError);
246
+ expect(() => parseSatoshiIdentifier(tooLarge)).toThrow("within Bitcoin's total supply");
247
+ });
248
+
249
+ test('throws on invalid did:btco format with too few parts', () => {
250
+ expect(() => parseSatoshiIdentifier('did:btco')).toThrow(StructuredError);
251
+ expect(() => parseSatoshiIdentifier('did:btco')).toThrow('non-negative integer');
252
+ });
253
+
254
+ test('throws on invalid did:btco format with too many parts', () => {
255
+ expect(() => parseSatoshiIdentifier('did:btco:test:123:extra')).toThrow(StructuredError);
256
+ expect(() => parseSatoshiIdentifier('did:btco:test:123:extra')).toThrow('Invalid did:btco DID format');
257
+ });
258
+
259
+ test('throws on invalid network in did:btco', () => {
260
+ expect(() => parseSatoshiIdentifier('did:btco:invalid:123')).toThrow(StructuredError);
261
+ expect(() => parseSatoshiIdentifier('did:btco:invalid:123')).toThrow('unsupported network');
262
+ });
263
+
264
+ test('throws on mainnet prefix in 4-part format', () => {
265
+ // mainnet should use 3-part format (did:btco:123), not 4-part (did:btco:mainnet:123)
266
+ expect(() => parseSatoshiIdentifier('did:btco:mainnet:123')).toThrow(StructuredError);
267
+ expect(() => parseSatoshiIdentifier('did:btco:mainnet:123')).toThrow('unsupported network');
268
+ });
269
+
270
+ test('throws on regtest prefix', () => {
271
+ expect(() => parseSatoshiIdentifier('did:btco:regtest:123')).toThrow(StructuredError);
272
+ expect(() => parseSatoshiIdentifier('did:btco:regtest:123')).toThrow('unsupported network');
273
+ });
274
+ });
275
+
276
+ describe('validation integration', () => {
277
+ test('validates extracted satoshi', () => {
278
+ expect(() => parseSatoshiIdentifier('did:btco:test:-1')).toThrow();
279
+ expect(() => parseSatoshiIdentifier('did:btco:abc')).toThrow();
280
+ });
281
+ });
282
+ });
283
+
284
+ describe('assertValidSatoshi', () => {
285
+ test('does not throw for valid satoshi', () => {
286
+ expect(() => assertValidSatoshi(123456)).not.toThrow();
287
+ expect(() => assertValidSatoshi('123456')).not.toThrow();
288
+ expect(() => assertValidSatoshi(0)).not.toThrow();
289
+ });
290
+
291
+ test('throws StructuredError for invalid satoshi', () => {
292
+ expect(() => assertValidSatoshi('')).toThrow(StructuredError);
293
+ expect(() => assertValidSatoshi(-1)).toThrow(StructuredError);
294
+ expect(() => assertValidSatoshi('abc')).toThrow(StructuredError);
295
+ });
296
+
297
+ test('throws with error code INVALID_SATOSHI', () => {
298
+ try {
299
+ assertValidSatoshi('');
300
+ fail('Should have thrown');
301
+ } catch (error: any) {
302
+ expect(error).toBeInstanceOf(StructuredError);
303
+ expect(error.code).toBe('INVALID_SATOSHI');
304
+ }
305
+ });
306
+
307
+ test('includes descriptive error message', () => {
308
+ try {
309
+ assertValidSatoshi('');
310
+ fail('Should have thrown');
311
+ } catch (error: any) {
312
+ expect(error.message).toContain('cannot be null, undefined, or empty string');
313
+ }
314
+ });
315
+ });
316
+
317
+ describe('edge cases', () => {
318
+ test('handles zero correctly', () => {
319
+ expect(validateSatoshiNumber(0).valid).toBe(true);
320
+ expect(validateSatoshiNumber('0').valid).toBe(true);
321
+ expect(parseSatoshiIdentifier('0')).toBe(0);
322
+ });
323
+
324
+ test('handles very long valid number string', () => {
325
+ const longValid = '100000000000000';
326
+ expect(validateSatoshiNumber(longValid).valid).toBe(true);
327
+ expect(parseSatoshiIdentifier(longValid)).toBe(100000000000000);
328
+ });
329
+
330
+ test('trims whitespace from input', () => {
331
+ expect(validateSatoshiNumber(' 123 ').valid).toBe(true);
332
+ expect(parseSatoshiIdentifier(' 123 ')).toBe(123);
333
+ });
334
+
335
+ test('rejects Infinity in validateSatoshiNumber', () => {
336
+ const result = validateSatoshiNumber(Infinity);
337
+ expect(result.valid).toBe(false);
338
+ expect(result.error).toBeDefined();
339
+ });
340
+
341
+ test('rejects decimal number', () => {
342
+ const result = validateSatoshiNumber(123.5);
343
+ expect(result.valid).toBe(false);
344
+ expect(result.error).toBeDefined();
345
+ });
346
+
347
+ test('rejects negative number', () => {
348
+ const result = validateSatoshiNumber(-5);
349
+ expect(result.valid).toBe(false);
350
+ expect(result.error).toBeDefined();
351
+ });
352
+ });
353
+
354
+ });
@@ -0,0 +1,124 @@
1
+ import { describe, test, expect } from 'bun:test';
2
+ import { serializeDIDDocument, deserializeDIDDocument, serializeCredential, deserializeCredential, canonicalizeDocument } from '../../../src/utils/serialization';
3
+
4
+ describe('serialization utils', () => {
5
+ test('serialize/deserialize DID document roundtrip', () => {
6
+ const doc: any = { '@context': ['https://www.w3.org/ns/did/v1'], id: 'did:peer:abc' };
7
+ const ser = serializeDIDDocument(doc);
8
+ const round = deserializeDIDDocument(ser);
9
+ expect(round).toEqual(doc);
10
+ });
11
+
12
+ test('serialize/deserialize VC roundtrip', () => {
13
+ const vc: any = { '@context': ['https://www.w3.org/2018/credentials/v1'], type: ['VerifiableCredential'], issuer: 'did:peer:abc', issuanceDate: new Date().toISOString(), credentialSubject: { id: 'did:peer:abc' } };
14
+ const ser = serializeCredential(vc);
15
+ const round = deserializeCredential(ser);
16
+ expect(round).toEqual(vc);
17
+ });
18
+
19
+ test('canonicalizeDocument normalizes nested structures deterministically', async () => {
20
+ const context = {
21
+ '@version': 1.1,
22
+ id: '@id',
23
+ type: '@type',
24
+ name: 'https://schema.org/name',
25
+ details: {
26
+ '@id': 'https://example.org/details',
27
+ '@context': {
28
+ '@version': 1.1,
29
+ count: 'https://example.org/count',
30
+ tags: {
31
+ '@id': 'https://example.org/tags',
32
+ '@container': '@set'
33
+ }
34
+ }
35
+ }
36
+ };
37
+
38
+ const docA = {
39
+ '@context': context,
40
+ id: 'urn:example:1',
41
+ type: ['ExampleCredential'],
42
+ name: 'Sample',
43
+ details: {
44
+ count: 2,
45
+ tags: ['b', 'a']
46
+ }
47
+ };
48
+
49
+ const docB = {
50
+ name: 'Sample',
51
+ details: {
52
+ tags: ['a', 'b'],
53
+ count: 2
54
+ },
55
+ type: ['ExampleCredential'],
56
+ id: 'urn:example:1',
57
+ '@context': context
58
+ };
59
+
60
+ const canonA = await canonicalizeDocument(docA);
61
+ const canonB = await canonicalizeDocument(docB);
62
+ expect(canonA).toEqual(canonB);
63
+ });
64
+
65
+ test('deserializeDIDDocument throws on invalid JSON', () => {
66
+ expect(() => deserializeDIDDocument('{invalid')).toThrow('Invalid DID Document JSON');
67
+ });
68
+
69
+ test('deserializeCredential throws on invalid JSON', () => {
70
+ expect(() => deserializeCredential('{invalid')).toThrow('Invalid Verifiable Credential JSON');
71
+ });
72
+
73
+ test('canonicalizeDocument with custom documentLoader', async () => {
74
+ const customLoader = async (url: string) => {
75
+ return {
76
+ documentUrl: url,
77
+ document: {
78
+ '@context': {
79
+ '@version': 1.1,
80
+ 'TestType': 'https://example.org/TestType',
81
+ 'id': '@id',
82
+ 'type': '@type'
83
+ }
84
+ },
85
+ contextUrl: null
86
+ };
87
+ };
88
+
89
+ const doc = {
90
+ '@context': 'https://custom.example/context',
91
+ id: 'urn:test:1',
92
+ type: 'TestType'
93
+ };
94
+
95
+ const canon = await canonicalizeDocument(doc, { documentLoader: customLoader });
96
+ expect(typeof canon).toBe('string');
97
+ // canonicalizeDocument might return empty string for certain contexts/documents
98
+ expect(canon).toBeDefined();
99
+ });
100
+
101
+ test('canonicalizeDocument throws error with message when canonize fails', async () => {
102
+ const invalidDoc = {
103
+ '@context': 'https://invalid-context-that-doesnt-exist-12345.example/',
104
+ id: 'urn:test:1'
105
+ };
106
+
107
+ await expect(canonicalizeDocument(invalidDoc)).rejects.toThrow('Failed to canonicalize document');
108
+ });
109
+
110
+ test('canonicalizeDocument handles non-Error thrown values', async () => {
111
+ // This is difficult to test directly, but we can verify the error handling path exists
112
+ const doc = {
113
+ '@context': ['https://www.w3.org/2018/credentials/v1'],
114
+ id: 'urn:test:1',
115
+ type: 'TestType'
116
+ };
117
+
118
+ // Normal case should work
119
+ const canon = await canonicalizeDocument(doc);
120
+ expect(typeof canon).toBe('string');
121
+ });
122
+ });
123
+
124
+
@@ -0,0 +1,52 @@
1
+ import { describe, test, expect, mock } from 'bun:test';
2
+ import { emitTelemetry, emitError, StructuredError } from '../../../src/utils/telemetry';
3
+
4
+ describe('utils/telemetry', () => {
5
+ test('emitTelemetry invokes onEvent with default level info', () => {
6
+ const onEvent = mock();
7
+ emitTelemetry({ onEvent }, { name: 'event-1' });
8
+ expect(onEvent).toHaveBeenCalledTimes(1);
9
+ const arg = onEvent.mock.calls[0][0];
10
+ expect(arg.name).toBe('event-1');
11
+ expect(arg.level).toBe('info');
12
+ });
13
+
14
+ test('emitTelemetry respects provided level', () => {
15
+ const onEvent = mock();
16
+ emitTelemetry({ onEvent }, { name: 'event-2', level: 'debug' });
17
+ expect(onEvent).toHaveBeenCalledTimes(1);
18
+ expect(onEvent.mock.calls[0][0].level).toBe('debug');
19
+ });
20
+
21
+ test('emitTelemetry is no-op when hooks missing or handler absent', () => {
22
+ expect(() => emitTelemetry(undefined, { name: 'noop' })).not.toThrow();
23
+ expect(() => emitTelemetry({}, { name: 'noop' })).not.toThrow();
24
+ });
25
+
26
+ test('emitTelemetry swallows handler exceptions', () => {
27
+ const onEvent = mock(() => { throw new Error('boom'); });
28
+ expect(() => emitTelemetry({ onEvent }, { name: 'err' })).not.toThrow();
29
+ });
30
+
31
+ test('StructuredError holds code, message, and optional details', () => {
32
+ const err = new StructuredError('E_TEST', 'message', { a: 1 });
33
+ expect(err).toBeInstanceOf(Error);
34
+ expect(err.name).toBe('StructuredError');
35
+ expect(err.code).toBe('E_TEST');
36
+ expect(err.message).toBe('message');
37
+ expect(err.details).toEqual({ a: 1 });
38
+ });
39
+
40
+ test('emitError calls onError and swallows handler exceptions', () => {
41
+ const onErrorOk = mock();
42
+ const se = new StructuredError('E', 'm');
43
+ emitError({ onError: onErrorOk }, se);
44
+ expect(onErrorOk).toHaveBeenCalledWith(se);
45
+
46
+ const onErrorThrow = mock(() => { throw new Error('boom'); });
47
+ expect(() => emitError({ onError: onErrorThrow }, se)).not.toThrow();
48
+ expect(() => emitError(undefined, se)).not.toThrow();
49
+ expect(() => emitError({}, se)).not.toThrow();
50
+ });
51
+ });
52
+
@@ -0,0 +1,141 @@
1
+ import { describe, test, expect, spyOn } from 'bun:test';
2
+ import { validateDID, validateCredential, validateDIDDocument, hashResource } from '../../../src/utils/validation';
3
+
4
+ describe('validation utils', () => {
5
+ test('validateDID supports peer, webvh, btco', () => {
6
+ expect(validateDID('did:peer:123')).toBe(true);
7
+ expect(validateDID('did:webvh:example.com:abc')).toBe(true);
8
+ expect(validateDID('did:btco:123')).toBe(true);
9
+ expect(validateDID('did:web:example.com')).toBe(false);
10
+ expect(validateDID('invalid')).toBe(false);
11
+ // Cover defensive branch: regex.test returns true but match returns null
12
+ const testSpy = spyOn(RegExp.prototype, 'test').mockReturnValueOnce(true);
13
+ const matchSpy = spyOn(String.prototype, 'match').mockReturnValueOnce(null as any);
14
+ expect(validateDID('did:peer:abc')).toBe(false);
15
+ testSpy.mockRestore();
16
+ matchSpy.mockRestore();
17
+ });
18
+
19
+ test('validateCredential basic VC shape', () => {
20
+ const vc: any = {
21
+ '@context': ['https://www.w3.org/2018/credentials/v1'],
22
+ type: ['VerifiableCredential', 'Test'],
23
+ issuer: 'did:peer:abc',
24
+ issuanceDate: new Date().toISOString(),
25
+ credentialSubject: { id: 'did:peer:abc', foo: 'bar' }
26
+ };
27
+ expect(validateCredential(vc)).toBe(true);
28
+ });
29
+
30
+ test('validateCredential accepts issuer object with DID id', () => {
31
+ const vc: any = {
32
+ '@context': ['https://www.w3.org/2018/credentials/v1'],
33
+ type: ['VerifiableCredential'],
34
+ issuer: { id: 'did:webvh:example.com:abc' },
35
+ issuanceDate: new Date().toISOString(),
36
+ credentialSubject: { id: 'did:webvh:example.com:abc' }
37
+ };
38
+ expect(validateCredential(vc)).toBe(true);
39
+ });
40
+
41
+ test('validateCredential returns false when issuer object has no id', () => {
42
+ const vc: any = {
43
+ '@context': ['https://www.w3.org/2018/credentials/v1'],
44
+ type: ['VerifiableCredential'],
45
+ issuer: { name: 'Test Issuer' },
46
+ issuanceDate: new Date().toISOString(),
47
+ credentialSubject: { id: 'did:webvh:example.com:abc' }
48
+ };
49
+ expect(validateCredential(vc)).toBe(false);
50
+ });
51
+
52
+ test('validateCredential negative cases', () => {
53
+ expect(validateCredential({} as any)).toBe(false);
54
+ // Fails on missing type array
55
+ expect(validateCredential({ '@context': ['https://www.w3.org/2018/credentials/v1'], type: undefined, issuer: 'did:peer:x', issuanceDate: new Date().toISOString(), credentialSubject: {} } as any)).toBe(false);
56
+ // Fails on missing VerifiableCredential type
57
+ expect(validateCredential({ '@context': ['https://www.w3.org/2018/credentials/v1'], type: ['Other'], issuer: 'did:peer:x', issuanceDate: new Date().toISOString(), credentialSubject: {} } as any)).toBe(false);
58
+ // Fails on missing issuanceDate
59
+ expect(validateCredential({ '@context': ['https://www.w3.org/2018/credentials/v1'], type: ['VerifiableCredential'], issuer: 'did:peer:x', credentialSubject: {} } as any)).toBe(false);
60
+ // Fails on missing credentialSubject
61
+ expect(validateCredential({ '@context': ['https://www.w3.org/2018/credentials/v1'], type: ['VerifiableCredential'], issuer: 'did:peer:x', issuanceDate: new Date().toISOString() } as any)).toBe(false);
62
+ // Fails when VC v1 context missing
63
+ expect(validateCredential({ '@context': ['https://example.com/ctx'], type: ['VerifiableCredential'], issuer: 'did:peer:x', issuanceDate: new Date().toISOString(), credentialSubject: {} } as any)).toBe(false);
64
+ // Fails when issuer is not DID
65
+ expect(validateCredential({ '@context': ['https://www.w3.org/2018/credentials/v1'], type: ['VerifiableCredential'], issuer: 'https://example.com', issuanceDate: new Date().toISOString(), credentialSubject: {} } as any)).toBe(false);
66
+ // Fails when issuer object has non-DID id
67
+ expect(validateCredential({ '@context': ['https://www.w3.org/2018/credentials/v1'], type: ['VerifiableCredential'], issuer: { id: 'example.com' }, issuanceDate: new Date().toISOString(), credentialSubject: {} } as any)).toBe(false);
68
+ // Fails when issuanceDate is not valid ISO
69
+ expect(validateCredential({ '@context': ['https://www.w3.org/2018/credentials/v1'], type: ['VerifiableCredential'], issuer: 'did:peer:x', issuanceDate: 'not-a-date', credentialSubject: {} } as any)).toBe(false);
70
+ });
71
+
72
+ test('validateDIDDocument shape', () => {
73
+ const didDoc: any = {
74
+ '@context': ['https://www.w3.org/ns/did/v1'],
75
+ id: 'did:peer:abc',
76
+ verificationMethod: [
77
+ { id: '#key-1', type: 'Ed25519VerificationKey2020', controller: 'did:peer:abc', publicKeyMultibase: 'z...' }
78
+ ]
79
+ };
80
+ expect(validateDIDDocument(didDoc)).toBe(true);
81
+ });
82
+
83
+ test('validateDIDDocument negative cases', () => {
84
+ expect(validateDIDDocument({} as any)).toBe(false);
85
+ expect(validateDIDDocument({ '@context': [], id: 'invalid' } as any)).toBe(false);
86
+ const badVm: any = {
87
+ '@context': ['https://www.w3.org/ns/did/v1'],
88
+ id: 'did:peer:abc',
89
+ verificationMethod: [{ id: '#1', type: 'X', controller: '', publicKeyMultibase: '' }, { id: null }]
90
+ };
91
+ expect(validateDIDDocument(badVm)).toBe(false);
92
+ // invalid multibase prefix for VM
93
+ const badMultibase: any = {
94
+ '@context': ['https://www.w3.org/ns/did/v1'],
95
+ id: 'did:peer:abc',
96
+ verificationMethod: [{ id: '#key', type: 'Ed25519VerificationKey2020', controller: 'did:peer:abc', publicKeyMultibase: 'xabc' }]
97
+ };
98
+ expect(validateDIDDocument(badMultibase)).toBe(false);
99
+ // invalid controller on verificationMethod
100
+ const badVmController: any = {
101
+ '@context': ['https://www.w3.org/ns/did/v1'],
102
+ id: 'did:peer:abc',
103
+ verificationMethod: [{ id: '#key', type: 'Ed25519VerificationKey2020', controller: 'not-a-did', publicKeyMultibase: 'zabc' }]
104
+ };
105
+ expect(validateDIDDocument(badVmController)).toBe(false);
106
+ // invalid controller array
107
+ const badControllers: any = {
108
+ '@context': ['https://www.w3.org/ns/did/v1'],
109
+ id: 'did:peer:abc',
110
+ controller: ['not-a-did']
111
+ };
112
+ expect(validateDIDDocument(badControllers)).toBe(false);
113
+ });
114
+
115
+ test('validateDIDDocument with controller array passes when entries are DIDs', () => {
116
+ const ok: any = {
117
+ '@context': ['https://www.w3.org/ns/did/v1'],
118
+ id: 'did:peer:abc',
119
+ controller: ['did:peer:abc', 'did:webvh:example.com:abc']
120
+ };
121
+ expect(validateDIDDocument(ok)).toBe(true);
122
+ });
123
+
124
+ test('hashResource returns sha256 hex', () => {
125
+ const hash = hashResource(Buffer.from('hello'));
126
+ expect(hash).toHaveLength(64);
127
+ expect(/[0-9a-f]{64}/.test(hash)).toBe(true);
128
+ });
129
+ });
130
+
131
+ /** Inlined from validation.no-vm.part.ts */
132
+
133
+ describe('validateDIDDocument when verificationMethod absent', () => {
134
+ test('returns true for valid doc with no verificationMethod property', () => {
135
+ const didDoc: any = {
136
+ '@context': ['https://www.w3.org/ns/did/v1'],
137
+ id: 'did:btco:1234'
138
+ };
139
+ expect(validateDIDDocument(didDoc)).toBe(true);
140
+ });
141
+ });