@arkade-os/sdk 0.3.13 → 0.4.0-next.1

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 (270) hide show
  1. package/README.md +586 -54
  2. package/dist/cjs/asset/assetGroup.js +141 -0
  3. package/dist/cjs/asset/assetId.js +88 -0
  4. package/dist/cjs/asset/assetInput.js +204 -0
  5. package/dist/cjs/asset/assetOutput.js +159 -0
  6. package/dist/cjs/asset/assetRef.js +82 -0
  7. package/dist/cjs/asset/index.js +24 -0
  8. package/dist/cjs/asset/metadata.js +172 -0
  9. package/dist/cjs/asset/packet.js +164 -0
  10. package/dist/cjs/asset/types.js +25 -0
  11. package/dist/cjs/asset/utils.js +105 -0
  12. package/dist/cjs/bip322/index.js +270 -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/forfeit.js +12 -8
  25. package/dist/cjs/identity/index.js +1 -0
  26. package/dist/cjs/identity/seedIdentity.js +255 -0
  27. package/dist/cjs/index.js +72 -14
  28. package/dist/cjs/intent/index.js +47 -11
  29. package/dist/cjs/providers/ark.js +7 -0
  30. package/dist/cjs/providers/delegator.js +66 -0
  31. package/dist/cjs/providers/expoIndexer.js +5 -0
  32. package/dist/cjs/providers/indexer.js +68 -1
  33. package/dist/cjs/providers/utils.js +1 -0
  34. package/dist/cjs/repositories/contractRepository.js +0 -103
  35. package/dist/cjs/repositories/inMemory/contractRepository.js +55 -0
  36. package/dist/cjs/repositories/inMemory/walletRepository.js +80 -0
  37. package/dist/cjs/repositories/index.js +16 -0
  38. package/dist/cjs/repositories/indexedDB/contractRepository.js +187 -0
  39. package/dist/cjs/repositories/indexedDB/db.js +19 -0
  40. package/dist/cjs/repositories/indexedDB/manager.js +97 -0
  41. package/dist/cjs/repositories/indexedDB/schema.js +159 -0
  42. package/dist/cjs/repositories/indexedDB/walletRepository.js +338 -0
  43. package/dist/cjs/repositories/indexedDB/websqlAdapter.js +144 -0
  44. package/dist/cjs/repositories/migrations/contractRepositoryImpl.js +127 -0
  45. package/dist/cjs/repositories/migrations/fromStorageAdapter.js +66 -0
  46. package/dist/cjs/repositories/migrations/walletRepositoryImpl.js +180 -0
  47. package/dist/cjs/repositories/realm/contractRepository.js +120 -0
  48. package/dist/cjs/repositories/realm/index.js +9 -0
  49. package/dist/cjs/repositories/realm/schemas.js +108 -0
  50. package/dist/cjs/repositories/realm/types.js +7 -0
  51. package/dist/cjs/repositories/realm/walletRepository.js +273 -0
  52. package/dist/cjs/repositories/serialization.js +49 -0
  53. package/dist/cjs/repositories/sqlite/contractRepository.js +139 -0
  54. package/dist/cjs/repositories/sqlite/index.js +7 -0
  55. package/dist/cjs/repositories/sqlite/types.js +2 -0
  56. package/dist/cjs/repositories/sqlite/walletRepository.js +328 -0
  57. package/dist/cjs/repositories/walletRepository.js +0 -169
  58. package/dist/cjs/script/base.js +54 -0
  59. package/dist/cjs/script/delegate.js +49 -0
  60. package/dist/cjs/storage/asyncStorage.js +4 -1
  61. package/dist/cjs/storage/fileSystem.js +3 -0
  62. package/dist/cjs/storage/inMemory.js +3 -0
  63. package/dist/cjs/storage/indexedDB.js +5 -1
  64. package/dist/cjs/storage/localStorage.js +3 -0
  65. package/dist/cjs/utils/arkTransaction.js +16 -0
  66. package/dist/cjs/utils/transactionHistory.js +50 -0
  67. package/dist/cjs/wallet/asset-manager.js +338 -0
  68. package/dist/cjs/wallet/asset.js +117 -0
  69. package/dist/cjs/wallet/batch.js +1 -1
  70. package/dist/cjs/wallet/delegator.js +235 -0
  71. package/dist/cjs/wallet/expo/background.js +133 -0
  72. package/dist/cjs/wallet/expo/index.js +9 -0
  73. package/dist/cjs/wallet/expo/wallet.js +231 -0
  74. package/dist/cjs/wallet/serviceWorker/wallet-message-handler.js +568 -0
  75. package/dist/cjs/wallet/serviceWorker/wallet.js +383 -102
  76. package/dist/cjs/wallet/utils.js +58 -0
  77. package/dist/cjs/wallet/validation.js +151 -0
  78. package/dist/cjs/wallet/vtxo-manager.js +8 -1
  79. package/dist/cjs/wallet/wallet.js +702 -260
  80. package/dist/cjs/worker/browser/service-worker-manager.js +82 -0
  81. package/dist/cjs/{wallet/serviceWorker → worker/browser}/utils.js +2 -1
  82. package/dist/cjs/worker/expo/asyncStorageTaskQueue.js +78 -0
  83. package/dist/cjs/worker/expo/index.js +12 -0
  84. package/dist/cjs/worker/expo/processors/contractPollProcessor.js +61 -0
  85. package/dist/cjs/worker/expo/processors/index.js +6 -0
  86. package/dist/cjs/worker/expo/taskQueue.js +41 -0
  87. package/dist/cjs/worker/expo/taskRunner.js +57 -0
  88. package/dist/cjs/worker/messageBus.js +252 -0
  89. package/dist/esm/asset/assetGroup.js +137 -0
  90. package/dist/esm/asset/assetId.js +84 -0
  91. package/dist/esm/asset/assetInput.js +199 -0
  92. package/dist/esm/asset/assetOutput.js +154 -0
  93. package/dist/esm/asset/assetRef.js +78 -0
  94. package/dist/esm/asset/index.js +8 -0
  95. package/dist/esm/asset/metadata.js +167 -0
  96. package/dist/esm/asset/packet.js +159 -0
  97. package/dist/esm/asset/types.js +22 -0
  98. package/dist/esm/asset/utils.js +99 -0
  99. package/dist/esm/bip322/index.js +267 -0
  100. package/dist/esm/contracts/arkcontract.js +141 -0
  101. package/dist/esm/contracts/contractManager.js +432 -0
  102. package/dist/esm/contracts/contractWatcher.js +563 -0
  103. package/dist/esm/contracts/handlers/default.js +82 -0
  104. package/dist/esm/contracts/handlers/delegate.js +86 -0
  105. package/dist/esm/contracts/handlers/helpers.js +66 -0
  106. package/dist/esm/contracts/handlers/index.js +12 -0
  107. package/dist/esm/contracts/handlers/registry.js +86 -0
  108. package/dist/esm/contracts/handlers/vhtlc.js +190 -0
  109. package/dist/esm/contracts/index.js +13 -0
  110. package/dist/esm/contracts/types.js +1 -0
  111. package/dist/esm/forfeit.js +11 -8
  112. package/dist/esm/identity/index.js +1 -0
  113. package/dist/esm/identity/seedIdentity.js +249 -0
  114. package/dist/esm/index.js +28 -15
  115. package/dist/esm/intent/index.js +44 -9
  116. package/dist/esm/providers/ark.js +7 -0
  117. package/dist/esm/providers/delegator.js +62 -0
  118. package/dist/esm/providers/expoIndexer.js +5 -0
  119. package/dist/esm/providers/indexer.js +68 -1
  120. package/dist/esm/providers/utils.js +1 -0
  121. package/dist/esm/repositories/contractRepository.js +1 -101
  122. package/dist/esm/repositories/inMemory/contractRepository.js +51 -0
  123. package/dist/esm/repositories/inMemory/walletRepository.js +76 -0
  124. package/dist/esm/repositories/index.js +8 -0
  125. package/dist/esm/repositories/indexedDB/contractRepository.js +183 -0
  126. package/dist/esm/repositories/indexedDB/db.js +4 -0
  127. package/dist/esm/repositories/indexedDB/manager.js +92 -0
  128. package/dist/esm/repositories/indexedDB/schema.js +155 -0
  129. package/dist/esm/repositories/indexedDB/walletRepository.js +334 -0
  130. package/dist/esm/repositories/indexedDB/websqlAdapter.js +138 -0
  131. package/dist/esm/repositories/migrations/contractRepositoryImpl.js +121 -0
  132. package/dist/esm/repositories/migrations/fromStorageAdapter.js +58 -0
  133. package/dist/esm/repositories/migrations/walletRepositoryImpl.js +176 -0
  134. package/dist/esm/repositories/realm/contractRepository.js +116 -0
  135. package/dist/esm/repositories/realm/index.js +3 -0
  136. package/dist/esm/repositories/realm/schemas.js +105 -0
  137. package/dist/esm/repositories/realm/types.js +6 -0
  138. package/dist/esm/repositories/realm/walletRepository.js +269 -0
  139. package/dist/esm/repositories/serialization.js +40 -0
  140. package/dist/esm/repositories/sqlite/contractRepository.js +135 -0
  141. package/dist/esm/repositories/sqlite/index.js +2 -0
  142. package/dist/esm/repositories/sqlite/types.js +1 -0
  143. package/dist/esm/repositories/sqlite/walletRepository.js +324 -0
  144. package/dist/esm/repositories/walletRepository.js +1 -167
  145. package/dist/esm/script/base.js +21 -1
  146. package/dist/esm/script/delegate.js +46 -0
  147. package/dist/esm/storage/asyncStorage.js +4 -1
  148. package/dist/esm/storage/fileSystem.js +3 -0
  149. package/dist/esm/storage/inMemory.js +3 -0
  150. package/dist/esm/storage/indexedDB.js +5 -1
  151. package/dist/esm/storage/localStorage.js +3 -0
  152. package/dist/esm/utils/arkTransaction.js +15 -0
  153. package/dist/esm/utils/transactionHistory.js +50 -0
  154. package/dist/esm/wallet/asset-manager.js +333 -0
  155. package/dist/esm/wallet/asset.js +111 -0
  156. package/dist/esm/wallet/batch.js +1 -1
  157. package/dist/esm/wallet/delegator.js +231 -0
  158. package/dist/esm/wallet/expo/background.js +128 -0
  159. package/dist/esm/wallet/expo/index.js +2 -0
  160. package/dist/esm/wallet/expo/wallet.js +194 -0
  161. package/dist/esm/wallet/serviceWorker/wallet-message-handler.js +564 -0
  162. package/dist/esm/wallet/serviceWorker/wallet.js +382 -101
  163. package/dist/esm/wallet/utils.js +54 -0
  164. package/dist/esm/wallet/validation.js +139 -0
  165. package/dist/esm/wallet/vtxo-manager.js +8 -1
  166. package/dist/esm/wallet/wallet.js +704 -229
  167. package/dist/esm/worker/browser/service-worker-manager.js +76 -0
  168. package/dist/esm/{wallet/serviceWorker → worker/browser}/utils.js +2 -1
  169. package/dist/esm/worker/expo/asyncStorageTaskQueue.js +74 -0
  170. package/dist/esm/worker/expo/index.js +4 -0
  171. package/dist/esm/worker/expo/processors/contractPollProcessor.js +58 -0
  172. package/dist/esm/worker/expo/processors/index.js +1 -0
  173. package/dist/esm/worker/expo/taskQueue.js +37 -0
  174. package/dist/esm/worker/expo/taskRunner.js +54 -0
  175. package/dist/esm/worker/messageBus.js +248 -0
  176. package/dist/types/asset/assetGroup.d.ts +28 -0
  177. package/dist/types/asset/assetId.d.ts +19 -0
  178. package/dist/types/asset/assetInput.d.ts +46 -0
  179. package/dist/types/asset/assetOutput.d.ts +39 -0
  180. package/dist/types/asset/assetRef.d.ts +25 -0
  181. package/dist/types/asset/index.d.ts +8 -0
  182. package/dist/types/asset/metadata.d.ts +37 -0
  183. package/dist/types/asset/packet.d.ts +27 -0
  184. package/dist/types/asset/types.d.ts +18 -0
  185. package/dist/types/asset/utils.d.ts +21 -0
  186. package/dist/types/bip322/index.d.ts +55 -0
  187. package/dist/types/contracts/arkcontract.d.ts +101 -0
  188. package/dist/types/contracts/contractManager.d.ts +331 -0
  189. package/dist/types/contracts/contractWatcher.d.ts +192 -0
  190. package/dist/types/contracts/handlers/default.d.ts +19 -0
  191. package/dist/types/contracts/handlers/delegate.d.ts +21 -0
  192. package/dist/types/contracts/handlers/helpers.d.ts +18 -0
  193. package/dist/types/contracts/handlers/index.d.ts +7 -0
  194. package/dist/types/contracts/handlers/registry.d.ts +65 -0
  195. package/dist/types/contracts/handlers/vhtlc.d.ts +32 -0
  196. package/dist/types/contracts/index.d.ts +14 -0
  197. package/dist/types/contracts/types.d.ts +222 -0
  198. package/dist/types/forfeit.d.ts +2 -1
  199. package/dist/types/identity/index.d.ts +1 -0
  200. package/dist/types/identity/seedIdentity.d.ts +128 -0
  201. package/dist/types/index.d.ts +22 -12
  202. package/dist/types/intent/index.d.ts +15 -1
  203. package/dist/types/providers/ark.d.ts +11 -2
  204. package/dist/types/providers/delegator.d.ts +29 -0
  205. package/dist/types/providers/indexer.d.ts +11 -1
  206. package/dist/types/repositories/contractRepository.d.ts +30 -19
  207. package/dist/types/repositories/inMemory/contractRepository.d.ts +17 -0
  208. package/dist/types/repositories/inMemory/walletRepository.d.ts +26 -0
  209. package/dist/types/repositories/index.d.ts +7 -0
  210. package/dist/types/repositories/indexedDB/contractRepository.d.ts +21 -0
  211. package/dist/types/repositories/indexedDB/db.d.ts +4 -0
  212. package/dist/types/repositories/indexedDB/manager.d.ts +22 -0
  213. package/dist/types/repositories/indexedDB/schema.d.ts +8 -0
  214. package/dist/types/repositories/indexedDB/walletRepository.d.ts +25 -0
  215. package/dist/types/repositories/indexedDB/websqlAdapter.d.ts +49 -0
  216. package/dist/types/repositories/migrations/contractRepositoryImpl.d.ts +24 -0
  217. package/dist/types/repositories/migrations/fromStorageAdapter.d.ts +19 -0
  218. package/dist/types/repositories/migrations/walletRepositoryImpl.d.ts +27 -0
  219. package/dist/types/repositories/realm/contractRepository.d.ts +24 -0
  220. package/dist/types/repositories/realm/index.d.ts +4 -0
  221. package/dist/types/repositories/realm/schemas.d.ts +208 -0
  222. package/dist/types/repositories/realm/types.d.ts +16 -0
  223. package/dist/types/repositories/realm/walletRepository.d.ts +31 -0
  224. package/dist/types/repositories/serialization.d.ts +40 -0
  225. package/dist/types/repositories/sqlite/contractRepository.d.ts +33 -0
  226. package/dist/types/repositories/sqlite/index.d.ts +3 -0
  227. package/dist/types/repositories/sqlite/types.d.ts +18 -0
  228. package/dist/types/repositories/sqlite/walletRepository.d.ts +40 -0
  229. package/dist/types/repositories/walletRepository.d.ts +13 -24
  230. package/dist/types/script/base.d.ts +1 -0
  231. package/dist/types/script/delegate.d.ts +36 -0
  232. package/dist/types/storage/asyncStorage.d.ts +4 -0
  233. package/dist/types/storage/fileSystem.d.ts +3 -0
  234. package/dist/types/storage/inMemory.d.ts +3 -0
  235. package/dist/types/storage/index.d.ts +3 -0
  236. package/dist/types/storage/indexedDB.d.ts +3 -0
  237. package/dist/types/storage/localStorage.d.ts +3 -0
  238. package/dist/types/utils/arkTransaction.d.ts +6 -0
  239. package/dist/types/wallet/asset-manager.d.ts +78 -0
  240. package/dist/types/wallet/asset.d.ts +21 -0
  241. package/dist/types/wallet/batch.d.ts +1 -1
  242. package/dist/types/wallet/delegator.d.ts +24 -0
  243. package/dist/types/wallet/expo/background.d.ts +66 -0
  244. package/dist/types/wallet/expo/index.d.ts +4 -0
  245. package/dist/types/wallet/expo/wallet.d.ts +97 -0
  246. package/dist/types/wallet/index.d.ts +75 -2
  247. package/dist/types/wallet/serviceWorker/wallet-message-handler.d.ts +366 -0
  248. package/dist/types/wallet/serviceWorker/wallet.d.ts +20 -11
  249. package/dist/types/wallet/utils.d.ts +12 -1
  250. package/dist/types/wallet/validation.d.ts +24 -0
  251. package/dist/types/wallet/wallet.d.ts +111 -17
  252. package/dist/types/worker/browser/service-worker-manager.d.ts +21 -0
  253. package/dist/types/{wallet/serviceWorker → worker/browser}/utils.d.ts +2 -1
  254. package/dist/types/worker/expo/asyncStorageTaskQueue.d.ts +46 -0
  255. package/dist/types/worker/expo/index.d.ts +7 -0
  256. package/dist/types/worker/expo/processors/contractPollProcessor.d.ts +14 -0
  257. package/dist/types/worker/expo/processors/index.d.ts +1 -0
  258. package/dist/types/worker/expo/taskQueue.d.ts +50 -0
  259. package/dist/types/worker/expo/taskRunner.d.ts +42 -0
  260. package/dist/types/worker/messageBus.d.ts +109 -0
  261. package/package.json +69 -11
  262. package/dist/cjs/wallet/serviceWorker/request.js +0 -78
  263. package/dist/cjs/wallet/serviceWorker/response.js +0 -222
  264. package/dist/cjs/wallet/serviceWorker/worker.js +0 -655
  265. package/dist/esm/wallet/serviceWorker/request.js +0 -75
  266. package/dist/esm/wallet/serviceWorker/response.js +0 -219
  267. package/dist/esm/wallet/serviceWorker/worker.js +0 -651
  268. package/dist/types/wallet/serviceWorker/request.d.ts +0 -74
  269. package/dist/types/wallet/serviceWorker/response.d.ts +0 -123
  270. package/dist/types/wallet/serviceWorker/worker.d.ts +0 -53
@@ -3,6 +3,13 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.DUST_AMOUNT = void 0;
4
4
  exports.extendVirtualCoin = extendVirtualCoin;
5
5
  exports.extendCoin = extendCoin;
6
+ exports.extendVtxoFromContract = extendVtxoFromContract;
7
+ exports.getRandomId = getRandomId;
8
+ exports.isValidArkAddress = isValidArkAddress;
9
+ exports.validateRecipients = validateRecipients;
10
+ const __1 = require("..");
11
+ const handlers_1 = require("../contracts/handlers");
12
+ const base_1 = require("@scure/base");
6
13
  exports.DUST_AMOUNT = 546; // sats
7
14
  function extendVirtualCoin(wallet, vtxo) {
8
15
  return {
@@ -20,3 +27,54 @@ function extendCoin(wallet, utxo) {
20
27
  tapTree: wallet.boardingTapscript.encode(),
21
28
  };
22
29
  }
30
+ function extendVtxoFromContract(vtxo, contract) {
31
+ const handler = handlers_1.contractHandlers.get(contract.type);
32
+ if (!handler) {
33
+ throw new Error(`No handler for contract type '${contract.type}'`);
34
+ }
35
+ const script = handler.createScript(contract.params);
36
+ return {
37
+ ...vtxo,
38
+ forfeitTapLeafScript: script.forfeit(),
39
+ intentTapLeafScript: script.forfeit(),
40
+ tapTree: script.encode(),
41
+ };
42
+ }
43
+ function getRandomId() {
44
+ const randomValue = crypto.getRandomValues(new Uint8Array(16));
45
+ return base_1.hex.encode(randomValue);
46
+ }
47
+ function isValidArkAddress(address) {
48
+ try {
49
+ __1.ArkAddress.decode(address);
50
+ return true;
51
+ }
52
+ catch (e) {
53
+ return false;
54
+ }
55
+ }
56
+ function validateRecipients(recipients, dustAmount) {
57
+ const validatedRecipients = [];
58
+ for (const recipient of recipients) {
59
+ let address;
60
+ try {
61
+ address = __1.ArkAddress.decode(recipient.address);
62
+ }
63
+ catch (e) {
64
+ throw new Error(`Invalid Ark address: ${recipient.address}`);
65
+ }
66
+ const amount = recipient.amount || dustAmount;
67
+ if (amount <= 0) {
68
+ throw new Error("Amount must be positive");
69
+ }
70
+ validatedRecipients.push({
71
+ address: recipient.address,
72
+ assets: recipient.assets ?? [],
73
+ amount,
74
+ script: amount < dustAmount
75
+ ? address.subdustPkScript
76
+ : address.pkScript,
77
+ });
78
+ }
79
+ return validatedRecipients;
80
+ }
@@ -0,0 +1,151 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ErrInvalidOffchainOutputAmount = exports.ErrOnchainOutputNotFound = exports.ErrInvalidOnchainOutputAssets = exports.ErrInvalidOnchainOutputAmount = exports.ErrAssetOutputNotFound = exports.ErrAssetGroupNotFound = exports.ErrInvalidAssetOutputAmount = exports.ErrOffchainOutputNotFound = void 0;
4
+ exports.validateBatchRecipients = validateBatchRecipients;
5
+ const utils_js_1 = require("@scure/btc-signer/utils.js");
6
+ const address_1 = require("../script/address");
7
+ const asset_1 = require("../asset");
8
+ const btc_signer_1 = require("@scure/btc-signer");
9
+ const ErrOffchainOutputNotFound = (address) => new Error(`offchain send output not found: ${address}`);
10
+ exports.ErrOffchainOutputNotFound = ErrOffchainOutputNotFound;
11
+ const ErrInvalidAssetOutputAmount = (got, want, assetId) => new Error(`invalid asset output amount for ${assetId}: got ${got}, want ${want}`);
12
+ exports.ErrInvalidAssetOutputAmount = ErrInvalidAssetOutputAmount;
13
+ const ErrAssetGroupNotFound = (assetId) => new Error(`asset group not found in batch leaf: ${assetId}`);
14
+ exports.ErrAssetGroupNotFound = ErrAssetGroupNotFound;
15
+ const ErrAssetOutputNotFound = (assetId, outputIndex) => new Error(`asset output not found in asset group ${assetId} at index ${outputIndex}`);
16
+ exports.ErrAssetOutputNotFound = ErrAssetOutputNotFound;
17
+ const ErrInvalidOnchainOutputAmount = (address) => new Error(`invalid onchain output amount: ${address}`);
18
+ exports.ErrInvalidOnchainOutputAmount = ErrInvalidOnchainOutputAmount;
19
+ const ErrInvalidOnchainOutputAssets = (address) => new Error(`onchain output ${address} cannot have assets`);
20
+ exports.ErrInvalidOnchainOutputAssets = ErrInvalidOnchainOutputAssets;
21
+ const ErrOnchainOutputNotFound = (address) => new Error(`onchain output not found: ${address}`);
22
+ exports.ErrOnchainOutputNotFound = ErrOnchainOutputNotFound;
23
+ const ErrInvalidOffchainOutputAmount = (address) => new Error(`invalid offchain output ${address}, missing amount`);
24
+ exports.ErrInvalidOffchainOutputAmount = ErrInvalidOffchainOutputAmount;
25
+ /**
26
+ * Validates both offchain and onchain recipients.
27
+ * Offchain recipients are checked against vtxo tree leaves for correct amounts and assets.
28
+ * Onchain recipients are validated against the round transaction outputs (amounts and scripts)
29
+ * via validateOnchainRecipient.
30
+ *
31
+ * @param commitmentTx - The commitment transaction to validate against
32
+ * @param vtxoTreeLeaves - The vtxo tree leaves to validate against
33
+ * @param recipients - The expected recipients to validate (both offchain and onchain)
34
+ * @param network - Network for decoding onchain addresses (e.g. mainnet, testnet)
35
+ * @throws {Error} if a recipient is not present or invalid in the vtxo tree or commitment tx
36
+ */
37
+ function validateBatchRecipients(commitmentTx, vtxoTreeLeaves, recipients, network) {
38
+ // usedOutputs is used to track which outputs are validated to handle
39
+ // duplicate recipients in the list
40
+ const usedOutputs = new Set();
41
+ const usedOnchainOutputs = new Set();
42
+ for (const recipient of recipients) {
43
+ let arkAddress;
44
+ try {
45
+ arkAddress = address_1.ArkAddress.decode(recipient.address);
46
+ }
47
+ catch {
48
+ validateOnchainRecipient(commitmentTx, recipient, network, usedOnchainOutputs);
49
+ continue;
50
+ }
51
+ validateOffchainRecipient(vtxoTreeLeaves, arkAddress, recipient, usedOutputs);
52
+ }
53
+ }
54
+ // validateOnchainRecipient verifies the given recipient is present in the commitment tx outputs list
55
+ function validateOnchainRecipient(commitmentTx, recipient, network, usedOutputs) {
56
+ const addr = (0, btc_signer_1.Address)(network).decode(recipient.address);
57
+ const expectedPkScript = btc_signer_1.OutScript.encode(addr);
58
+ if (!recipient.amount) {
59
+ throw (0, exports.ErrInvalidOnchainOutputAmount)(recipient.address);
60
+ }
61
+ if (recipient.assets && recipient.assets.length > 0) {
62
+ throw (0, exports.ErrInvalidOnchainOutputAssets)(recipient.address);
63
+ }
64
+ for (let i = 0; i < commitmentTx.outputsLength; i++) {
65
+ if (usedOutputs.has(i)) {
66
+ continue;
67
+ }
68
+ const output = commitmentTx.getOutput(i);
69
+ if (!output?.script || output.script.length === 0) {
70
+ continue;
71
+ }
72
+ if ((0, utils_js_1.equalBytes)(output.script, expectedPkScript)) {
73
+ if (output.amount !== BigInt(recipient.amount)) {
74
+ continue; // if amount does not match, continue
75
+ }
76
+ // we found the right output, recipient is valid, return
77
+ usedOutputs.add(i);
78
+ return;
79
+ }
80
+ }
81
+ // if we get here, the recipient is not present in the commitment tx outputs list
82
+ throw (0, exports.ErrOnchainOutputNotFound)(recipient.address);
83
+ }
84
+ // validate the offchain recipient is present in one of the leaf output
85
+ // also verify the asset packet is here, and point the same output index
86
+ function validateOffchainRecipient(leaves, arkAddress, recipient, usedOutputs // leafIndex:outputIndex
87
+ ) {
88
+ const expectedPkScript = arkAddress.pkScript;
89
+ if (!recipient.amount) {
90
+ throw (0, exports.ErrInvalidOffchainOutputAmount)(recipient.address);
91
+ }
92
+ const expectedAmount = BigInt(recipient.amount);
93
+ let found = false;
94
+ for (let leafIdx = 0; leafIdx < leaves.length; leafIdx++) {
95
+ const leaf = leaves[leafIdx];
96
+ for (let outputIndex = 0; outputIndex < leaf.outputsLength; outputIndex++) {
97
+ const output = leaf.getOutput(outputIndex);
98
+ if (!output?.script || output.script.length === 0) {
99
+ continue;
100
+ }
101
+ if (!(0, utils_js_1.equalBytes)(output.script, expectedPkScript)) {
102
+ continue;
103
+ }
104
+ if (output.amount !== expectedAmount) {
105
+ continue;
106
+ }
107
+ const key = `${leafIdx}:${outputIndex}`;
108
+ if (usedOutputs.has(key)) {
109
+ continue;
110
+ }
111
+ usedOutputs.add(key);
112
+ found = true;
113
+ // if assets, validate the asset packet
114
+ if (recipient.assets && recipient.assets.length > 0) {
115
+ validateAssetOutputs(leaf, outputIndex, recipient.assets);
116
+ }
117
+ break;
118
+ }
119
+ if (found) {
120
+ break;
121
+ }
122
+ }
123
+ if (!found) {
124
+ throw (0, exports.ErrOffchainOutputNotFound)(recipient.address);
125
+ }
126
+ }
127
+ function validateAssetOutputs(leafTx, outputIndex, expectedAssets) {
128
+ const assetPacket = asset_1.Packet.fromTx(leafTx);
129
+ for (const { assetId, amount } of expectedAssets) {
130
+ validateAssetGroupOutput(assetPacket, outputIndex, assetId, amount);
131
+ }
132
+ }
133
+ function validateAssetGroupOutput(packet, outputIndex, assetId, expectedAmount) {
134
+ const assetGroup = packet.groups.find((group) => {
135
+ if (group.isIssuance())
136
+ return false;
137
+ return group.assetId.toString() === assetId;
138
+ });
139
+ if (!assetGroup) {
140
+ throw (0, exports.ErrAssetGroupNotFound)(assetId);
141
+ }
142
+ // find the output at the expected index
143
+ const assetOutput = assetGroup.outputs.find((output) => output.vout === outputIndex);
144
+ if (!assetOutput) {
145
+ throw (0, exports.ErrAssetOutputNotFound)(assetId, outputIndex);
146
+ }
147
+ const expectedAmountBigInt = BigInt(expectedAmount);
148
+ if (assetOutput.amount !== expectedAmountBigInt) {
149
+ throw (0, exports.ErrInvalidAssetOutputAmount)(assetOutput.amount, expectedAmountBigInt, assetId);
150
+ }
151
+ }
@@ -93,6 +93,13 @@ function isVtxoExpiringSoon(vtxo, thresholdMs // in milliseconds
93
93
  const { batchExpiry } = vtxo.virtualStatus;
94
94
  if (!batchExpiry)
95
95
  return false; // it doesn't expire
96
+ // we use this as a workaround to avoid issue on regtest where expiry date is
97
+ // expressed in blockheight instead of timestamp. If expiry, as Date, is before 2025,
98
+ // then we admit it's too small to be a timestamp
99
+ // TODO: API should return the expiry unit
100
+ const expireAt = new Date(batchExpiry);
101
+ if (expireAt.getFullYear() < 2025)
102
+ return false;
96
103
  const now = Date.now();
97
104
  if (batchExpiry <= now)
98
105
  return false; // already expired
@@ -193,7 +200,7 @@ class VtxoManager {
193
200
  // Get dust amount from wallet
194
201
  const dustAmount = getDustAmount(this.wallet);
195
202
  // Filter recoverable VTXOs and handle subdust logic
196
- const { vtxosToRecover, includesSubdust, totalAmount } = getRecoverableWithSubdust(allVtxos, dustAmount);
203
+ const { vtxosToRecover, totalAmount } = getRecoverableWithSubdust(allVtxos, dustAmount);
197
204
  if (vtxosToRecover.length === 0) {
198
205
  throw new Error("No recoverable VTXOs found");
199
206
  }