@arkade-os/sdk 0.3.12 → 0.4.0-next.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 (250) hide show
  1. package/README.md +483 -54
  2. package/dist/cjs/adapters/expo-db.js +35 -0
  3. package/dist/cjs/asset/assetGroup.js +141 -0
  4. package/dist/cjs/asset/assetId.js +88 -0
  5. package/dist/cjs/asset/assetInput.js +204 -0
  6. package/dist/cjs/asset/assetOutput.js +159 -0
  7. package/dist/cjs/asset/assetRef.js +82 -0
  8. package/dist/cjs/asset/index.js +24 -0
  9. package/dist/cjs/asset/metadata.js +172 -0
  10. package/dist/cjs/asset/packet.js +164 -0
  11. package/dist/cjs/asset/types.js +25 -0
  12. package/dist/cjs/asset/utils.js +105 -0
  13. package/dist/cjs/contracts/arkcontract.js +148 -0
  14. package/dist/cjs/contracts/contractManager.js +436 -0
  15. package/dist/cjs/contracts/contractWatcher.js +567 -0
  16. package/dist/cjs/contracts/handlers/default.js +85 -0
  17. package/dist/cjs/contracts/handlers/delegate.js +89 -0
  18. package/dist/cjs/contracts/handlers/helpers.js +105 -0
  19. package/dist/cjs/contracts/handlers/index.js +19 -0
  20. package/dist/cjs/contracts/handlers/registry.js +89 -0
  21. package/dist/cjs/contracts/handlers/vhtlc.js +193 -0
  22. package/dist/cjs/contracts/index.js +41 -0
  23. package/dist/cjs/contracts/types.js +2 -0
  24. package/dist/cjs/db/manager.js +97 -0
  25. package/dist/cjs/forfeit.js +12 -8
  26. package/dist/cjs/identity/index.js +1 -0
  27. package/dist/cjs/identity/seedIdentity.js +255 -0
  28. package/dist/cjs/index.js +70 -14
  29. package/dist/cjs/intent/index.js +28 -2
  30. package/dist/cjs/providers/ark.js +7 -0
  31. package/dist/cjs/providers/delegator.js +66 -0
  32. package/dist/cjs/providers/expoIndexer.js +5 -0
  33. package/dist/cjs/providers/indexer.js +68 -1
  34. package/dist/cjs/providers/onchain.js +2 -2
  35. package/dist/cjs/providers/utils.js +1 -0
  36. package/dist/cjs/repositories/contractRepository.js +0 -103
  37. package/dist/cjs/repositories/inMemory/contractRepository.js +55 -0
  38. package/dist/cjs/repositories/inMemory/walletRepository.js +80 -0
  39. package/dist/cjs/repositories/index.js +16 -0
  40. package/dist/cjs/repositories/indexedDB/contractRepository.js +187 -0
  41. package/dist/cjs/repositories/indexedDB/db.js +57 -0
  42. package/dist/cjs/repositories/indexedDB/schema.js +159 -0
  43. package/dist/cjs/repositories/indexedDB/walletRepository.js +338 -0
  44. package/dist/cjs/repositories/indexedDB/websqlAdapter.js +144 -0
  45. package/dist/cjs/repositories/migrations/contractRepositoryImpl.js +127 -0
  46. package/dist/cjs/repositories/migrations/fromStorageAdapter.js +66 -0
  47. package/dist/cjs/repositories/migrations/walletRepositoryImpl.js +180 -0
  48. package/dist/cjs/repositories/walletRepository.js +0 -169
  49. package/dist/cjs/script/base.js +54 -0
  50. package/dist/cjs/script/delegate.js +49 -0
  51. package/dist/cjs/storage/asyncStorage.js +4 -1
  52. package/dist/cjs/storage/fileSystem.js +3 -0
  53. package/dist/cjs/storage/inMemory.js +3 -0
  54. package/dist/cjs/storage/indexedDB.js +5 -1
  55. package/dist/cjs/storage/localStorage.js +3 -0
  56. package/dist/cjs/utils/arkTransaction.js +16 -0
  57. package/dist/cjs/utils/transactionHistory.js +50 -0
  58. package/dist/cjs/utils/txSizeEstimator.js +39 -14
  59. package/dist/cjs/wallet/asset-manager.js +338 -0
  60. package/dist/cjs/wallet/asset.js +117 -0
  61. package/dist/cjs/wallet/batch.js +1 -1
  62. package/dist/cjs/wallet/delegator.js +235 -0
  63. package/dist/cjs/wallet/expo/background.js +133 -0
  64. package/dist/cjs/wallet/expo/index.js +9 -0
  65. package/dist/cjs/wallet/expo/wallet.js +231 -0
  66. package/dist/cjs/wallet/onchain.js +57 -12
  67. package/dist/cjs/wallet/serviceWorker/wallet-message-handler.js +568 -0
  68. package/dist/cjs/wallet/serviceWorker/wallet.js +383 -102
  69. package/dist/cjs/wallet/unroll.js +7 -2
  70. package/dist/cjs/wallet/utils.js +60 -0
  71. package/dist/cjs/wallet/validation.js +151 -0
  72. package/dist/cjs/wallet/vtxo-manager.js +1 -1
  73. package/dist/cjs/wallet/wallet.js +702 -260
  74. package/dist/cjs/worker/browser/service-worker-manager.js +82 -0
  75. package/dist/cjs/{wallet/serviceWorker → worker/browser}/utils.js +2 -1
  76. package/dist/cjs/worker/expo/asyncStorageTaskQueue.js +78 -0
  77. package/dist/cjs/worker/expo/index.js +12 -0
  78. package/dist/cjs/worker/expo/processors/contractPollProcessor.js +61 -0
  79. package/dist/cjs/worker/expo/processors/index.js +6 -0
  80. package/dist/cjs/worker/expo/taskQueue.js +41 -0
  81. package/dist/cjs/worker/expo/taskRunner.js +57 -0
  82. package/dist/cjs/worker/messageBus.js +252 -0
  83. package/dist/esm/adapters/expo-db.js +27 -0
  84. package/dist/esm/asset/assetGroup.js +137 -0
  85. package/dist/esm/asset/assetId.js +84 -0
  86. package/dist/esm/asset/assetInput.js +199 -0
  87. package/dist/esm/asset/assetOutput.js +154 -0
  88. package/dist/esm/asset/assetRef.js +78 -0
  89. package/dist/esm/asset/index.js +8 -0
  90. package/dist/esm/asset/metadata.js +167 -0
  91. package/dist/esm/asset/packet.js +159 -0
  92. package/dist/esm/asset/types.js +22 -0
  93. package/dist/esm/asset/utils.js +99 -0
  94. package/dist/esm/contracts/arkcontract.js +141 -0
  95. package/dist/esm/contracts/contractManager.js +432 -0
  96. package/dist/esm/contracts/contractWatcher.js +563 -0
  97. package/dist/esm/contracts/handlers/default.js +82 -0
  98. package/dist/esm/contracts/handlers/delegate.js +86 -0
  99. package/dist/esm/contracts/handlers/helpers.js +66 -0
  100. package/dist/esm/contracts/handlers/index.js +12 -0
  101. package/dist/esm/contracts/handlers/registry.js +86 -0
  102. package/dist/esm/contracts/handlers/vhtlc.js +190 -0
  103. package/dist/esm/contracts/index.js +13 -0
  104. package/dist/esm/contracts/types.js +1 -0
  105. package/dist/esm/db/manager.js +92 -0
  106. package/dist/esm/forfeit.js +11 -8
  107. package/dist/esm/identity/index.js +1 -0
  108. package/dist/esm/identity/seedIdentity.js +249 -0
  109. package/dist/esm/index.js +25 -15
  110. package/dist/esm/intent/index.js +28 -2
  111. package/dist/esm/providers/ark.js +7 -0
  112. package/dist/esm/providers/delegator.js +62 -0
  113. package/dist/esm/providers/expoIndexer.js +5 -0
  114. package/dist/esm/providers/indexer.js +68 -1
  115. package/dist/esm/providers/onchain.js +2 -2
  116. package/dist/esm/providers/utils.js +1 -0
  117. package/dist/esm/repositories/contractRepository.js +1 -101
  118. package/dist/esm/repositories/inMemory/contractRepository.js +51 -0
  119. package/dist/esm/repositories/inMemory/walletRepository.js +76 -0
  120. package/dist/esm/repositories/index.js +8 -0
  121. package/dist/esm/repositories/indexedDB/contractRepository.js +183 -0
  122. package/dist/esm/repositories/indexedDB/db.js +42 -0
  123. package/dist/esm/repositories/indexedDB/schema.js +155 -0
  124. package/dist/esm/repositories/indexedDB/walletRepository.js +334 -0
  125. package/dist/esm/repositories/indexedDB/websqlAdapter.js +138 -0
  126. package/dist/esm/repositories/migrations/contractRepositoryImpl.js +121 -0
  127. package/dist/esm/repositories/migrations/fromStorageAdapter.js +58 -0
  128. package/dist/esm/repositories/migrations/walletRepositoryImpl.js +176 -0
  129. package/dist/esm/repositories/walletRepository.js +1 -167
  130. package/dist/esm/script/base.js +21 -1
  131. package/dist/esm/script/delegate.js +46 -0
  132. package/dist/esm/storage/asyncStorage.js +4 -1
  133. package/dist/esm/storage/fileSystem.js +3 -0
  134. package/dist/esm/storage/inMemory.js +3 -0
  135. package/dist/esm/storage/indexedDB.js +5 -1
  136. package/dist/esm/storage/localStorage.js +3 -0
  137. package/dist/esm/utils/arkTransaction.js +15 -0
  138. package/dist/esm/utils/transactionHistory.js +50 -0
  139. package/dist/esm/utils/txSizeEstimator.js +39 -14
  140. package/dist/esm/wallet/asset-manager.js +333 -0
  141. package/dist/esm/wallet/asset.js +111 -0
  142. package/dist/esm/wallet/batch.js +1 -1
  143. package/dist/esm/wallet/delegator.js +231 -0
  144. package/dist/esm/wallet/expo/background.js +128 -0
  145. package/dist/esm/wallet/expo/index.js +2 -0
  146. package/dist/esm/wallet/expo/wallet.js +194 -0
  147. package/dist/esm/wallet/onchain.js +57 -12
  148. package/dist/esm/wallet/serviceWorker/wallet-message-handler.js +564 -0
  149. package/dist/esm/wallet/serviceWorker/wallet.js +382 -101
  150. package/dist/esm/wallet/unroll.js +7 -2
  151. package/dist/esm/wallet/utils.js +55 -0
  152. package/dist/esm/wallet/validation.js +139 -0
  153. package/dist/esm/wallet/vtxo-manager.js +1 -1
  154. package/dist/esm/wallet/wallet.js +704 -229
  155. package/dist/esm/worker/browser/service-worker-manager.js +76 -0
  156. package/dist/esm/{wallet/serviceWorker → worker/browser}/utils.js +2 -1
  157. package/dist/esm/worker/expo/asyncStorageTaskQueue.js +74 -0
  158. package/dist/esm/worker/expo/index.js +4 -0
  159. package/dist/esm/worker/expo/processors/contractPollProcessor.js +58 -0
  160. package/dist/esm/worker/expo/processors/index.js +1 -0
  161. package/dist/esm/worker/expo/taskQueue.js +37 -0
  162. package/dist/esm/worker/expo/taskRunner.js +54 -0
  163. package/dist/esm/worker/messageBus.js +248 -0
  164. package/dist/types/adapters/expo-db.d.ts +7 -0
  165. package/dist/types/asset/assetGroup.d.ts +28 -0
  166. package/dist/types/asset/assetId.d.ts +19 -0
  167. package/dist/types/asset/assetInput.d.ts +46 -0
  168. package/dist/types/asset/assetOutput.d.ts +39 -0
  169. package/dist/types/asset/assetRef.d.ts +25 -0
  170. package/dist/types/asset/index.d.ts +8 -0
  171. package/dist/types/asset/metadata.d.ts +37 -0
  172. package/dist/types/asset/packet.d.ts +27 -0
  173. package/dist/types/asset/types.d.ts +18 -0
  174. package/dist/types/asset/utils.d.ts +21 -0
  175. package/dist/types/contracts/arkcontract.d.ts +101 -0
  176. package/dist/types/contracts/contractManager.d.ts +331 -0
  177. package/dist/types/contracts/contractWatcher.d.ts +192 -0
  178. package/dist/types/contracts/handlers/default.d.ts +19 -0
  179. package/dist/types/contracts/handlers/delegate.d.ts +21 -0
  180. package/dist/types/contracts/handlers/helpers.d.ts +18 -0
  181. package/dist/types/contracts/handlers/index.d.ts +7 -0
  182. package/dist/types/contracts/handlers/registry.d.ts +65 -0
  183. package/dist/types/contracts/handlers/vhtlc.d.ts +32 -0
  184. package/dist/types/contracts/index.d.ts +14 -0
  185. package/dist/types/contracts/types.d.ts +222 -0
  186. package/dist/types/db/manager.d.ts +22 -0
  187. package/dist/types/forfeit.d.ts +2 -1
  188. package/dist/types/identity/index.d.ts +1 -0
  189. package/dist/types/identity/seedIdentity.d.ts +128 -0
  190. package/dist/types/index.d.ts +21 -12
  191. package/dist/types/intent/index.d.ts +2 -1
  192. package/dist/types/providers/ark.d.ts +11 -2
  193. package/dist/types/providers/delegator.d.ts +29 -0
  194. package/dist/types/providers/indexer.d.ts +11 -1
  195. package/dist/types/repositories/contractRepository.d.ts +30 -19
  196. package/dist/types/repositories/inMemory/contractRepository.d.ts +17 -0
  197. package/dist/types/repositories/inMemory/walletRepository.d.ts +26 -0
  198. package/dist/types/repositories/index.d.ts +7 -0
  199. package/dist/types/repositories/indexedDB/contractRepository.d.ts +21 -0
  200. package/dist/types/repositories/indexedDB/db.d.ts +56 -0
  201. package/dist/types/repositories/indexedDB/schema.d.ts +8 -0
  202. package/dist/types/repositories/indexedDB/walletRepository.d.ts +25 -0
  203. package/dist/types/repositories/indexedDB/websqlAdapter.d.ts +49 -0
  204. package/dist/types/repositories/migrations/contractRepositoryImpl.d.ts +24 -0
  205. package/dist/types/repositories/migrations/fromStorageAdapter.d.ts +19 -0
  206. package/dist/types/repositories/migrations/walletRepositoryImpl.d.ts +27 -0
  207. package/dist/types/repositories/walletRepository.d.ts +13 -24
  208. package/dist/types/script/base.d.ts +1 -0
  209. package/dist/types/script/delegate.d.ts +36 -0
  210. package/dist/types/storage/asyncStorage.d.ts +4 -0
  211. package/dist/types/storage/fileSystem.d.ts +3 -0
  212. package/dist/types/storage/inMemory.d.ts +3 -0
  213. package/dist/types/storage/index.d.ts +3 -0
  214. package/dist/types/storage/indexedDB.d.ts +3 -0
  215. package/dist/types/storage/localStorage.d.ts +3 -0
  216. package/dist/types/utils/arkTransaction.d.ts +6 -0
  217. package/dist/types/utils/txSizeEstimator.d.ts +12 -2
  218. package/dist/types/wallet/asset-manager.d.ts +78 -0
  219. package/dist/types/wallet/asset.d.ts +21 -0
  220. package/dist/types/wallet/batch.d.ts +1 -1
  221. package/dist/types/wallet/delegator.d.ts +24 -0
  222. package/dist/types/wallet/expo/background.d.ts +66 -0
  223. package/dist/types/wallet/expo/index.d.ts +4 -0
  224. package/dist/types/wallet/expo/wallet.d.ts +97 -0
  225. package/dist/types/wallet/index.d.ts +75 -2
  226. package/dist/types/wallet/onchain.d.ts +22 -1
  227. package/dist/types/wallet/serviceWorker/wallet-message-handler.d.ts +366 -0
  228. package/dist/types/wallet/serviceWorker/wallet.d.ts +20 -11
  229. package/dist/types/wallet/utils.d.ts +13 -1
  230. package/dist/types/wallet/validation.d.ts +24 -0
  231. package/dist/types/wallet/wallet.d.ts +111 -17
  232. package/dist/types/worker/browser/service-worker-manager.d.ts +21 -0
  233. package/dist/types/{wallet/serviceWorker → worker/browser}/utils.d.ts +2 -1
  234. package/dist/types/worker/expo/asyncStorageTaskQueue.d.ts +46 -0
  235. package/dist/types/worker/expo/index.d.ts +7 -0
  236. package/dist/types/worker/expo/processors/contractPollProcessor.d.ts +14 -0
  237. package/dist/types/worker/expo/processors/index.d.ts +1 -0
  238. package/dist/types/worker/expo/taskQueue.d.ts +50 -0
  239. package/dist/types/worker/expo/taskRunner.d.ts +42 -0
  240. package/dist/types/worker/messageBus.d.ts +109 -0
  241. package/package.json +71 -17
  242. package/dist/cjs/wallet/serviceWorker/request.js +0 -78
  243. package/dist/cjs/wallet/serviceWorker/response.js +0 -222
  244. package/dist/cjs/wallet/serviceWorker/worker.js +0 -655
  245. package/dist/esm/wallet/serviceWorker/request.js +0 -75
  246. package/dist/esm/wallet/serviceWorker/response.js +0 -219
  247. package/dist/esm/wallet/serviceWorker/worker.js +0 -651
  248. package/dist/types/wallet/serviceWorker/request.d.ts +0 -74
  249. package/dist/types/wallet/serviceWorker/response.d.ts +0 -123
  250. package/dist/types/wallet/serviceWorker/worker.d.ts +0 -53
@@ -0,0 +1,86 @@
1
+ import { hex } from "@scure/base";
2
+ import { DelegateVtxo } from '../../script/delegate.js';
3
+ import { DefaultVtxo } from '../../script/default.js';
4
+ import { isCsvSpendable, sequenceToTimelock, timelockToSequence, } from './helpers.js';
5
+ /**
6
+ * Handler for delegate wallet VTXOs.
7
+ *
8
+ * Delegate contracts extend the default tapscript with an additional delegate path:
9
+ * - forfeit: (Alice + Server) multisig for collaborative spending
10
+ * - exit: (Alice) + CSV timelock for unilateral exit
11
+ * - delegate: (Alice + Delegate + Server) multisig for delegated renewal
12
+ */
13
+ export const DelegateContractHandler = {
14
+ type: "delegate",
15
+ createScript(params) {
16
+ const typed = this.deserializeParams(params);
17
+ return new DelegateVtxo.Script(typed);
18
+ },
19
+ serializeParams(params) {
20
+ return {
21
+ pubKey: hex.encode(params.pubKey),
22
+ serverPubKey: hex.encode(params.serverPubKey),
23
+ delegatePubKey: hex.encode(params.delegatePubKey),
24
+ csvTimelock: timelockToSequence(params.csvTimelock).toString(),
25
+ };
26
+ },
27
+ deserializeParams(params) {
28
+ const csvTimelock = params.csvTimelock
29
+ ? sequenceToTimelock(Number(params.csvTimelock))
30
+ : DefaultVtxo.Script.DEFAULT_TIMELOCK;
31
+ return {
32
+ pubKey: hex.decode(params.pubKey),
33
+ serverPubKey: hex.decode(params.serverPubKey),
34
+ delegatePubKey: hex.decode(params.delegatePubKey),
35
+ csvTimelock,
36
+ };
37
+ },
38
+ selectPath(script, contract, context) {
39
+ if (context.collaborative) {
40
+ return { leaf: script.forfeit() };
41
+ }
42
+ const sequence = contract.params.csvTimelock
43
+ ? Number(contract.params.csvTimelock)
44
+ : undefined;
45
+ if (!isCsvSpendable(context, sequence)) {
46
+ return null;
47
+ }
48
+ return {
49
+ leaf: script.exit(),
50
+ sequence,
51
+ };
52
+ },
53
+ getAllSpendingPaths(script, contract, context) {
54
+ const paths = [];
55
+ if (context.collaborative) {
56
+ paths.push({ leaf: script.forfeit() });
57
+ }
58
+ const exitPath = { leaf: script.exit() };
59
+ if (contract.params.csvTimelock) {
60
+ exitPath.sequence = Number(contract.params.csvTimelock);
61
+ }
62
+ paths.push(exitPath);
63
+ // Delegate path (Alice + Delegate + Server) — collaborative only
64
+ if (context.collaborative) {
65
+ paths.push({ leaf: script.delegate() });
66
+ }
67
+ return paths;
68
+ },
69
+ getSpendablePaths(script, contract, context) {
70
+ const paths = [];
71
+ if (context.collaborative) {
72
+ paths.push({ leaf: script.forfeit() });
73
+ }
74
+ const exitSequence = contract.params.csvTimelock
75
+ ? Number(contract.params.csvTimelock)
76
+ : undefined;
77
+ if (isCsvSpendable(context, exitSequence)) {
78
+ const exitPath = { leaf: script.exit() };
79
+ if (exitSequence !== undefined) {
80
+ exitPath.sequence = exitSequence;
81
+ }
82
+ paths.push(exitPath);
83
+ }
84
+ return paths;
85
+ },
86
+ };
@@ -0,0 +1,66 @@
1
+ import * as bip68 from "bip68";
2
+ /**
3
+ * Convert RelativeTimelock to BIP68 sequence number.
4
+ */
5
+ export function timelockToSequence(timelock) {
6
+ return bip68.encode(timelock.type === "blocks"
7
+ ? { blocks: Number(timelock.value) }
8
+ : { seconds: Number(timelock.value) });
9
+ }
10
+ /**
11
+ * Convert BIP68 sequence number back to RelativeTimelock.
12
+ */
13
+ export function sequenceToTimelock(sequence) {
14
+ const decoded = bip68.decode(sequence);
15
+ if ("blocks" in decoded && decoded.blocks !== undefined) {
16
+ return { type: "blocks", value: BigInt(decoded.blocks) };
17
+ }
18
+ if ("seconds" in decoded && decoded.seconds !== undefined) {
19
+ return { type: "seconds", value: BigInt(decoded.seconds) };
20
+ }
21
+ throw new Error(`Invalid BIP68 sequence: ${sequence}`);
22
+ }
23
+ /**
24
+ * Resolve wallet's role from explicit role or by matching pubkey.
25
+ */
26
+ export function resolveRole(contract, context) {
27
+ // Explicit role takes precedence
28
+ if (context.role === "sender" || context.role === "receiver") {
29
+ return context.role;
30
+ }
31
+ // Try to match wallet pubkey against contract params
32
+ if (context.walletPubKey) {
33
+ if (context.walletPubKey === contract.params.sender) {
34
+ return "sender";
35
+ }
36
+ if (context.walletPubKey === contract.params.receiver) {
37
+ return "receiver";
38
+ }
39
+ }
40
+ return undefined;
41
+ }
42
+ /**
43
+ * Check if a CSV timelock is currently satisfied for the given context/VTXO.
44
+ */
45
+ export function isCsvSpendable(context, sequence) {
46
+ if (sequence === undefined)
47
+ return true;
48
+ if (!context.vtxo)
49
+ return false;
50
+ const timelock = sequenceToTimelock(sequence);
51
+ if (timelock.type === "blocks") {
52
+ if (context.blockHeight === undefined ||
53
+ context.vtxo.status.block_height === undefined) {
54
+ return false;
55
+ }
56
+ return (context.blockHeight - context.vtxo.status.block_height >=
57
+ Number(timelock.value));
58
+ }
59
+ if (timelock.type === "seconds") {
60
+ const blockTime = context.vtxo.status.block_time;
61
+ if (blockTime === undefined)
62
+ return false;
63
+ return context.currentTime / 1000 - blockTime >= Number(timelock.value);
64
+ }
65
+ return false;
66
+ }
@@ -0,0 +1,12 @@
1
+ export { contractHandlers } from './registry.js';
2
+ export { DefaultContractHandler } from './default.js';
3
+ export { DelegateContractHandler } from './delegate.js';
4
+ export { VHTLCContractHandler } from './vhtlc.js';
5
+ // Register built-in handlers
6
+ import { contractHandlers } from './registry.js';
7
+ import { DefaultContractHandler } from './default.js';
8
+ import { DelegateContractHandler } from './delegate.js';
9
+ import { VHTLCContractHandler } from './vhtlc.js';
10
+ contractHandlers.register(DefaultContractHandler);
11
+ contractHandlers.register(DelegateContractHandler);
12
+ contractHandlers.register(VHTLCContractHandler);
@@ -0,0 +1,86 @@
1
+ /**
2
+ * Registry for contract handlers.
3
+ *
4
+ * Each contract type ("default", "vhtlc", etc.) has a handler that knows
5
+ * how to create VtxoScripts, serialize params, and select spending paths.
6
+ *
7
+ * @example
8
+ * ```typescript
9
+ * // Register a custom handler
10
+ * contractHandlers.register(myCustomHandler);
11
+ *
12
+ * // Get handler for a type
13
+ * const handler = contractHandlers.get("vhtlc");
14
+ * const script = handler.createScript(contract.params);
15
+ * ```
16
+ */
17
+ class ContractHandlerRegistry {
18
+ constructor() {
19
+ this.handlers = new Map();
20
+ }
21
+ /**
22
+ * Register a contract handler.
23
+ *
24
+ * @param handler - The handler to register
25
+ * @throws If a handler for this type is already registered
26
+ */
27
+ register(handler) {
28
+ if (this.handlers.has(handler.type)) {
29
+ throw new Error(`Contract handler for type '${handler.type}' is already registered`);
30
+ }
31
+ this.handlers.set(handler.type, handler);
32
+ }
33
+ /**
34
+ * Get a handler by type.
35
+ *
36
+ * @param type - The contract type
37
+ * @returns The handler, or undefined if not found
38
+ */
39
+ get(type) {
40
+ return this.handlers.get(type);
41
+ }
42
+ /**
43
+ * Get a handler by type, throwing if not found.
44
+ *
45
+ * @param type - The contract type
46
+ * @returns The handler
47
+ * @throws If no handler is registered for this type
48
+ */
49
+ getOrThrow(type) {
50
+ const handler = this.get(type);
51
+ if (!handler) {
52
+ throw new Error(`No contract handler registered for type '${type}'`);
53
+ }
54
+ return handler;
55
+ }
56
+ /**
57
+ * Check if a handler is registered.
58
+ *
59
+ * @param type - The contract type
60
+ */
61
+ has(type) {
62
+ return this.handlers.has(type);
63
+ }
64
+ /**
65
+ * Get all registered types.
66
+ */
67
+ getRegisteredTypes() {
68
+ return Array.from(this.handlers.keys());
69
+ }
70
+ /**
71
+ * Unregister a handler (mainly for testing).
72
+ */
73
+ unregister(type) {
74
+ return this.handlers.delete(type);
75
+ }
76
+ /**
77
+ * Clear all handlers (mainly for testing).
78
+ */
79
+ clear() {
80
+ this.handlers.clear();
81
+ }
82
+ }
83
+ /**
84
+ * Global registry of contract handlers.
85
+ */
86
+ export const contractHandlers = new ContractHandlerRegistry();
@@ -0,0 +1,190 @@
1
+ import { hex } from "@scure/base";
2
+ import { VHTLC } from '../../script/vhtlc.js';
3
+ import { isCsvSpendable, resolveRole, sequenceToTimelock, timelockToSequence, } from './helpers.js';
4
+ /**
5
+ * Handler for Virtual Hash Time Lock Contract (VHTLC).
6
+ *
7
+ * VHTLC supports multiple spending paths:
8
+ *
9
+ * Collaborative paths (with server):
10
+ * - claim: Receiver + Server with preimage
11
+ * - refund: Sender + Receiver + Server
12
+ * - refundWithoutReceiver: Sender + Server after CLTV locktime
13
+ *
14
+ * Unilateral paths (without server):
15
+ * - unilateralClaim: Receiver with preimage after CSV delay
16
+ * - unilateralRefund: Sender + Receiver after CSV delay
17
+ * - unilateralRefundWithoutReceiver: Sender after CSV delay
18
+ */
19
+ export const VHTLCContractHandler = {
20
+ type: "vhtlc",
21
+ createScript(params) {
22
+ const typed = this.deserializeParams(params);
23
+ return new VHTLC.Script(typed);
24
+ },
25
+ serializeParams(params) {
26
+ return {
27
+ sender: hex.encode(params.sender),
28
+ receiver: hex.encode(params.receiver),
29
+ server: hex.encode(params.server),
30
+ hash: hex.encode(params.preimageHash),
31
+ refundLocktime: params.refundLocktime.toString(),
32
+ claimDelay: timelockToSequence(params.unilateralClaimDelay).toString(),
33
+ refundDelay: timelockToSequence(params.unilateralRefundDelay).toString(),
34
+ refundNoReceiverDelay: timelockToSequence(params.unilateralRefundWithoutReceiverDelay).toString(),
35
+ };
36
+ },
37
+ deserializeParams(params) {
38
+ return {
39
+ sender: hex.decode(params.sender),
40
+ receiver: hex.decode(params.receiver),
41
+ server: hex.decode(params.server),
42
+ preimageHash: hex.decode(params.hash),
43
+ refundLocktime: BigInt(params.refundLocktime),
44
+ unilateralClaimDelay: sequenceToTimelock(Number(params.claimDelay)),
45
+ unilateralRefundDelay: sequenceToTimelock(Number(params.refundDelay)),
46
+ unilateralRefundWithoutReceiverDelay: sequenceToTimelock(Number(params.refundNoReceiverDelay)),
47
+ };
48
+ },
49
+ /**
50
+ * Select spending path based on context.
51
+ *
52
+ * Role is determined from `context.role` or by matching `context.walletPubKey`
53
+ * against sender/receiver in contract params.
54
+ */
55
+ selectPath(script, contract, context) {
56
+ const role = resolveRole(contract, context);
57
+ const preimage = contract.params?.preimage;
58
+ const refundLocktime = BigInt(contract.params.refundLocktime);
59
+ const currentTimeSec = Math.floor(context.currentTime / 1000);
60
+ if (!role) {
61
+ return null;
62
+ }
63
+ if (context.collaborative) {
64
+ if (role === "receiver" && preimage) {
65
+ return {
66
+ leaf: script.claim(),
67
+ extraWitness: [hex.decode(preimage)],
68
+ };
69
+ }
70
+ if (role === "sender" && BigInt(currentTimeSec) >= refundLocktime) {
71
+ return {
72
+ leaf: script.refundWithoutReceiver(),
73
+ };
74
+ }
75
+ return null;
76
+ }
77
+ // Unilateral paths
78
+ if (role === "receiver" && preimage) {
79
+ const sequence = Number(contract.params.claimDelay);
80
+ if (!isCsvSpendable(context, sequence))
81
+ return null;
82
+ return {
83
+ leaf: script.unilateralClaim(),
84
+ extraWitness: [hex.decode(preimage)],
85
+ sequence,
86
+ };
87
+ }
88
+ if (role === "sender") {
89
+ const sequence = Number(contract.params.refundNoReceiverDelay);
90
+ if (!isCsvSpendable(context, sequence))
91
+ return null;
92
+ return {
93
+ leaf: script.unilateralRefundWithoutReceiver(),
94
+ sequence,
95
+ };
96
+ }
97
+ return null;
98
+ },
99
+ /**
100
+ * Get all possible spending paths (no timelock checks).
101
+ *
102
+ * Role is determined from `context.role` or by matching `context.walletPubKey`
103
+ * against sender/receiver in contract params.
104
+ */
105
+ getAllSpendingPaths(script, contract, context) {
106
+ const role = resolveRole(contract, context);
107
+ const paths = [];
108
+ if (!role) {
109
+ return paths;
110
+ }
111
+ const preimage = contract.params?.preimage;
112
+ if (context.collaborative) {
113
+ // Collaborative paths (no timelock checks)
114
+ if (role === "receiver" && preimage) {
115
+ paths.push({
116
+ leaf: script.claim(),
117
+ extraWitness: [hex.decode(preimage)],
118
+ });
119
+ }
120
+ if (role === "sender") {
121
+ paths.push({
122
+ leaf: script.refundWithoutReceiver(),
123
+ });
124
+ }
125
+ }
126
+ else {
127
+ // Unilateral paths (no timelock checks)
128
+ if (role === "receiver" && preimage) {
129
+ const sequence = Number(contract.params.claimDelay);
130
+ paths.push({
131
+ leaf: script.unilateralClaim(),
132
+ extraWitness: [hex.decode(preimage)],
133
+ sequence,
134
+ });
135
+ }
136
+ if (role === "sender") {
137
+ const sequence = Number(contract.params.refundNoReceiverDelay);
138
+ paths.push({
139
+ leaf: script.unilateralRefundWithoutReceiver(),
140
+ sequence,
141
+ });
142
+ }
143
+ }
144
+ return paths;
145
+ },
146
+ getSpendablePaths(script, contract, context) {
147
+ const role = resolveRole(contract, context);
148
+ const paths = [];
149
+ if (!role) {
150
+ return paths;
151
+ }
152
+ const preimage = contract.params?.preimage;
153
+ const refundLocktime = BigInt(contract.params.refundLocktime);
154
+ const currentTimeSec = Math.floor(context.currentTime / 1000);
155
+ if (context.collaborative) {
156
+ if (role === "receiver" && preimage) {
157
+ paths.push({
158
+ leaf: script.claim(),
159
+ extraWitness: [hex.decode(preimage)],
160
+ });
161
+ }
162
+ if (role === "sender" && BigInt(currentTimeSec) >= refundLocktime) {
163
+ paths.push({
164
+ leaf: script.refundWithoutReceiver(),
165
+ });
166
+ }
167
+ return paths;
168
+ }
169
+ if (role === "receiver" && preimage) {
170
+ const sequence = Number(contract.params.claimDelay);
171
+ if (isCsvSpendable(context, sequence)) {
172
+ paths.push({
173
+ leaf: script.unilateralClaim(),
174
+ extraWitness: [hex.decode(preimage)],
175
+ sequence,
176
+ });
177
+ }
178
+ }
179
+ if (role === "sender") {
180
+ const sequence = Number(contract.params.refundNoReceiverDelay);
181
+ if (isCsvSpendable(context, sequence)) {
182
+ paths.push({
183
+ leaf: script.unilateralRefundWithoutReceiver(),
184
+ sequence,
185
+ });
186
+ }
187
+ }
188
+ return paths;
189
+ },
190
+ };
@@ -0,0 +1,13 @@
1
+ // Types
2
+ export * from './types.js';
3
+ // Contract handlers
4
+ export { contractHandlers } from './handlers/index.js';
5
+ export { DefaultContractHandler } from './handlers/index.js';
6
+ export { DelegateContractHandler } from './handlers/index.js';
7
+ export { VHTLCContractHandler } from './handlers/index.js';
8
+ // arkcontract string codec
9
+ export { encodeArkContract, decodeArkContract, contractFromArkContract, contractFromArkContractWithAddress, isArkContract, } from './arkcontract.js';
10
+ // Contract watcher
11
+ export { ContractWatcher } from './contractWatcher.js';
12
+ // Contract manager
13
+ export { ContractManager } from './contractManager.js';
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,92 @@
1
+ export function getGlobalObject() {
2
+ if (typeof globalThis !== "undefined") {
3
+ if (typeof globalThis.self === "object" && globalThis.self !== null) {
4
+ return { globalObject: globalThis.self };
5
+ }
6
+ if (typeof globalThis.window === "object" &&
7
+ globalThis.window !== null) {
8
+ return { globalObject: globalThis.window };
9
+ }
10
+ return { globalObject: globalThis };
11
+ }
12
+ throw new Error("Global object not found");
13
+ }
14
+ // database instance cache, avoiding multiple open requests
15
+ const dbCache = new Map();
16
+ // track reference counts for each database to avoid closing it prematurely
17
+ const refCounts = new Map();
18
+ /**
19
+ * Opens an IndexedDB database and increments the reference count.
20
+ * Handles global object detection and callbacks.
21
+ *
22
+ * @param dbName The name of the database to open.
23
+ * @param dbVersion The database version to open.
24
+ * @param initDatabase A function that migrates the database schema, called on `onupgradeneeded` only.
25
+ *
26
+ * @returns A promise that resolves to the database instance.
27
+ */
28
+ export async function openDatabase(dbName, dbVersion, initDatabase) {
29
+ const { globalObject } = getGlobalObject();
30
+ if (!globalObject.indexedDB) {
31
+ throw new Error("IndexedDB is not available in this environment");
32
+ }
33
+ // Return cached promise if available (handles concurrent calls)
34
+ const cached = dbCache.get(dbName);
35
+ if (cached) {
36
+ if (cached.version !== dbVersion) {
37
+ throw new Error(`Database "${dbName}" already opened with version ${cached.version}; requested ${dbVersion}`);
38
+ }
39
+ refCounts.set(dbName, (refCounts.get(dbName) ?? 0) + 1);
40
+ return cached.promise;
41
+ }
42
+ const dbPromise = new Promise((resolve, reject) => {
43
+ const request = globalObject.indexedDB.open(dbName, dbVersion);
44
+ request.onerror = () => {
45
+ dbCache.delete(dbName); // Clean up on failure
46
+ refCounts.delete(dbName);
47
+ reject(request.error);
48
+ };
49
+ request.onsuccess = () => {
50
+ resolve(request.result);
51
+ };
52
+ request.onupgradeneeded = () => {
53
+ const db = request.result;
54
+ initDatabase(db);
55
+ };
56
+ request.onblocked = () => {
57
+ console.warn("Database upgrade blocked - close other tabs/connections");
58
+ };
59
+ });
60
+ // Cache immediately before awaiting
61
+ dbCache.set(dbName, { version: dbVersion, promise: dbPromise });
62
+ refCounts.set(dbName, 1);
63
+ return dbPromise;
64
+ }
65
+ /**
66
+ * Decrements the reference count and closes the database when no references remain.
67
+ *
68
+ * @param dbName The name of the database to close.
69
+ *
70
+ * @returns True if the database was closed, false otherwise.
71
+ */
72
+ export async function closeDatabase(dbName) {
73
+ const cachedEntry = dbCache.get(dbName);
74
+ if (!cachedEntry)
75
+ return false;
76
+ const count = (refCounts.get(dbName) ?? 1) - 1;
77
+ if (count > 0) {
78
+ refCounts.set(dbName, count);
79
+ return false;
80
+ }
81
+ // Last reference — actually close
82
+ refCounts.delete(dbName);
83
+ dbCache.delete(dbName);
84
+ try {
85
+ const db = await cachedEntry.promise;
86
+ db.close();
87
+ }
88
+ catch {
89
+ // DB failed to open, nothing to close
90
+ }
91
+ return true;
92
+ }
@@ -1,24 +1,27 @@
1
1
  import { Transaction } from './utils/transaction.js';
2
2
  import { P2A } from './utils/anchor.js';
3
3
  export function buildForfeitTx(inputs, forfeitPkScript, txLocktime) {
4
- const tx = new Transaction({
5
- version: 3,
6
- lockTime: txLocktime,
7
- });
8
4
  let amount = 0n;
9
5
  for (const input of inputs) {
10
6
  if (!input.witnessUtxo) {
11
7
  throw new Error("input needs witness utxo");
12
8
  }
13
9
  amount += input.witnessUtxo.amount;
14
- tx.addInput(input);
15
10
  }
16
- // Add main output to server
17
- tx.addOutput({
11
+ return buildForfeitTxWithOutput(inputs, {
18
12
  script: forfeitPkScript,
19
13
  amount,
14
+ }, txLocktime);
15
+ }
16
+ export function buildForfeitTxWithOutput(inputs, output, txLocktime) {
17
+ const tx = new Transaction({
18
+ version: 3,
19
+ lockTime: txLocktime,
20
20
  });
21
- // Add P2A output
21
+ for (const input of inputs) {
22
+ tx.addInput(input);
23
+ }
24
+ tx.addOutput(output);
22
25
  tx.addOutput(P2A);
23
26
  return tx;
24
27
  }
@@ -1 +1,2 @@
1
1
  export * from './singleKey.js';
2
+ export * from './seedIdentity.js';