@atomiqlabs/sdk 8.8.3 → 8.8.4

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 (339) hide show
  1. package/LICENSE +201 -201
  2. package/README.md +1760 -1760
  3. package/dist/SmartChainAssets.d.ts +181 -181
  4. package/dist/SmartChainAssets.js +181 -181
  5. package/dist/bitcoin/coinselect2/accumulative.d.ts +7 -7
  6. package/dist/bitcoin/coinselect2/accumulative.js +52 -52
  7. package/dist/bitcoin/coinselect2/blackjack.d.ts +7 -7
  8. package/dist/bitcoin/coinselect2/blackjack.js +38 -38
  9. package/dist/bitcoin/coinselect2/index.d.ts +20 -20
  10. package/dist/bitcoin/coinselect2/index.js +69 -69
  11. package/dist/bitcoin/coinselect2/utils.d.ts +82 -82
  12. package/dist/bitcoin/coinselect2/utils.js +158 -158
  13. package/dist/bitcoin/wallet/BitcoinWallet.d.ts +113 -113
  14. package/dist/bitcoin/wallet/BitcoinWallet.js +335 -335
  15. package/dist/bitcoin/wallet/IBitcoinWallet.d.ts +116 -116
  16. package/dist/bitcoin/wallet/IBitcoinWallet.js +21 -21
  17. package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.d.ts +106 -106
  18. package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.js +196 -196
  19. package/dist/enums/FeeType.d.ts +15 -15
  20. package/dist/enums/FeeType.js +19 -19
  21. package/dist/enums/SwapAmountType.d.ts +15 -15
  22. package/dist/enums/SwapAmountType.js +19 -19
  23. package/dist/enums/SwapDirection.d.ts +15 -15
  24. package/dist/enums/SwapDirection.js +19 -19
  25. package/dist/enums/SwapSide.d.ts +15 -15
  26. package/dist/enums/SwapSide.js +19 -19
  27. package/dist/enums/SwapType.d.ts +75 -75
  28. package/dist/enums/SwapType.js +79 -79
  29. package/dist/errors/IntermediaryError.d.ts +13 -13
  30. package/dist/errors/IntermediaryError.js +27 -27
  31. package/dist/errors/RequestError.d.ts +32 -32
  32. package/dist/errors/RequestError.js +54 -54
  33. package/dist/errors/UserError.d.ts +8 -8
  34. package/dist/errors/UserError.js +16 -16
  35. package/dist/events/UnifiedSwapEventListener.d.ts +23 -23
  36. package/dist/events/UnifiedSwapEventListener.js +132 -132
  37. package/dist/http/HttpUtils.d.ts +27 -27
  38. package/dist/http/HttpUtils.js +91 -91
  39. package/dist/http/paramcoders/IParamReader.d.ts +8 -8
  40. package/dist/http/paramcoders/IParamReader.js +2 -2
  41. package/dist/http/paramcoders/ParamDecoder.d.ts +44 -44
  42. package/dist/http/paramcoders/ParamDecoder.js +137 -137
  43. package/dist/http/paramcoders/ParamEncoder.d.ts +20 -20
  44. package/dist/http/paramcoders/ParamEncoder.js +36 -36
  45. package/dist/http/paramcoders/SchemaVerifier.d.ts +26 -26
  46. package/dist/http/paramcoders/SchemaVerifier.js +145 -145
  47. package/dist/http/paramcoders/client/ResponseParamDecoder.d.ts +11 -11
  48. package/dist/http/paramcoders/client/ResponseParamDecoder.js +57 -57
  49. package/dist/http/paramcoders/client/StreamParamEncoder.d.ts +13 -13
  50. package/dist/http/paramcoders/client/StreamParamEncoder.js +26 -26
  51. package/dist/http/paramcoders/client/StreamingFetchPromise.d.ts +16 -16
  52. package/dist/http/paramcoders/client/StreamingFetchPromise.js +174 -174
  53. package/dist/index.d.ts +85 -85
  54. package/dist/index.js +158 -158
  55. package/dist/intermediaries/Intermediary.d.ts +178 -178
  56. package/dist/intermediaries/Intermediary.js +166 -166
  57. package/dist/intermediaries/IntermediaryDiscovery.d.ts +211 -211
  58. package/dist/intermediaries/IntermediaryDiscovery.js +424 -424
  59. package/dist/intermediaries/apis/IntermediaryAPI.d.ts +450 -450
  60. package/dist/intermediaries/apis/IntermediaryAPI.js +618 -618
  61. package/dist/intermediaries/apis/TrustedIntermediaryAPI.d.ts +155 -155
  62. package/dist/intermediaries/apis/TrustedIntermediaryAPI.js +137 -137
  63. package/dist/lnurl/LNURL.d.ts +102 -102
  64. package/dist/lnurl/LNURL.js +321 -321
  65. package/dist/prices/RedundantSwapPrice.d.ts +110 -110
  66. package/dist/prices/RedundantSwapPrice.js +222 -222
  67. package/dist/prices/SingleSwapPrice.d.ts +34 -34
  68. package/dist/prices/SingleSwapPrice.js +44 -44
  69. package/dist/prices/SwapPriceWithChain.d.ts +107 -107
  70. package/dist/prices/SwapPriceWithChain.js +128 -128
  71. package/dist/prices/abstract/ICachedSwapPrice.d.ts +28 -28
  72. package/dist/prices/abstract/ICachedSwapPrice.js +62 -62
  73. package/dist/prices/abstract/IPriceProvider.d.ts +81 -81
  74. package/dist/prices/abstract/IPriceProvider.js +74 -74
  75. package/dist/prices/abstract/ISwapPrice.d.ts +168 -168
  76. package/dist/prices/abstract/ISwapPrice.js +279 -279
  77. package/dist/prices/providers/BinancePriceProvider.d.ts +23 -23
  78. package/dist/prices/providers/BinancePriceProvider.js +30 -30
  79. package/dist/prices/providers/CoinGeckoPriceProvider.d.ts +23 -23
  80. package/dist/prices/providers/CoinGeckoPriceProvider.js +29 -29
  81. package/dist/prices/providers/CoinPaprikaPriceProvider.d.ts +25 -25
  82. package/dist/prices/providers/CoinPaprikaPriceProvider.js +29 -29
  83. package/dist/prices/providers/CustomPriceProvider.d.ts +24 -24
  84. package/dist/prices/providers/CustomPriceProvider.js +35 -35
  85. package/dist/prices/providers/KrakenPriceProvider.d.ts +38 -38
  86. package/dist/prices/providers/KrakenPriceProvider.js +45 -45
  87. package/dist/prices/providers/OKXPriceProvider.d.ts +34 -34
  88. package/dist/prices/providers/OKXPriceProvider.js +29 -29
  89. package/dist/prices/providers/abstract/ExchangePriceProvider.d.ts +17 -17
  90. package/dist/prices/providers/abstract/ExchangePriceProvider.js +21 -21
  91. package/dist/prices/providers/abstract/HttpPriceProvider.d.ts +7 -7
  92. package/dist/prices/providers/abstract/HttpPriceProvider.js +12 -12
  93. package/dist/storage/IUnifiedStorage.d.ts +85 -85
  94. package/dist/storage/IUnifiedStorage.js +2 -2
  95. package/dist/storage/UnifiedSwapStorage.d.ts +114 -114
  96. package/dist/storage/UnifiedSwapStorage.js +116 -116
  97. package/dist/storage-browser/IndexedDBUnifiedStorage.d.ts +63 -63
  98. package/dist/storage-browser/IndexedDBUnifiedStorage.js +298 -298
  99. package/dist/storage-browser/LocalStorageManager.d.ts +49 -49
  100. package/dist/storage-browser/LocalStorageManager.js +93 -93
  101. package/dist/swapper/Swapper.d.ts +732 -732
  102. package/dist/swapper/Swapper.js +1713 -1713
  103. package/dist/swapper/SwapperFactory.d.ts +135 -135
  104. package/dist/swapper/SwapperFactory.js +162 -162
  105. package/dist/swapper/SwapperUtils.d.ts +206 -206
  106. package/dist/swapper/SwapperUtils.js +481 -481
  107. package/dist/swapper/SwapperWithChain.d.ts +404 -404
  108. package/dist/swapper/SwapperWithChain.js +469 -469
  109. package/dist/swapper/SwapperWithSigner.d.ts +322 -322
  110. package/dist/swapper/SwapperWithSigner.js +318 -318
  111. package/dist/swaps/IAddressSwap.d.ts +22 -22
  112. package/dist/swaps/IAddressSwap.js +14 -14
  113. package/dist/swaps/IBTCWalletSwap.d.ts +73 -73
  114. package/dist/swaps/IBTCWalletSwap.js +18 -18
  115. package/dist/swaps/IClaimableSwap.d.ts +49 -49
  116. package/dist/swaps/IClaimableSwap.js +15 -15
  117. package/dist/swaps/IClaimableSwapWrapper.d.ts +15 -15
  118. package/dist/swaps/IClaimableSwapWrapper.js +2 -2
  119. package/dist/swaps/IRefundableSwap.d.ts +43 -43
  120. package/dist/swaps/IRefundableSwap.js +14 -14
  121. package/dist/swaps/ISwap.d.ts +392 -392
  122. package/dist/swaps/ISwap.js +349 -349
  123. package/dist/swaps/ISwapWithGasDrop.d.ts +21 -21
  124. package/dist/swaps/ISwapWithGasDrop.js +12 -12
  125. package/dist/swaps/ISwapWrapper.d.ts +285 -285
  126. package/dist/swaps/ISwapWrapper.js +353 -353
  127. package/dist/swaps/escrow_swaps/IEscrowSelfInitSwap.d.ts +98 -98
  128. package/dist/swaps/escrow_swaps/IEscrowSelfInitSwap.js +126 -126
  129. package/dist/swaps/escrow_swaps/IEscrowSwap.d.ts +139 -139
  130. package/dist/swaps/escrow_swaps/IEscrowSwap.js +170 -170
  131. package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.d.ts +128 -128
  132. package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.js +167 -167
  133. package/dist/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.d.ts +105 -105
  134. package/dist/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.js +129 -129
  135. package/dist/swaps/escrow_swaps/frombtc/IFromBTCSelfInitSwap.d.ts +162 -162
  136. package/dist/swaps/escrow_swaps/frombtc/IFromBTCSelfInitSwap.js +190 -190
  137. package/dist/swaps/escrow_swaps/frombtc/IFromBTCWrapper.d.ts +64 -64
  138. package/dist/swaps/escrow_swaps/frombtc/IFromBTCWrapper.js +82 -82
  139. package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.d.ts +531 -531
  140. package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.js +1285 -1285
  141. package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.d.ts +190 -190
  142. package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.js +432 -432
  143. package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.d.ts +583 -583
  144. package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.js +1371 -1371
  145. package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.d.ts +235 -235
  146. package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.js +525 -525
  147. package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.d.ts +458 -458
  148. package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.js +1126 -1126
  149. package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.d.ts +202 -202
  150. package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.js +406 -406
  151. package/dist/swaps/escrow_swaps/tobtc/IToBTCSwap.d.ts +403 -403
  152. package/dist/swaps/escrow_swaps/tobtc/IToBTCSwap.js +924 -924
  153. package/dist/swaps/escrow_swaps/tobtc/IToBTCWrapper.d.ts +68 -68
  154. package/dist/swaps/escrow_swaps/tobtc/IToBTCWrapper.js +117 -117
  155. package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.d.ts +127 -127
  156. package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.js +256 -256
  157. package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.d.ts +251 -251
  158. package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.js +536 -536
  159. package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCSwap.d.ts +73 -73
  160. package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCSwap.js +155 -155
  161. package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCWrapper.d.ts +132 -132
  162. package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCWrapper.js +286 -286
  163. package/dist/swaps/spv_swaps/SpvFromBTCSwap.d.ts +637 -637
  164. package/dist/swaps/spv_swaps/SpvFromBTCSwap.js +1448 -1448
  165. package/dist/swaps/spv_swaps/SpvFromBTCWrapper.d.ts +257 -257
  166. package/dist/swaps/spv_swaps/SpvFromBTCWrapper.js +947 -947
  167. package/dist/swaps/trusted/ln/LnForGasSwap.d.ts +261 -261
  168. package/dist/swaps/trusted/ln/LnForGasSwap.js +511 -511
  169. package/dist/swaps/trusted/ln/LnForGasWrapper.d.ts +40 -40
  170. package/dist/swaps/trusted/ln/LnForGasWrapper.js +83 -83
  171. package/dist/swaps/trusted/onchain/OnchainForGasSwap.d.ts +342 -342
  172. package/dist/swaps/trusted/onchain/OnchainForGasSwap.js +715 -715
  173. package/dist/swaps/trusted/onchain/OnchainForGasWrapper.d.ts +69 -69
  174. package/dist/swaps/trusted/onchain/OnchainForGasWrapper.js +93 -93
  175. package/dist/types/AmountData.d.ts +10 -10
  176. package/dist/types/AmountData.js +2 -2
  177. package/dist/types/CustomPriceFunction.d.ts +11 -11
  178. package/dist/types/CustomPriceFunction.js +2 -2
  179. package/dist/types/PriceInfoType.d.ts +28 -28
  180. package/dist/types/PriceInfoType.js +57 -57
  181. package/dist/types/SwapExecutionAction.d.ts +88 -88
  182. package/dist/types/SwapExecutionAction.js +2 -2
  183. package/dist/types/SwapStateInfo.d.ts +5 -5
  184. package/dist/types/SwapStateInfo.js +2 -2
  185. package/dist/types/SwapWithSigner.d.ts +17 -17
  186. package/dist/types/SwapWithSigner.js +43 -43
  187. package/dist/types/Token.d.ts +99 -99
  188. package/dist/types/Token.js +76 -76
  189. package/dist/types/TokenAmount.d.ts +69 -69
  190. package/dist/types/TokenAmount.js +60 -60
  191. package/dist/types/fees/Fee.d.ts +50 -50
  192. package/dist/types/fees/Fee.js +2 -2
  193. package/dist/types/fees/FeeBreakdown.d.ts +11 -11
  194. package/dist/types/fees/FeeBreakdown.js +2 -2
  195. package/dist/types/fees/PercentagePPM.d.ts +17 -17
  196. package/dist/types/fees/PercentagePPM.js +18 -18
  197. package/dist/types/lnurl/LNURLPay.d.ts +61 -61
  198. package/dist/types/lnurl/LNURLPay.js +31 -31
  199. package/dist/types/lnurl/LNURLWithdraw.d.ts +48 -48
  200. package/dist/types/lnurl/LNURLWithdraw.js +27 -27
  201. package/dist/types/wallets/LightningInvoiceCreateService.d.ts +24 -24
  202. package/dist/types/wallets/LightningInvoiceCreateService.js +15 -15
  203. package/dist/types/wallets/MinimalBitcoinWalletInterface.d.ts +23 -23
  204. package/dist/types/wallets/MinimalBitcoinWalletInterface.js +2 -2
  205. package/dist/types/wallets/MinimalLightningNetworkWalletInterface.d.ts +9 -9
  206. package/dist/types/wallets/MinimalLightningNetworkWalletInterface.js +2 -2
  207. package/dist/utils/AutomaticClockDriftCorrection.d.ts +1 -1
  208. package/dist/utils/AutomaticClockDriftCorrection.js +70 -70
  209. package/dist/utils/BitcoinUtils.d.ts +16 -16
  210. package/dist/utils/BitcoinUtils.js +141 -141
  211. package/dist/utils/BitcoinWalletUtils.d.ts +7 -7
  212. package/dist/utils/BitcoinWalletUtils.js +14 -14
  213. package/dist/utils/Logger.d.ts +7 -7
  214. package/dist/utils/Logger.js +12 -12
  215. package/dist/utils/RetryUtils.d.ts +22 -22
  216. package/dist/utils/RetryUtils.js +67 -67
  217. package/dist/utils/SwapUtils.d.ts +88 -88
  218. package/dist/utils/SwapUtils.js +72 -72
  219. package/dist/utils/TimeoutUtils.d.ts +17 -17
  220. package/dist/utils/TimeoutUtils.js +55 -55
  221. package/dist/utils/TokenUtils.d.ts +19 -19
  222. package/dist/utils/TokenUtils.js +37 -37
  223. package/dist/utils/TypeUtils.d.ts +7 -7
  224. package/dist/utils/TypeUtils.js +2 -2
  225. package/dist/utils/Utils.d.ts +67 -67
  226. package/dist/utils/Utils.js +208 -208
  227. package/package.json +43 -43
  228. package/src/SmartChainAssets.ts +186 -186
  229. package/src/bitcoin/coinselect2/accumulative.ts +69 -69
  230. package/src/bitcoin/coinselect2/blackjack.ts +50 -50
  231. package/src/bitcoin/coinselect2/index.ts +93 -93
  232. package/src/bitcoin/coinselect2/utils.ts +236 -236
  233. package/src/bitcoin/wallet/BitcoinWallet.ts +439 -439
  234. package/src/bitcoin/wallet/IBitcoinWallet.ts +140 -140
  235. package/src/bitcoin/wallet/SingleAddressBitcoinWallet.ts +225 -225
  236. package/src/enums/FeeType.ts +15 -15
  237. package/src/enums/SwapAmountType.ts +16 -16
  238. package/src/enums/SwapDirection.ts +15 -15
  239. package/src/enums/SwapSide.ts +16 -16
  240. package/src/enums/SwapType.ts +75 -75
  241. package/src/errors/IntermediaryError.ts +28 -28
  242. package/src/errors/RequestError.ts +64 -64
  243. package/src/errors/UserError.ts +15 -15
  244. package/src/events/UnifiedSwapEventListener.ts +173 -173
  245. package/src/http/HttpUtils.ts +91 -91
  246. package/src/http/paramcoders/IParamReader.ts +9 -9
  247. package/src/http/paramcoders/ParamDecoder.ts +145 -145
  248. package/src/http/paramcoders/ParamEncoder.ts +40 -40
  249. package/src/http/paramcoders/SchemaVerifier.ts +153 -153
  250. package/src/http/paramcoders/client/ResponseParamDecoder.ts +57 -57
  251. package/src/http/paramcoders/client/StreamParamEncoder.ts +28 -28
  252. package/src/http/paramcoders/client/StreamingFetchPromise.ts +192 -192
  253. package/src/index.ts +140 -140
  254. package/src/intermediaries/Intermediary.ts +280 -280
  255. package/src/intermediaries/IntermediaryDiscovery.ts +541 -541
  256. package/src/intermediaries/apis/IntermediaryAPI.ts +963 -963
  257. package/src/intermediaries/apis/TrustedIntermediaryAPI.ts +257 -257
  258. package/src/lnurl/LNURL.ts +402 -402
  259. package/src/prices/RedundantSwapPrice.ts +264 -264
  260. package/src/prices/SingleSwapPrice.ts +50 -50
  261. package/src/prices/SwapPriceWithChain.ts +194 -194
  262. package/src/prices/abstract/ICachedSwapPrice.ts +85 -85
  263. package/src/prices/abstract/IPriceProvider.ts +127 -127
  264. package/src/prices/abstract/ISwapPrice.ts +390 -390
  265. package/src/prices/providers/BinancePriceProvider.ts +48 -48
  266. package/src/prices/providers/CoinGeckoPriceProvider.ts +46 -46
  267. package/src/prices/providers/CoinPaprikaPriceProvider.ts +49 -49
  268. package/src/prices/providers/CustomPriceProvider.ts +40 -40
  269. package/src/prices/providers/KrakenPriceProvider.ts +83 -83
  270. package/src/prices/providers/OKXPriceProvider.ts +59 -59
  271. package/src/prices/providers/abstract/ExchangePriceProvider.ts +31 -31
  272. package/src/prices/providers/abstract/HttpPriceProvider.ts +14 -14
  273. package/src/storage/IUnifiedStorage.ts +95 -95
  274. package/src/storage/UnifiedSwapStorage.ts +141 -141
  275. package/src/storage-browser/IndexedDBUnifiedStorage.ts +350 -350
  276. package/src/storage-browser/LocalStorageManager.ts +106 -106
  277. package/src/swapper/Swapper.ts +2488 -2488
  278. package/src/swapper/SwapperFactory.ts +307 -307
  279. package/src/swapper/SwapperUtils.ts +570 -570
  280. package/src/swapper/SwapperWithChain.ts +707 -707
  281. package/src/swapper/SwapperWithSigner.ts +511 -511
  282. package/src/swaps/IAddressSwap.ts +30 -30
  283. package/src/swaps/IBTCWalletSwap.ts +92 -92
  284. package/src/swaps/IClaimableSwap.ts +65 -65
  285. package/src/swaps/IClaimableSwapWrapper.ts +17 -17
  286. package/src/swaps/IRefundableSwap.ts +58 -58
  287. package/src/swaps/ISwap.ts +703 -703
  288. package/src/swaps/ISwapWithGasDrop.ts +25 -25
  289. package/src/swaps/ISwapWrapper.ts +539 -539
  290. package/src/swaps/escrow_swaps/IEscrowSelfInitSwap.ts +217 -217
  291. package/src/swaps/escrow_swaps/IEscrowSwap.ts +269 -269
  292. package/src/swaps/escrow_swaps/IEscrowSwapWrapper.ts +282 -282
  293. package/src/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.ts +169 -169
  294. package/src/swaps/escrow_swaps/frombtc/IFromBTCSelfInitSwap.ts +300 -300
  295. package/src/swaps/escrow_swaps/frombtc/IFromBTCWrapper.ts +107 -107
  296. package/src/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.ts +1473 -1474
  297. package/src/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.ts +601 -601
  298. package/src/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.ts +1582 -1582
  299. package/src/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.ts +750 -750
  300. package/src/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.ts +1299 -1299
  301. package/src/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.ts +610 -610
  302. package/src/swaps/escrow_swaps/tobtc/IToBTCSwap.ts +1096 -1096
  303. package/src/swaps/escrow_swaps/tobtc/IToBTCWrapper.ts +138 -138
  304. package/src/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.ts +304 -304
  305. package/src/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.ts +786 -786
  306. package/src/swaps/escrow_swaps/tobtc/onchain/ToBTCSwap.ts +206 -206
  307. package/src/swaps/escrow_swaps/tobtc/onchain/ToBTCWrapper.ts +401 -401
  308. package/src/swaps/spv_swaps/SpvFromBTCSwap.ts +1812 -1812
  309. package/src/swaps/spv_swaps/SpvFromBTCWrapper.ts +1236 -1236
  310. package/src/swaps/trusted/ln/LnForGasSwap.ts +589 -589
  311. package/src/swaps/trusted/ln/LnForGasWrapper.ts +91 -91
  312. package/src/swaps/trusted/onchain/OnchainForGasSwap.ts +862 -862
  313. package/src/swaps/trusted/onchain/OnchainForGasWrapper.ts +131 -131
  314. package/src/types/AmountData.ts +9 -9
  315. package/src/types/CustomPriceFunction.ts +11 -11
  316. package/src/types/PriceInfoType.ts +66 -66
  317. package/src/types/SwapExecutionAction.ts +99 -99
  318. package/src/types/SwapStateInfo.ts +6 -6
  319. package/src/types/SwapWithSigner.ts +61 -61
  320. package/src/types/Token.ts +163 -163
  321. package/src/types/TokenAmount.ts +132 -132
  322. package/src/types/fees/Fee.ts +56 -56
  323. package/src/types/fees/FeeBreakdown.ts +11 -11
  324. package/src/types/fees/PercentagePPM.ts +26 -26
  325. package/src/types/lnurl/LNURLPay.ts +79 -79
  326. package/src/types/lnurl/LNURLWithdraw.ts +61 -61
  327. package/src/types/wallets/LightningInvoiceCreateService.ts +30 -30
  328. package/src/types/wallets/MinimalBitcoinWalletInterface.ts +21 -21
  329. package/src/types/wallets/MinimalLightningNetworkWalletInterface.ts +9 -9
  330. package/src/utils/AutomaticClockDriftCorrection.ts +71 -71
  331. package/src/utils/BitcoinUtils.ts +132 -132
  332. package/src/utils/BitcoinWalletUtils.ts +15 -15
  333. package/src/utils/Logger.ts +14 -14
  334. package/src/utils/RetryUtils.ts +78 -78
  335. package/src/utils/SwapUtils.ts +99 -99
  336. package/src/utils/TimeoutUtils.ts +49 -49
  337. package/src/utils/TokenUtils.ts +33 -33
  338. package/src/utils/TypeUtils.ts +8 -8
  339. package/src/utils/Utils.ts +212 -212
@@ -1,571 +1,571 @@
1
- import {decode as bolt11Decode} from "@atomiqlabs/bolt11";
2
- import {Address, Transaction} from "@scure/btc-signer";
3
- import {LNURL} from "../lnurl/LNURL";
4
- import {BTC_NETWORK} from "@scure/btc-signer/utils";
5
- import {SwapType} from "../enums/SwapType";
6
- import {ChainIds, MultiChain, Swapper} from "./Swapper";
7
- import {IBitcoinWallet} from "../bitcoin/wallet/IBitcoinWallet";
8
- import {SingleAddressBitcoinWallet} from "../bitcoin/wallet/SingleAddressBitcoinWallet";
9
- import {BigIntBufferUtils, ChainSwapType, isAbstractSigner} from "@atomiqlabs/base";
10
- import {bigIntMax, fromDecimal, randomBytes} from "../utils/Utils";
11
- import {MinimalBitcoinWalletInterface} from "../types/wallets/MinimalBitcoinWalletInterface";
12
- import {TokenAmount, toTokenAmount} from "../types/TokenAmount";
13
- import {BitcoinTokens, SCToken} from "../types/Token";
14
- import {isLNURLWithdraw, LNURLWithdraw} from "../types/lnurl/LNURLWithdraw";
15
- import {isLNURLPay, LNURLPay} from "../types/lnurl/LNURLPay";
16
- import {toBitcoinWallet} from "../utils/BitcoinWalletUtils";
17
-
18
- /**
19
- * Utility class providing helper methods for address parsing, token balances, serialization
20
- * and other miscellaneous things.
21
- *
22
- * @category Core
23
- */
24
- export class SwapperUtils<T extends MultiChain> {
25
-
26
- readonly bitcoinNetwork: BTC_NETWORK;
27
- private readonly root: Swapper<T>;
28
-
29
- constructor(root: Swapper<T>) {
30
- this.bitcoinNetwork = root._btcNetwork;
31
- this.root = root;
32
- }
33
-
34
- /**
35
- * Checks whether a passed address is a valid address on the smart chain
36
- *
37
- * @param address Address
38
- * @param chainId Smart chain identifier string to check the address for
39
- */
40
- isValidSmartChainAddress(address: string, chainId?: ChainIds<T>): boolean {
41
- if(chainId!=null) {
42
- if(this.root._chains[chainId]==null) throw new Error(`Unknown chain id: ${chainId}`);
43
- return this.root._chains[chainId].chainInterface.isValidAddress(address);
44
- }
45
- for(let chainId of this.root.getSmartChains()) {
46
- if(this.root._chains[chainId].chainInterface.isValidAddress(address)) return true;
47
- }
48
- return false;
49
- }
50
-
51
- /**
52
- * Checks whether an address is a valid BOLT11 bitcoin lightning invoice
53
- *
54
- * @param address Address to check
55
- */
56
- isLightningInvoice(address: string): boolean {
57
- try {
58
- bolt11Decode(address);
59
- return true;
60
- } catch (e) {}
61
- return false;
62
- }
63
-
64
- /**
65
- * Checks whether an address is a valid bitcoin address
66
- *
67
- * @param address Address to check
68
- */
69
- isValidBitcoinAddress(address: string): boolean {
70
- try {
71
- Address(this.bitcoinNetwork).decode(address);
72
- return true;
73
- } catch (e) {
74
- return false;
75
- }
76
- }
77
-
78
- /**
79
- * Checks whether an address is a valid BOLT11 bitcoin lightning invoice WITH AMOUNT
80
- *
81
- * @param address Address to check
82
- */
83
- isValidLightningInvoice(address: string): boolean {
84
- try {
85
- const parsed = bolt11Decode(address);
86
- if(parsed.millisatoshis!=null) return true;
87
- } catch (e) {}
88
- return false;
89
- }
90
-
91
- /**
92
- * Checks whether an address is a valid LNURL (no checking on type is performed)
93
- *
94
- * @param address Address to check
95
- */
96
- isValidLNURL(address: string): boolean {
97
- return LNURL.isLNURL(address);
98
- }
99
-
100
- /**
101
- * Returns type and data about an LNURL
102
- *
103
- * @param lnurl LNURL link to check, can be either `pay` or `withdraw` type
104
- * @param shouldRetry Optional whether HTTP requests should retried on failure
105
- */
106
- getLNURLTypeAndData(lnurl: string, shouldRetry?: boolean): Promise<LNURLPay | LNURLWithdraw | null> {
107
- return LNURL.getLNURLType(lnurl, shouldRetry);
108
- }
109
-
110
- /**
111
- * Returns satoshi value of BOLT11 bitcoin lightning invoice WITH AMOUNT, returns null otherwise
112
- *
113
- * @param lnpr
114
- */
115
- getLightningInvoiceValue(lnpr: string): bigint | null {
116
- const parsed = bolt11Decode(lnpr);
117
- if(parsed.millisatoshis!=null) return (BigInt(parsed.millisatoshis) + 999n) / 1000n;
118
- return null;
119
- }
120
-
121
- private parseBitcoinAddress(resultText: string): {
122
- address: string,
123
- type: "BITCOIN",
124
- swapType: SwapType.TO_BTC,
125
- amount?: TokenAmount
126
- } | null {
127
- let _amount: bigint | undefined = undefined;
128
- if(resultText.includes("?")) {
129
- const arr = resultText.split("?");
130
- resultText = arr[0];
131
- const params = arr[1].split("&");
132
- for(let param of params) {
133
- const arr2 = param.split("=");
134
- const key = arr2[0];
135
- const value = decodeURIComponent(arr2[1]);
136
- if(key==="amount") {
137
- _amount = fromDecimal(parseFloat(value).toFixed(8), 8);
138
- }
139
- }
140
- }
141
- if(this.isValidBitcoinAddress(resultText)) {
142
- return {
143
- address: resultText,
144
- type: "BITCOIN",
145
- swapType: SwapType.TO_BTC,
146
- amount: _amount==null ? undefined : toTokenAmount(_amount, BitcoinTokens.BTC, this.root.prices)
147
- };
148
- }
149
- return null;
150
- }
151
-
152
- private parseLNURLSync(resultText: string): {
153
- address: string,
154
- type: "LNURL",
155
- swapType: null
156
- } | null {
157
- if(this.isValidLNURL(resultText)) {
158
- return {
159
- address: resultText,
160
- type: "LNURL",
161
- swapType: null
162
- };
163
- }
164
- return null;
165
- }
166
-
167
- private async parseLNURL(resultText: string): Promise<{
168
- address: string,
169
- type: "LNURL",
170
- swapType: SwapType.TO_BTCLN | SwapType.FROM_BTCLN,
171
- lnurl: LNURLPay | LNURLWithdraw,
172
- min?: TokenAmount,
173
- max?: TokenAmount,
174
- amount?: TokenAmount
175
- } | null> {
176
- if(this.isValidLNURL(resultText)) {
177
- try {
178
- const result = await this.getLNURLTypeAndData(resultText);
179
- if(result==null) throw new Error("Invalid LNURL specified!");
180
- const swapType = isLNURLPay(result) ? SwapType.TO_BTCLN : isLNURLWithdraw(result) ? SwapType.FROM_BTCLN : null;
181
- if(swapType==null) return null;
182
- const response = {
183
- address: resultText,
184
- type: "LNURL",
185
- swapType,
186
- lnurl: result
187
- } as const;
188
- if(result.min===result.max) {
189
- return {
190
- ...response,
191
- amount: result.min==null ? undefined : toTokenAmount(result.min, BitcoinTokens.BTCLN, this.root.prices)
192
- }
193
- } else {
194
- return {
195
- ...response,
196
- min: result.min==null ? undefined : toTokenAmount(result.min, BitcoinTokens.BTCLN, this.root.prices),
197
- max: result.min==null ? undefined : toTokenAmount(result.max, BitcoinTokens.BTCLN, this.root.prices)
198
- }
199
- }
200
- } catch (e) {
201
- throw new Error("Failed to contact LNURL service, check your internet connection and retry later.");
202
- }
203
- }
204
- return null;
205
- }
206
-
207
- private parseLightningInvoice(resultText: string): {
208
- address: string,
209
- type: "LIGHTNING",
210
- swapType: SwapType.TO_BTCLN,
211
- amount: TokenAmount
212
- } | null {
213
- if(this.isLightningInvoice(resultText)) {
214
- if(this.isValidLightningInvoice(resultText)) {
215
- const amount = this.getLightningInvoiceValue(resultText);
216
- if(amount==null) throw new Error();
217
- return {
218
- address: resultText,
219
- type: "LIGHTNING",
220
- swapType: SwapType.TO_BTCLN,
221
- amount: toTokenAmount(amount, BitcoinTokens.BTCLN, this.root.prices)
222
- }
223
- } else {
224
- throw new Error("Lightning invoice needs to contain an amount!");
225
- }
226
- }
227
- return null;
228
- }
229
-
230
- private parseSmartchainAddress(resultText: string): {
231
- address: string,
232
- type: ChainIds<T>,
233
- swapType: null,
234
- min?: TokenAmount,
235
- max?: TokenAmount
236
- } | null {
237
- for(let chainId of this.root.getSmartChains()) {
238
- if(this.root._chains[chainId].chainInterface.isValidAddress(resultText)) {
239
- return {
240
- address: resultText,
241
- type: chainId,
242
- swapType: null
243
- }
244
- }
245
- }
246
- return null;
247
- }
248
-
249
- /**
250
- * General parser for bitcoin addresses, LNURLs, lightning invoices, smart chain addresses. Also fetches LNURL data
251
- * (hence async and returns Promise).
252
- *
253
- * @param addressString Address to parse
254
- * @throws {Error} Error in address parsing
255
- * @returns Address data or `null` if address doesn't conform to any known format
256
- */
257
- async parseAddress(addressString: string): Promise<{
258
- address: string,
259
- type: "BITCOIN" | "LIGHTNING" | "LNURL" | ChainIds<T>,
260
- swapType: SwapType.TO_BTC | SwapType.TO_BTCLN | SwapType.SPV_VAULT_FROM_BTC | SwapType.FROM_BTCLN | null,
261
- lnurl?: LNURLPay | LNURLWithdraw,
262
- min?: TokenAmount,
263
- max?: TokenAmount,
264
- amount?: TokenAmount
265
- } | null> {
266
- if(addressString.startsWith("bitcoin:")) {
267
- const parsedBitcoinAddress = this.parseBitcoinAddress(addressString.substring(8));
268
- if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
269
- throw new Error("Invalid bitcoin address!");
270
- }
271
-
272
- const parsedBitcoinAddress = this.parseBitcoinAddress(addressString);
273
- if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
274
-
275
- if(addressString.startsWith("lightning:")) {
276
- const resultText = addressString.substring(10);
277
- const resultLnurl = await this.parseLNURL(resultText);
278
- if(resultLnurl!=null) return resultLnurl;
279
-
280
- const resultLightningInvoice = this.parseLightningInvoice(resultText);
281
- if(resultLightningInvoice!=null) return resultLightningInvoice;
282
-
283
- throw new Error("Invalid lightning network invoice or LNURL!");
284
- }
285
-
286
- const resultLnurl = await this.parseLNURL(addressString);
287
- if(resultLnurl!=null) return resultLnurl;
288
-
289
- const resultLightningInvoice = this.parseLightningInvoice(addressString);
290
- if(resultLightningInvoice!=null) return resultLightningInvoice;
291
-
292
- return this.parseSmartchainAddress(addressString);
293
- }
294
-
295
- /**
296
- * Synchronous general parser for bitcoin addresses, LNURLs, lightning invoices, smart chain addresses, doesn't fetch
297
- * LNURL data, returns `swapType: null` instead to prevent returning a Promise
298
- *
299
- * @param addressString Address to parse
300
- * @throws {Error} Error in address parsing
301
- * @returns Address data or `null` if address doesn't conform to any known format
302
- */
303
- parseAddressSync(addressString: string): {
304
- address: string,
305
- type: "BITCOIN" | "LIGHTNING" | "LNURL" | ChainIds<T>,
306
- swapType: SwapType.TO_BTC | SwapType.TO_BTCLN | SwapType.SPV_VAULT_FROM_BTC | null,
307
- min?: TokenAmount,
308
- max?: TokenAmount,
309
- amount?: TokenAmount
310
- } | null {
311
- if(addressString.startsWith("bitcoin:")) {
312
- const parsedBitcoinAddress = this.parseBitcoinAddress(addressString.substring(8));
313
- if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
314
- throw new Error("Invalid bitcoin address!");
315
- }
316
-
317
- const parsedBitcoinAddress = this.parseBitcoinAddress(addressString);
318
- if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
319
-
320
- if(addressString.startsWith("lightning:")) {
321
- const resultText = addressString.substring(10);
322
- const resultLnurl = this.parseLNURLSync(resultText);
323
- if(resultLnurl!=null) return resultLnurl;
324
-
325
- const resultLightningInvoice = this.parseLightningInvoice(resultText);
326
- if(resultLightningInvoice!=null) return resultLightningInvoice;
327
-
328
- throw new Error("Invalid lightning network invoice or LNURL!");
329
- }
330
-
331
- const resultLnurl = this.parseLNURLSync(addressString);
332
- if(resultLnurl!=null) return resultLnurl;
333
-
334
- const resultLightningInvoice = this.parseLightningInvoice(addressString);
335
- if(resultLightningInvoice!=null) return resultLightningInvoice;
336
-
337
- return this.parseSmartchainAddress(addressString);
338
- }
339
-
340
- /**
341
- * Returns a random PSBT that can be used for fee estimation for SPV vault (UTXO-controlled vault) based swaps
342
- * {@link SwapType.SPV_VAULT_FROM_BTC}, the last output (the LP output) is omitted to allow for coinselection
343
- * algorithm to determine maximum sendable amount there
344
- *
345
- * @param chainIdentifier Smart chain to swap to
346
- * @param includeGasToken Whether to return the PSBT also with the gas token amount (increases the vSize by 8)
347
- */
348
- getRandomSpvVaultPsbt<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, includeGasToken?: boolean): Transaction {
349
- const wrapper = this.root._chains[chainIdentifier].wrappers[SwapType.SPV_VAULT_FROM_BTC];
350
- if(wrapper==null) throw new Error("Chain doesn't support spv vault swaps!");
351
- return wrapper.getDummySwapPsbt(includeGasToken);
352
- }
353
-
354
- /**
355
- * Returns the spendable balance of a bitcoin wallet
356
- *
357
- * @param wallet Bitcoin wallet to check the spendable balance for, can either be a simple
358
- * bitcoin address string or a wallet object
359
- * @param targetChain Destination smart chain for the swap, the ensures proper spendable balance
360
- * is estimated taking into consideration different swap primitives available on different chains
361
- * @param options Additional options
362
- */
363
- async getBitcoinSpendableBalance(
364
- wallet: string | IBitcoinWallet | MinimalBitcoinWalletInterface,
365
- targetChain?: ChainIds<T>,
366
- options?: {
367
- gasDrop?: boolean,
368
- feeRate?: number,
369
- minFeeRate?: number
370
- }
371
- ): Promise<{
372
- balance: TokenAmount,
373
- feeRate: number
374
- }> {
375
- let bitcoinWallet: IBitcoinWallet;
376
- if(typeof(wallet)==="string") {
377
- bitcoinWallet = new SingleAddressBitcoinWallet(this.root._bitcoinRpc, this.bitcoinNetwork, {address: wallet, publicKey: ""});
378
- } else {
379
- bitcoinWallet = toBitcoinWallet(wallet, this.root._bitcoinRpc, this.bitcoinNetwork);
380
- }
381
-
382
- let feeRate = options?.feeRate ?? await bitcoinWallet.getFeeRate();
383
- if(options?.minFeeRate!=null) feeRate = Math.max(feeRate, options.minFeeRate);
384
-
385
- let result: {balance: bigint, feeRate: number, totalFee: number};
386
- if(targetChain!=null && this.root.supportsSwapType(targetChain, SwapType.SPV_VAULT_FROM_BTC)) {
387
- result = await bitcoinWallet.getSpendableBalance(this.getRandomSpvVaultPsbt(targetChain, options?.gasDrop), feeRate);
388
- } else {
389
- result = await bitcoinWallet.getSpendableBalance(undefined, feeRate);
390
- }
391
-
392
- return {
393
- balance: toTokenAmount(result.balance, BitcoinTokens.BTC, this.root.prices),
394
- feeRate: result.feeRate
395
- }
396
- }
397
-
398
- /**
399
- * Returns the maximum spendable balance of the smart chain wallet, deducting the fee needed
400
- * to initiate a swap for native balances
401
- */
402
- async getSpendableBalance<ChainIdentifier extends ChainIds<T>>(wallet: string | T[ChainIdentifier]["Signer"] | T[ChainIdentifier]["NativeSigner"], token: SCToken<ChainIdentifier>, options?: {
403
- feeMultiplier?: number,
404
- feeRate?: any
405
- }): Promise<TokenAmount> {
406
- if(this.root._chains[token.chainId]==null) throw new Error("Invalid chain identifier! Unknown chain: "+token.chainId);
407
- const {defaultVersion, versionedContracts, chainInterface} = this.root._chains[token.chainId];
408
-
409
- const {swapContract} = versionedContracts[defaultVersion];
410
-
411
- let signer: string;
412
- if(typeof(wallet)==="string") {
413
- signer = wallet;
414
- } else {
415
- const abstractSigner = isAbstractSigner(wallet) ? wallet : await chainInterface.wrapSigner(wallet);
416
- signer = abstractSigner.getAddress();
417
- }
418
-
419
- let finalBalance: bigint;
420
- if(chainInterface.getNativeCurrencyAddress()!==token.address) {
421
- finalBalance = await chainInterface.getBalance(signer, token.address);
422
- } else {
423
- let [balance, commitFee] = await Promise.all([
424
- chainInterface.getBalance(signer, token.address),
425
- swapContract.getCommitFee(
426
- signer,
427
- //Use large amount, such that the fee for wrapping more tokens is always included!
428
- await swapContract.createSwapData(
429
- ChainSwapType.HTLC, signer, chainInterface.randomAddress(), token.address,
430
- 0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffn,
431
- swapContract.getHashForHtlc(randomBytes(32)).toString("hex"),
432
- BigIntBufferUtils.fromBuffer(randomBytes(8)), BigInt(Math.floor(Date.now()/1000)),
433
- true, false, BigIntBufferUtils.fromBuffer(randomBytes(2)), BigIntBufferUtils.fromBuffer(randomBytes(2))
434
- ),
435
- options?.feeRate
436
- )
437
- ]);
438
-
439
- if(options?.feeMultiplier!=null) {
440
- commitFee = commitFee * (BigInt(Math.floor(options.feeMultiplier*1000000))) / 1000000n;
441
- }
442
-
443
- finalBalance = bigIntMax(balance - commitFee, 0n);
444
- }
445
-
446
- return toTokenAmount(finalBalance, token, this.root.prices);
447
- }
448
-
449
- /**
450
- * Returns the address of the native currency of the smart chain
451
- */
452
- getNativeToken<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier): SCToken<ChainIdentifier> {
453
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
454
- return this.root._tokens[chainIdentifier][this.root._chains[chainIdentifier].chainInterface.getNativeCurrencyAddress()] as SCToken<ChainIdentifier>;
455
- }
456
-
457
- /**
458
- * Returns whether when swapping to the provided token a gas drop can be requested
459
- *
460
- * @param token
461
- */
462
- destinationTokenSupportsGasDrop<ChainIdentifier extends ChainIds<T>>(token: SCToken<ChainIdentifier>): boolean {
463
- if(this.root._chains[token.chainId]==null) throw new Error("Invalid chain identifier! Unknown chain: "+token.chainId);
464
- const {chainInterface} = this.root._chains[token.chainId];
465
- if(chainInterface.shouldGetNativeTokenDrop!=null) return chainInterface.shouldGetNativeTokenDrop(token.address);
466
- return chainInterface.getNativeCurrencyAddress() !== token.address;
467
- }
468
-
469
- /**
470
- * Returns a random signer for a given smart chain
471
- *
472
- * @param chainIdentifier
473
- */
474
- randomSigner<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier): T[ChainIdentifier]["Signer"] {
475
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
476
- return this.root._chains[chainIdentifier].chainInterface.randomSigner();
477
- }
478
-
479
- /**
480
- * Returns a random address for a given smart chain
481
- *
482
- * @param chainIdentifier
483
- */
484
- randomAddress<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier): string {
485
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
486
- return this.root._chains[chainIdentifier].chainInterface.randomAddress();
487
- }
488
-
489
- /**
490
- * Signs and broadcasts the supplied smart chain transaction
491
- *
492
- * @param chainIdentifier Smart chain identifier string
493
- * @param signer Signer to use for signing the transactions
494
- * @param txs An array of transactions to sign
495
- * @param abortSignal Abort signal
496
- * @param onBeforePublish Callback invoked before a transaction is sent (invoked for every transaction to be sent)
497
- */
498
- sendAndConfirm<ChainIdentifier extends ChainIds<T>>(
499
- chainIdentifier: ChainIdentifier,
500
- signer: T[ChainIdentifier]["NativeSigner"] | T[ChainIdentifier]["Signer"],
501
- txs: T[ChainIdentifier]["TX"][],
502
- abortSignal?: AbortSignal,
503
- onBeforePublish?: (txId: string, rawTx: string) => Promise<void>
504
- ): Promise<string[]> {
505
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
506
- return this.root._chains[chainIdentifier].chainInterface.sendAndConfirm(signer, txs, true, abortSignal, false, onBeforePublish);
507
- }
508
-
509
- /**
510
- * Broadcasts already signed smart chain transactions
511
- *
512
- * @param chainIdentifier Smart chain identifier string
513
- * @param txs An array of already signed transactions
514
- * @param abortSignal Abort signal
515
- * @param onBeforePublish Callback invoked before a transaction is sent (invoked for every transaction to be sent)
516
- */
517
- sendSignedAndConfirm<ChainIdentifier extends ChainIds<T>>(
518
- chainIdentifier: ChainIdentifier,
519
- txs: T[ChainIdentifier]["SignedTXType"][],
520
- abortSignal?: AbortSignal,
521
- onBeforePublish?: (txId: string, rawTx: string) => Promise<void>
522
- ): Promise<string[]> {
523
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
524
- return this.root._chains[chainIdentifier].chainInterface.sendSignedAndConfirm(txs, true, abortSignal, false, onBeforePublish);
525
- }
526
-
527
- /**
528
- * Serializes an unsigned smart chain transaction
529
- *
530
- * @param chainIdentifier Smart chain string identifier
531
- * @param tx An unsigned transaction to serialize
532
- */
533
- serializeUnsignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: T[ChainIdentifier]["TX"]): Promise<string> {
534
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
535
- return this.root._chains[chainIdentifier].chainInterface.serializeTx(tx);
536
- }
537
-
538
- /**
539
- * Deserializes an unsigned smart chain transaction
540
- *
541
- * @param chainIdentifier Smart chain string identifier
542
- * @param tx Serialized unsigned transaction
543
- */
544
- deserializeUnsignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: string): Promise<T[ChainIdentifier]["TX"]> {
545
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
546
- return this.root._chains[chainIdentifier].chainInterface.deserializeTx(tx);
547
- }
548
-
549
- /**
550
- * Serializes a signed smart chain transaction
551
- *
552
- * @param chainIdentifier Smart chain string identifier
553
- * @param tx A signed transaction to serialize
554
- */
555
- serializeSignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: T[ChainIdentifier]["SignedTXType"]): Promise<string> {
556
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
557
- return this.root._chains[chainIdentifier].chainInterface.serializeSignedTx(tx);
558
- }
559
-
560
- /**
561
- * Deserializes a signed smart chain transaction
562
- *
563
- * @param chainIdentifier Smart chain string identifier
564
- * @param tx Serialized signed transaction
565
- */
566
- deserializeSignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: string): Promise<T[ChainIdentifier]["SignedTXType"]> {
567
- if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
568
- return this.root._chains[chainIdentifier].chainInterface.deserializeSignedTx(tx);
569
- }
570
-
1
+ import {decode as bolt11Decode} from "@atomiqlabs/bolt11";
2
+ import {Address, Transaction} from "@scure/btc-signer";
3
+ import {LNURL} from "../lnurl/LNURL";
4
+ import {BTC_NETWORK} from "@scure/btc-signer/utils";
5
+ import {SwapType} from "../enums/SwapType";
6
+ import {ChainIds, MultiChain, Swapper} from "./Swapper";
7
+ import {IBitcoinWallet} from "../bitcoin/wallet/IBitcoinWallet";
8
+ import {SingleAddressBitcoinWallet} from "../bitcoin/wallet/SingleAddressBitcoinWallet";
9
+ import {BigIntBufferUtils, ChainSwapType, isAbstractSigner} from "@atomiqlabs/base";
10
+ import {bigIntMax, fromDecimal, randomBytes} from "../utils/Utils";
11
+ import {MinimalBitcoinWalletInterface} from "../types/wallets/MinimalBitcoinWalletInterface";
12
+ import {TokenAmount, toTokenAmount} from "../types/TokenAmount";
13
+ import {BitcoinTokens, SCToken} from "../types/Token";
14
+ import {isLNURLWithdraw, LNURLWithdraw} from "../types/lnurl/LNURLWithdraw";
15
+ import {isLNURLPay, LNURLPay} from "../types/lnurl/LNURLPay";
16
+ import {toBitcoinWallet} from "../utils/BitcoinWalletUtils";
17
+
18
+ /**
19
+ * Utility class providing helper methods for address parsing, token balances, serialization
20
+ * and other miscellaneous things.
21
+ *
22
+ * @category Core
23
+ */
24
+ export class SwapperUtils<T extends MultiChain> {
25
+
26
+ readonly bitcoinNetwork: BTC_NETWORK;
27
+ private readonly root: Swapper<T>;
28
+
29
+ constructor(root: Swapper<T>) {
30
+ this.bitcoinNetwork = root._btcNetwork;
31
+ this.root = root;
32
+ }
33
+
34
+ /**
35
+ * Checks whether a passed address is a valid address on the smart chain
36
+ *
37
+ * @param address Address
38
+ * @param chainId Smart chain identifier string to check the address for
39
+ */
40
+ isValidSmartChainAddress(address: string, chainId?: ChainIds<T>): boolean {
41
+ if(chainId!=null) {
42
+ if(this.root._chains[chainId]==null) throw new Error(`Unknown chain id: ${chainId}`);
43
+ return this.root._chains[chainId].chainInterface.isValidAddress(address);
44
+ }
45
+ for(let chainId of this.root.getSmartChains()) {
46
+ if(this.root._chains[chainId].chainInterface.isValidAddress(address)) return true;
47
+ }
48
+ return false;
49
+ }
50
+
51
+ /**
52
+ * Checks whether an address is a valid BOLT11 bitcoin lightning invoice
53
+ *
54
+ * @param address Address to check
55
+ */
56
+ isLightningInvoice(address: string): boolean {
57
+ try {
58
+ bolt11Decode(address);
59
+ return true;
60
+ } catch (e) {}
61
+ return false;
62
+ }
63
+
64
+ /**
65
+ * Checks whether an address is a valid bitcoin address
66
+ *
67
+ * @param address Address to check
68
+ */
69
+ isValidBitcoinAddress(address: string): boolean {
70
+ try {
71
+ Address(this.bitcoinNetwork).decode(address);
72
+ return true;
73
+ } catch (e) {
74
+ return false;
75
+ }
76
+ }
77
+
78
+ /**
79
+ * Checks whether an address is a valid BOLT11 bitcoin lightning invoice WITH AMOUNT
80
+ *
81
+ * @param address Address to check
82
+ */
83
+ isValidLightningInvoice(address: string): boolean {
84
+ try {
85
+ const parsed = bolt11Decode(address);
86
+ if(parsed.millisatoshis!=null) return true;
87
+ } catch (e) {}
88
+ return false;
89
+ }
90
+
91
+ /**
92
+ * Checks whether an address is a valid LNURL (no checking on type is performed)
93
+ *
94
+ * @param address Address to check
95
+ */
96
+ isValidLNURL(address: string): boolean {
97
+ return LNURL.isLNURL(address);
98
+ }
99
+
100
+ /**
101
+ * Returns type and data about an LNURL
102
+ *
103
+ * @param lnurl LNURL link to check, can be either `pay` or `withdraw` type
104
+ * @param shouldRetry Optional whether HTTP requests should retried on failure
105
+ */
106
+ getLNURLTypeAndData(lnurl: string, shouldRetry?: boolean): Promise<LNURLPay | LNURLWithdraw | null> {
107
+ return LNURL.getLNURLType(lnurl, shouldRetry);
108
+ }
109
+
110
+ /**
111
+ * Returns satoshi value of BOLT11 bitcoin lightning invoice WITH AMOUNT, returns null otherwise
112
+ *
113
+ * @param lnpr
114
+ */
115
+ getLightningInvoiceValue(lnpr: string): bigint | null {
116
+ const parsed = bolt11Decode(lnpr);
117
+ if(parsed.millisatoshis!=null) return (BigInt(parsed.millisatoshis) + 999n) / 1000n;
118
+ return null;
119
+ }
120
+
121
+ private parseBitcoinAddress(resultText: string): {
122
+ address: string,
123
+ type: "BITCOIN",
124
+ swapType: SwapType.TO_BTC,
125
+ amount?: TokenAmount
126
+ } | null {
127
+ let _amount: bigint | undefined = undefined;
128
+ if(resultText.includes("?")) {
129
+ const arr = resultText.split("?");
130
+ resultText = arr[0];
131
+ const params = arr[1].split("&");
132
+ for(let param of params) {
133
+ const arr2 = param.split("=");
134
+ const key = arr2[0];
135
+ const value = decodeURIComponent(arr2[1]);
136
+ if(key==="amount") {
137
+ _amount = fromDecimal(parseFloat(value).toFixed(8), 8);
138
+ }
139
+ }
140
+ }
141
+ if(this.isValidBitcoinAddress(resultText)) {
142
+ return {
143
+ address: resultText,
144
+ type: "BITCOIN",
145
+ swapType: SwapType.TO_BTC,
146
+ amount: _amount==null ? undefined : toTokenAmount(_amount, BitcoinTokens.BTC, this.root.prices)
147
+ };
148
+ }
149
+ return null;
150
+ }
151
+
152
+ private parseLNURLSync(resultText: string): {
153
+ address: string,
154
+ type: "LNURL",
155
+ swapType: null
156
+ } | null {
157
+ if(this.isValidLNURL(resultText)) {
158
+ return {
159
+ address: resultText,
160
+ type: "LNURL",
161
+ swapType: null
162
+ };
163
+ }
164
+ return null;
165
+ }
166
+
167
+ private async parseLNURL(resultText: string): Promise<{
168
+ address: string,
169
+ type: "LNURL",
170
+ swapType: SwapType.TO_BTCLN | SwapType.FROM_BTCLN,
171
+ lnurl: LNURLPay | LNURLWithdraw,
172
+ min?: TokenAmount,
173
+ max?: TokenAmount,
174
+ amount?: TokenAmount
175
+ } | null> {
176
+ if(this.isValidLNURL(resultText)) {
177
+ try {
178
+ const result = await this.getLNURLTypeAndData(resultText);
179
+ if(result==null) throw new Error("Invalid LNURL specified!");
180
+ const swapType = isLNURLPay(result) ? SwapType.TO_BTCLN : isLNURLWithdraw(result) ? SwapType.FROM_BTCLN : null;
181
+ if(swapType==null) return null;
182
+ const response = {
183
+ address: resultText,
184
+ type: "LNURL",
185
+ swapType,
186
+ lnurl: result
187
+ } as const;
188
+ if(result.min===result.max) {
189
+ return {
190
+ ...response,
191
+ amount: result.min==null ? undefined : toTokenAmount(result.min, BitcoinTokens.BTCLN, this.root.prices)
192
+ }
193
+ } else {
194
+ return {
195
+ ...response,
196
+ min: result.min==null ? undefined : toTokenAmount(result.min, BitcoinTokens.BTCLN, this.root.prices),
197
+ max: result.min==null ? undefined : toTokenAmount(result.max, BitcoinTokens.BTCLN, this.root.prices)
198
+ }
199
+ }
200
+ } catch (e) {
201
+ throw new Error("Failed to contact LNURL service, check your internet connection and retry later.");
202
+ }
203
+ }
204
+ return null;
205
+ }
206
+
207
+ private parseLightningInvoice(resultText: string): {
208
+ address: string,
209
+ type: "LIGHTNING",
210
+ swapType: SwapType.TO_BTCLN,
211
+ amount: TokenAmount
212
+ } | null {
213
+ if(this.isLightningInvoice(resultText)) {
214
+ if(this.isValidLightningInvoice(resultText)) {
215
+ const amount = this.getLightningInvoiceValue(resultText);
216
+ if(amount==null) throw new Error();
217
+ return {
218
+ address: resultText,
219
+ type: "LIGHTNING",
220
+ swapType: SwapType.TO_BTCLN,
221
+ amount: toTokenAmount(amount, BitcoinTokens.BTCLN, this.root.prices)
222
+ }
223
+ } else {
224
+ throw new Error("Lightning invoice needs to contain an amount!");
225
+ }
226
+ }
227
+ return null;
228
+ }
229
+
230
+ private parseSmartchainAddress(resultText: string): {
231
+ address: string,
232
+ type: ChainIds<T>,
233
+ swapType: null,
234
+ min?: TokenAmount,
235
+ max?: TokenAmount
236
+ } | null {
237
+ for(let chainId of this.root.getSmartChains()) {
238
+ if(this.root._chains[chainId].chainInterface.isValidAddress(resultText)) {
239
+ return {
240
+ address: resultText,
241
+ type: chainId,
242
+ swapType: null
243
+ }
244
+ }
245
+ }
246
+ return null;
247
+ }
248
+
249
+ /**
250
+ * General parser for bitcoin addresses, LNURLs, lightning invoices, smart chain addresses. Also fetches LNURL data
251
+ * (hence async and returns Promise).
252
+ *
253
+ * @param addressString Address to parse
254
+ * @throws {Error} Error in address parsing
255
+ * @returns Address data or `null` if address doesn't conform to any known format
256
+ */
257
+ async parseAddress(addressString: string): Promise<{
258
+ address: string,
259
+ type: "BITCOIN" | "LIGHTNING" | "LNURL" | ChainIds<T>,
260
+ swapType: SwapType.TO_BTC | SwapType.TO_BTCLN | SwapType.SPV_VAULT_FROM_BTC | SwapType.FROM_BTCLN | null,
261
+ lnurl?: LNURLPay | LNURLWithdraw,
262
+ min?: TokenAmount,
263
+ max?: TokenAmount,
264
+ amount?: TokenAmount
265
+ } | null> {
266
+ if(addressString.startsWith("bitcoin:")) {
267
+ const parsedBitcoinAddress = this.parseBitcoinAddress(addressString.substring(8));
268
+ if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
269
+ throw new Error("Invalid bitcoin address!");
270
+ }
271
+
272
+ const parsedBitcoinAddress = this.parseBitcoinAddress(addressString);
273
+ if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
274
+
275
+ if(addressString.startsWith("lightning:")) {
276
+ const resultText = addressString.substring(10);
277
+ const resultLnurl = await this.parseLNURL(resultText);
278
+ if(resultLnurl!=null) return resultLnurl;
279
+
280
+ const resultLightningInvoice = this.parseLightningInvoice(resultText);
281
+ if(resultLightningInvoice!=null) return resultLightningInvoice;
282
+
283
+ throw new Error("Invalid lightning network invoice or LNURL!");
284
+ }
285
+
286
+ const resultLnurl = await this.parseLNURL(addressString);
287
+ if(resultLnurl!=null) return resultLnurl;
288
+
289
+ const resultLightningInvoice = this.parseLightningInvoice(addressString);
290
+ if(resultLightningInvoice!=null) return resultLightningInvoice;
291
+
292
+ return this.parseSmartchainAddress(addressString);
293
+ }
294
+
295
+ /**
296
+ * Synchronous general parser for bitcoin addresses, LNURLs, lightning invoices, smart chain addresses, doesn't fetch
297
+ * LNURL data, returns `swapType: null` instead to prevent returning a Promise
298
+ *
299
+ * @param addressString Address to parse
300
+ * @throws {Error} Error in address parsing
301
+ * @returns Address data or `null` if address doesn't conform to any known format
302
+ */
303
+ parseAddressSync(addressString: string): {
304
+ address: string,
305
+ type: "BITCOIN" | "LIGHTNING" | "LNURL" | ChainIds<T>,
306
+ swapType: SwapType.TO_BTC | SwapType.TO_BTCLN | SwapType.SPV_VAULT_FROM_BTC | null,
307
+ min?: TokenAmount,
308
+ max?: TokenAmount,
309
+ amount?: TokenAmount
310
+ } | null {
311
+ if(addressString.startsWith("bitcoin:")) {
312
+ const parsedBitcoinAddress = this.parseBitcoinAddress(addressString.substring(8));
313
+ if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
314
+ throw new Error("Invalid bitcoin address!");
315
+ }
316
+
317
+ const parsedBitcoinAddress = this.parseBitcoinAddress(addressString);
318
+ if(parsedBitcoinAddress!=null) return parsedBitcoinAddress;
319
+
320
+ if(addressString.startsWith("lightning:")) {
321
+ const resultText = addressString.substring(10);
322
+ const resultLnurl = this.parseLNURLSync(resultText);
323
+ if(resultLnurl!=null) return resultLnurl;
324
+
325
+ const resultLightningInvoice = this.parseLightningInvoice(resultText);
326
+ if(resultLightningInvoice!=null) return resultLightningInvoice;
327
+
328
+ throw new Error("Invalid lightning network invoice or LNURL!");
329
+ }
330
+
331
+ const resultLnurl = this.parseLNURLSync(addressString);
332
+ if(resultLnurl!=null) return resultLnurl;
333
+
334
+ const resultLightningInvoice = this.parseLightningInvoice(addressString);
335
+ if(resultLightningInvoice!=null) return resultLightningInvoice;
336
+
337
+ return this.parseSmartchainAddress(addressString);
338
+ }
339
+
340
+ /**
341
+ * Returns a random PSBT that can be used for fee estimation for SPV vault (UTXO-controlled vault) based swaps
342
+ * {@link SwapType.SPV_VAULT_FROM_BTC}, the last output (the LP output) is omitted to allow for coinselection
343
+ * algorithm to determine maximum sendable amount there
344
+ *
345
+ * @param chainIdentifier Smart chain to swap to
346
+ * @param includeGasToken Whether to return the PSBT also with the gas token amount (increases the vSize by 8)
347
+ */
348
+ getRandomSpvVaultPsbt<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, includeGasToken?: boolean): Transaction {
349
+ const wrapper = this.root._chains[chainIdentifier].wrappers[SwapType.SPV_VAULT_FROM_BTC];
350
+ if(wrapper==null) throw new Error("Chain doesn't support spv vault swaps!");
351
+ return wrapper.getDummySwapPsbt(includeGasToken);
352
+ }
353
+
354
+ /**
355
+ * Returns the spendable balance of a bitcoin wallet
356
+ *
357
+ * @param wallet Bitcoin wallet to check the spendable balance for, can either be a simple
358
+ * bitcoin address string or a wallet object
359
+ * @param targetChain Destination smart chain for the swap, the ensures proper spendable balance
360
+ * is estimated taking into consideration different swap primitives available on different chains
361
+ * @param options Additional options
362
+ */
363
+ async getBitcoinSpendableBalance(
364
+ wallet: string | IBitcoinWallet | MinimalBitcoinWalletInterface,
365
+ targetChain?: ChainIds<T>,
366
+ options?: {
367
+ gasDrop?: boolean,
368
+ feeRate?: number,
369
+ minFeeRate?: number
370
+ }
371
+ ): Promise<{
372
+ balance: TokenAmount,
373
+ feeRate: number
374
+ }> {
375
+ let bitcoinWallet: IBitcoinWallet;
376
+ if(typeof(wallet)==="string") {
377
+ bitcoinWallet = new SingleAddressBitcoinWallet(this.root._bitcoinRpc, this.bitcoinNetwork, {address: wallet, publicKey: ""});
378
+ } else {
379
+ bitcoinWallet = toBitcoinWallet(wallet, this.root._bitcoinRpc, this.bitcoinNetwork);
380
+ }
381
+
382
+ let feeRate = options?.feeRate ?? await bitcoinWallet.getFeeRate();
383
+ if(options?.minFeeRate!=null) feeRate = Math.max(feeRate, options.minFeeRate);
384
+
385
+ let result: {balance: bigint, feeRate: number, totalFee: number};
386
+ if(targetChain!=null && this.root.supportsSwapType(targetChain, SwapType.SPV_VAULT_FROM_BTC)) {
387
+ result = await bitcoinWallet.getSpendableBalance(this.getRandomSpvVaultPsbt(targetChain, options?.gasDrop), feeRate);
388
+ } else {
389
+ result = await bitcoinWallet.getSpendableBalance(undefined, feeRate);
390
+ }
391
+
392
+ return {
393
+ balance: toTokenAmount(result.balance, BitcoinTokens.BTC, this.root.prices),
394
+ feeRate: result.feeRate
395
+ }
396
+ }
397
+
398
+ /**
399
+ * Returns the maximum spendable balance of the smart chain wallet, deducting the fee needed
400
+ * to initiate a swap for native balances
401
+ */
402
+ async getSpendableBalance<ChainIdentifier extends ChainIds<T>>(wallet: string | T[ChainIdentifier]["Signer"] | T[ChainIdentifier]["NativeSigner"], token: SCToken<ChainIdentifier>, options?: {
403
+ feeMultiplier?: number,
404
+ feeRate?: any
405
+ }): Promise<TokenAmount> {
406
+ if(this.root._chains[token.chainId]==null) throw new Error("Invalid chain identifier! Unknown chain: "+token.chainId);
407
+ const {defaultVersion, versionedContracts, chainInterface} = this.root._chains[token.chainId];
408
+
409
+ const {swapContract} = versionedContracts[defaultVersion];
410
+
411
+ let signer: string;
412
+ if(typeof(wallet)==="string") {
413
+ signer = wallet;
414
+ } else {
415
+ const abstractSigner = isAbstractSigner(wallet) ? wallet : await chainInterface.wrapSigner(wallet);
416
+ signer = abstractSigner.getAddress();
417
+ }
418
+
419
+ let finalBalance: bigint;
420
+ if(chainInterface.getNativeCurrencyAddress()!==token.address) {
421
+ finalBalance = await chainInterface.getBalance(signer, token.address);
422
+ } else {
423
+ let [balance, commitFee] = await Promise.all([
424
+ chainInterface.getBalance(signer, token.address),
425
+ swapContract.getCommitFee(
426
+ signer,
427
+ //Use large amount, such that the fee for wrapping more tokens is always included!
428
+ await swapContract.createSwapData(
429
+ ChainSwapType.HTLC, signer, chainInterface.randomAddress(), token.address,
430
+ 0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffn,
431
+ swapContract.getHashForHtlc(randomBytes(32)).toString("hex"),
432
+ BigIntBufferUtils.fromBuffer(randomBytes(8)), BigInt(Math.floor(Date.now()/1000)),
433
+ true, false, BigIntBufferUtils.fromBuffer(randomBytes(2)), BigIntBufferUtils.fromBuffer(randomBytes(2))
434
+ ),
435
+ options?.feeRate
436
+ )
437
+ ]);
438
+
439
+ if(options?.feeMultiplier!=null) {
440
+ commitFee = commitFee * (BigInt(Math.floor(options.feeMultiplier*1000000))) / 1000000n;
441
+ }
442
+
443
+ finalBalance = bigIntMax(balance - commitFee, 0n);
444
+ }
445
+
446
+ return toTokenAmount(finalBalance, token, this.root.prices);
447
+ }
448
+
449
+ /**
450
+ * Returns the address of the native currency of the smart chain
451
+ */
452
+ getNativeToken<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier): SCToken<ChainIdentifier> {
453
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
454
+ return this.root._tokens[chainIdentifier][this.root._chains[chainIdentifier].chainInterface.getNativeCurrencyAddress()] as SCToken<ChainIdentifier>;
455
+ }
456
+
457
+ /**
458
+ * Returns whether when swapping to the provided token a gas drop can be requested
459
+ *
460
+ * @param token
461
+ */
462
+ destinationTokenSupportsGasDrop<ChainIdentifier extends ChainIds<T>>(token: SCToken<ChainIdentifier>): boolean {
463
+ if(this.root._chains[token.chainId]==null) throw new Error("Invalid chain identifier! Unknown chain: "+token.chainId);
464
+ const {chainInterface} = this.root._chains[token.chainId];
465
+ if(chainInterface.shouldGetNativeTokenDrop!=null) return chainInterface.shouldGetNativeTokenDrop(token.address);
466
+ return chainInterface.getNativeCurrencyAddress() !== token.address;
467
+ }
468
+
469
+ /**
470
+ * Returns a random signer for a given smart chain
471
+ *
472
+ * @param chainIdentifier
473
+ */
474
+ randomSigner<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier): T[ChainIdentifier]["Signer"] {
475
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
476
+ return this.root._chains[chainIdentifier].chainInterface.randomSigner();
477
+ }
478
+
479
+ /**
480
+ * Returns a random address for a given smart chain
481
+ *
482
+ * @param chainIdentifier
483
+ */
484
+ randomAddress<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier): string {
485
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
486
+ return this.root._chains[chainIdentifier].chainInterface.randomAddress();
487
+ }
488
+
489
+ /**
490
+ * Signs and broadcasts the supplied smart chain transaction
491
+ *
492
+ * @param chainIdentifier Smart chain identifier string
493
+ * @param signer Signer to use for signing the transactions
494
+ * @param txs An array of transactions to sign
495
+ * @param abortSignal Abort signal
496
+ * @param onBeforePublish Callback invoked before a transaction is sent (invoked for every transaction to be sent)
497
+ */
498
+ sendAndConfirm<ChainIdentifier extends ChainIds<T>>(
499
+ chainIdentifier: ChainIdentifier,
500
+ signer: T[ChainIdentifier]["NativeSigner"] | T[ChainIdentifier]["Signer"],
501
+ txs: T[ChainIdentifier]["TX"][],
502
+ abortSignal?: AbortSignal,
503
+ onBeforePublish?: (txId: string, rawTx: string) => Promise<void>
504
+ ): Promise<string[]> {
505
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
506
+ return this.root._chains[chainIdentifier].chainInterface.sendAndConfirm(signer, txs, true, abortSignal, false, onBeforePublish);
507
+ }
508
+
509
+ /**
510
+ * Broadcasts already signed smart chain transactions
511
+ *
512
+ * @param chainIdentifier Smart chain identifier string
513
+ * @param txs An array of already signed transactions
514
+ * @param abortSignal Abort signal
515
+ * @param onBeforePublish Callback invoked before a transaction is sent (invoked for every transaction to be sent)
516
+ */
517
+ sendSignedAndConfirm<ChainIdentifier extends ChainIds<T>>(
518
+ chainIdentifier: ChainIdentifier,
519
+ txs: T[ChainIdentifier]["SignedTXType"][],
520
+ abortSignal?: AbortSignal,
521
+ onBeforePublish?: (txId: string, rawTx: string) => Promise<void>
522
+ ): Promise<string[]> {
523
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
524
+ return this.root._chains[chainIdentifier].chainInterface.sendSignedAndConfirm(txs, true, abortSignal, false, onBeforePublish);
525
+ }
526
+
527
+ /**
528
+ * Serializes an unsigned smart chain transaction
529
+ *
530
+ * @param chainIdentifier Smart chain string identifier
531
+ * @param tx An unsigned transaction to serialize
532
+ */
533
+ serializeUnsignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: T[ChainIdentifier]["TX"]): Promise<string> {
534
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
535
+ return this.root._chains[chainIdentifier].chainInterface.serializeTx(tx);
536
+ }
537
+
538
+ /**
539
+ * Deserializes an unsigned smart chain transaction
540
+ *
541
+ * @param chainIdentifier Smart chain string identifier
542
+ * @param tx Serialized unsigned transaction
543
+ */
544
+ deserializeUnsignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: string): Promise<T[ChainIdentifier]["TX"]> {
545
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
546
+ return this.root._chains[chainIdentifier].chainInterface.deserializeTx(tx);
547
+ }
548
+
549
+ /**
550
+ * Serializes a signed smart chain transaction
551
+ *
552
+ * @param chainIdentifier Smart chain string identifier
553
+ * @param tx A signed transaction to serialize
554
+ */
555
+ serializeSignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: T[ChainIdentifier]["SignedTXType"]): Promise<string> {
556
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
557
+ return this.root._chains[chainIdentifier].chainInterface.serializeSignedTx(tx);
558
+ }
559
+
560
+ /**
561
+ * Deserializes a signed smart chain transaction
562
+ *
563
+ * @param chainIdentifier Smart chain string identifier
564
+ * @param tx Serialized signed transaction
565
+ */
566
+ deserializeSignedTransaction<ChainIdentifier extends ChainIds<T>>(chainIdentifier: ChainIdentifier, tx: string): Promise<T[ChainIdentifier]["SignedTXType"]> {
567
+ if(this.root._chains[chainIdentifier]==null) throw new Error("Invalid chain identifier! Unknown chain: "+chainIdentifier);
568
+ return this.root._chains[chainIdentifier].chainInterface.deserializeSignedTx(tx);
569
+ }
570
+
571
571
  }