@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
@@ -5,6 +5,7 @@ import { VtxoScript } from '../script/base.js';
5
5
  import { TxWeightEstimator } from '../utils/txSizeEstimator.js';
6
6
  import { Wallet } from './wallet.js';
7
7
  import { Transaction } from '../utils/transaction.js';
8
+ import { DUST_AMOUNT } from './utils.js';
8
9
  export var Unroll;
9
10
  (function (Unroll) {
10
11
  let StepType;
@@ -200,7 +201,7 @@ export var Unroll;
200
201
  for (const input of inputs) {
201
202
  tx.addInput(input);
202
203
  }
203
- txWeightEstimator.addP2TROutput();
204
+ txWeightEstimator.addOutputAddress(outputAddress, wallet.network);
204
205
  let feeRate = await wallet.onchainProvider.getFeeRate();
205
206
  if (!feeRate || feeRate < Wallet.MIN_FEE_RATE) {
206
207
  feeRate = Wallet.MIN_FEE_RATE;
@@ -209,7 +210,11 @@ export var Unroll;
209
210
  if (feeAmount > totalAmount) {
210
211
  throw new Error("fee amount is greater than the total amount");
211
212
  }
212
- tx.addOutputAddress(outputAddress, totalAmount - feeAmount);
213
+ const sendAmount = totalAmount - feeAmount;
214
+ if (sendAmount < BigInt(DUST_AMOUNT)) {
215
+ throw new Error("send amount is less than dust amount");
216
+ }
217
+ tx.addOutputAddress(outputAddress, sendAmount);
213
218
  const signedTx = await wallet.identity.sign(tx);
214
219
  signedTx.finalize();
215
220
  await wallet.onchainProvider.broadcastTransaction(signedTx.hex);
@@ -1,3 +1,7 @@
1
+ import { ArkAddress, } from '../index.js';
2
+ import { contractHandlers } from '../contracts/handlers/index.js';
3
+ import { hex } from "@scure/base";
4
+ export const DUST_AMOUNT = 546; // sats
1
5
  export function extendVirtualCoin(wallet, vtxo) {
2
6
  return {
3
7
  ...vtxo,
@@ -14,3 +18,54 @@ export function extendCoin(wallet, utxo) {
14
18
  tapTree: wallet.boardingTapscript.encode(),
15
19
  };
16
20
  }
21
+ export function extendVtxoFromContract(vtxo, contract) {
22
+ const handler = contractHandlers.get(contract.type);
23
+ if (!handler) {
24
+ throw new Error(`No handler for contract type '${contract.type}'`);
25
+ }
26
+ const script = handler.createScript(contract.params);
27
+ return {
28
+ ...vtxo,
29
+ forfeitTapLeafScript: script.forfeit(),
30
+ intentTapLeafScript: script.forfeit(),
31
+ tapTree: script.encode(),
32
+ };
33
+ }
34
+ export function getRandomId() {
35
+ const randomValue = crypto.getRandomValues(new Uint8Array(16));
36
+ return hex.encode(randomValue);
37
+ }
38
+ export function isValidArkAddress(address) {
39
+ try {
40
+ ArkAddress.decode(address);
41
+ return true;
42
+ }
43
+ catch (e) {
44
+ return false;
45
+ }
46
+ }
47
+ export function validateRecipients(recipients, dustAmount) {
48
+ const validatedRecipients = [];
49
+ for (const recipient of recipients) {
50
+ let address;
51
+ try {
52
+ address = ArkAddress.decode(recipient.address);
53
+ }
54
+ catch (e) {
55
+ throw new Error(`Invalid Ark address: ${recipient.address}`);
56
+ }
57
+ const amount = recipient.amount || dustAmount;
58
+ if (amount <= 0) {
59
+ throw new Error("Amount must be positive");
60
+ }
61
+ validatedRecipients.push({
62
+ address: recipient.address,
63
+ assets: recipient.assets ?? [],
64
+ amount,
65
+ script: amount < dustAmount
66
+ ? address.subdustPkScript
67
+ : address.pkScript,
68
+ });
69
+ }
70
+ return validatedRecipients;
71
+ }
@@ -0,0 +1,139 @@
1
+ import { equalBytes } from "@scure/btc-signer/utils.js";
2
+ import { ArkAddress } from '../script/address.js';
3
+ import { Packet } from '../asset/index.js';
4
+ import { Address, OutScript } from "@scure/btc-signer";
5
+ export const ErrOffchainOutputNotFound = (address) => new Error(`offchain send output not found: ${address}`);
6
+ export const ErrInvalidAssetOutputAmount = (got, want, assetId) => new Error(`invalid asset output amount for ${assetId}: got ${got}, want ${want}`);
7
+ export const ErrAssetGroupNotFound = (assetId) => new Error(`asset group not found in batch leaf: ${assetId}`);
8
+ export const ErrAssetOutputNotFound = (assetId, outputIndex) => new Error(`asset output not found in asset group ${assetId} at index ${outputIndex}`);
9
+ export const ErrInvalidOnchainOutputAmount = (address) => new Error(`invalid onchain output amount: ${address}`);
10
+ export const ErrInvalidOnchainOutputAssets = (address) => new Error(`onchain output ${address} cannot have assets`);
11
+ export const ErrOnchainOutputNotFound = (address) => new Error(`onchain output not found: ${address}`);
12
+ export const ErrInvalidOffchainOutputAmount = (address) => new Error(`invalid offchain output ${address}, missing amount`);
13
+ /**
14
+ * Validates both offchain and onchain recipients.
15
+ * Offchain recipients are checked against vtxo tree leaves for correct amounts and assets.
16
+ * Onchain recipients are validated against the round transaction outputs (amounts and scripts)
17
+ * via validateOnchainRecipient.
18
+ *
19
+ * @param commitmentTx - The commitment transaction to validate against
20
+ * @param vtxoTreeLeaves - The vtxo tree leaves to validate against
21
+ * @param recipients - The expected recipients to validate (both offchain and onchain)
22
+ * @param network - Network for decoding onchain addresses (e.g. mainnet, testnet)
23
+ * @throws {Error} if a recipient is not present or invalid in the vtxo tree or commitment tx
24
+ */
25
+ export function validateBatchRecipients(commitmentTx, vtxoTreeLeaves, recipients, network) {
26
+ // usedOutputs is used to track which outputs are validated to handle
27
+ // duplicate recipients in the list
28
+ const usedOutputs = new Set();
29
+ const usedOnchainOutputs = new Set();
30
+ for (const recipient of recipients) {
31
+ let arkAddress;
32
+ try {
33
+ arkAddress = ArkAddress.decode(recipient.address);
34
+ }
35
+ catch {
36
+ validateOnchainRecipient(commitmentTx, recipient, network, usedOnchainOutputs);
37
+ continue;
38
+ }
39
+ validateOffchainRecipient(vtxoTreeLeaves, arkAddress, recipient, usedOutputs);
40
+ }
41
+ }
42
+ // validateOnchainRecipient verifies the given recipient is present in the commitment tx outputs list
43
+ function validateOnchainRecipient(commitmentTx, recipient, network, usedOutputs) {
44
+ const addr = Address(network).decode(recipient.address);
45
+ const expectedPkScript = OutScript.encode(addr);
46
+ if (!recipient.amount) {
47
+ throw ErrInvalidOnchainOutputAmount(recipient.address);
48
+ }
49
+ if (recipient.assets && recipient.assets.length > 0) {
50
+ throw ErrInvalidOnchainOutputAssets(recipient.address);
51
+ }
52
+ for (let i = 0; i < commitmentTx.outputsLength; i++) {
53
+ if (usedOutputs.has(i)) {
54
+ continue;
55
+ }
56
+ const output = commitmentTx.getOutput(i);
57
+ if (!output?.script || output.script.length === 0) {
58
+ continue;
59
+ }
60
+ if (equalBytes(output.script, expectedPkScript)) {
61
+ if (output.amount !== BigInt(recipient.amount)) {
62
+ continue; // if amount does not match, continue
63
+ }
64
+ // we found the right output, recipient is valid, return
65
+ usedOutputs.add(i);
66
+ return;
67
+ }
68
+ }
69
+ // if we get here, the recipient is not present in the commitment tx outputs list
70
+ throw ErrOnchainOutputNotFound(recipient.address);
71
+ }
72
+ // validate the offchain recipient is present in one of the leaf output
73
+ // also verify the asset packet is here, and point the same output index
74
+ function validateOffchainRecipient(leaves, arkAddress, recipient, usedOutputs // leafIndex:outputIndex
75
+ ) {
76
+ const expectedPkScript = arkAddress.pkScript;
77
+ if (!recipient.amount) {
78
+ throw ErrInvalidOffchainOutputAmount(recipient.address);
79
+ }
80
+ const expectedAmount = BigInt(recipient.amount);
81
+ let found = false;
82
+ for (let leafIdx = 0; leafIdx < leaves.length; leafIdx++) {
83
+ const leaf = leaves[leafIdx];
84
+ for (let outputIndex = 0; outputIndex < leaf.outputsLength; outputIndex++) {
85
+ const output = leaf.getOutput(outputIndex);
86
+ if (!output?.script || output.script.length === 0) {
87
+ continue;
88
+ }
89
+ if (!equalBytes(output.script, expectedPkScript)) {
90
+ continue;
91
+ }
92
+ if (output.amount !== expectedAmount) {
93
+ continue;
94
+ }
95
+ const key = `${leafIdx}:${outputIndex}`;
96
+ if (usedOutputs.has(key)) {
97
+ continue;
98
+ }
99
+ usedOutputs.add(key);
100
+ found = true;
101
+ // if assets, validate the asset packet
102
+ if (recipient.assets && recipient.assets.length > 0) {
103
+ validateAssetOutputs(leaf, outputIndex, recipient.assets);
104
+ }
105
+ break;
106
+ }
107
+ if (found) {
108
+ break;
109
+ }
110
+ }
111
+ if (!found) {
112
+ throw ErrOffchainOutputNotFound(recipient.address);
113
+ }
114
+ }
115
+ function validateAssetOutputs(leafTx, outputIndex, expectedAssets) {
116
+ const assetPacket = Packet.fromTx(leafTx);
117
+ for (const { assetId, amount } of expectedAssets) {
118
+ validateAssetGroupOutput(assetPacket, outputIndex, assetId, amount);
119
+ }
120
+ }
121
+ function validateAssetGroupOutput(packet, outputIndex, assetId, expectedAmount) {
122
+ const assetGroup = packet.groups.find((group) => {
123
+ if (group.isIssuance())
124
+ return false;
125
+ return group.assetId.toString() === assetId;
126
+ });
127
+ if (!assetGroup) {
128
+ throw ErrAssetGroupNotFound(assetId);
129
+ }
130
+ // find the output at the expected index
131
+ const assetOutput = assetGroup.outputs.find((output) => output.vout === outputIndex);
132
+ if (!assetOutput) {
133
+ throw ErrAssetOutputNotFound(assetId, outputIndex);
134
+ }
135
+ const expectedAmountBigInt = BigInt(expectedAmount);
136
+ if (assetOutput.amount !== expectedAmountBigInt) {
137
+ throw ErrInvalidAssetOutputAmount(assetOutput.amount, expectedAmountBigInt, assetId);
138
+ }
139
+ }
@@ -188,7 +188,7 @@ export class VtxoManager {
188
188
  // Get dust amount from wallet
189
189
  const dustAmount = getDustAmount(this.wallet);
190
190
  // Filter recoverable VTXOs and handle subdust logic
191
- const { vtxosToRecover, includesSubdust, totalAmount } = getRecoverableWithSubdust(allVtxos, dustAmount);
191
+ const { vtxosToRecover, totalAmount } = getRecoverableWithSubdust(allVtxos, dustAmount);
192
192
  if (vtxosToRecover.length === 0) {
193
193
  throw new Error("No recoverable VTXOs found");
194
194
  }