@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,589 +1,589 @@
1
- import {decode as bolt11Decode} from "@atomiqlabs/bolt11";
2
- import {SwapType} from "../../../enums/SwapType";
3
- import {ChainType} from "@atomiqlabs/base";
4
- import {LnForGasSwapTypeDefinition, LnForGasWrapper} from "./LnForGasWrapper";
5
- import {toBigInt} from "../../../utils/Utils";
6
- import {isISwapInit, ISwap, ISwapInit} from "../../ISwap";
7
- import {InvoiceStatusResponseCodes, TrustedIntermediaryAPI} from "../../../intermediaries/apis/TrustedIntermediaryAPI";
8
- import {Fee} from "../../../types/fees/Fee";
9
- import {IAddressSwap} from "../../IAddressSwap";
10
- import {FeeType} from "../../../enums/FeeType";
11
- import {ppmToPercentage} from "../../../types/fees/PercentagePPM";
12
- import {TokenAmount, toTokenAmount} from "../../../types/TokenAmount";
13
- import {BitcoinTokens, BtcToken, SCToken} from "../../../types/Token";
14
- import {getLogger, LoggerType} from "../../../utils/Logger";
15
- import {timeoutPromise} from "../../../utils/TimeoutUtils";
16
- import {SwapExecutionAction, SwapExecutionActionLightning} from "../../../types/SwapExecutionAction";
17
-
18
- /**
19
- * State enum for trusted Lightning gas swaps
20
- *
21
- * @category Swaps/Trusted Gas Swaps
22
- */
23
- export enum LnForGasSwapState {
24
- /**
25
- * The swap quote expired without user sending in the lightning network payment
26
- */
27
- EXPIRED = -2,
28
- /**
29
- * The swap has failed after the intermediary already received a lightning network payment on the source
30
- */
31
- FAILED = -1,
32
- /**
33
- * Swap was created, pay the provided lightning network invoice
34
- */
35
- PR_CREATED = 0,
36
- /**
37
- * User paid the lightning network invoice on the source
38
- */
39
- PR_PAID = 1,
40
- /**
41
- * The swap is finished after the intermediary sent funds on the destination chain
42
- */
43
- FINISHED = 2
44
- }
45
-
46
- const LnForGasSwapStateDescription = {
47
- [LnForGasSwapState.EXPIRED]:
48
- "The swap quote expired without user sending in the lightning network payment",
49
- [LnForGasSwapState.FAILED]:
50
- "The swap has failed after the intermediary already received a lightning network payment on the source",
51
- [LnForGasSwapState.PR_CREATED]:
52
- "Swap was created, pay the provided lightning network invoice",
53
- [LnForGasSwapState.PR_PAID]:
54
- "User paid the lightning network invoice on the source",
55
- [LnForGasSwapState.FINISHED]:
56
- "The swap is finished after the intermediary sent funds on the destination chain"
57
- }
58
-
59
- export type LnForGasSwapInit = ISwapInit & {
60
- pr: string;
61
- outputAmount: bigint;
62
- recipient: string;
63
- token: string;
64
- };
65
-
66
- export function isLnForGasSwapInit(obj: any): obj is LnForGasSwapInit {
67
- return typeof(obj.pr)==="string" &&
68
- typeof(obj.outputAmount) === "bigint" &&
69
- typeof(obj.recipient)==="string" &&
70
- typeof(obj.token)==="string" &&
71
- isISwapInit(obj);
72
- }
73
-
74
- /**
75
- * Trusted swap for Bitcoin Lightning -> Smart chains, to be used for minor amounts to get gas tokens on
76
- * the destination chain, which is only needed for Solana, which still uses legacy swaps
77
- *
78
- * @category Swaps/Trusted Gas Swaps
79
- */
80
- export class LnForGasSwap<T extends ChainType = ChainType> extends ISwap<T, LnForGasSwapTypeDefinition<T>, LnForGasSwapState> implements IAddressSwap {
81
- protected readonly TYPE: SwapType.TRUSTED_FROM_BTCLN = SwapType.TRUSTED_FROM_BTCLN;
82
-
83
- /**
84
- * @internal
85
- */
86
- protected readonly swapStateDescription = LnForGasSwapStateDescription;
87
- /**
88
- * @internal
89
- */
90
- protected readonly swapStateName = (state: number) => LnForGasSwapState[state];
91
-
92
- /**
93
- * @internal
94
- */
95
- protected readonly currentVersion: number = 2;
96
- /**
97
- * @internal
98
- */
99
- protected readonly logger: LoggerType;
100
-
101
- //State: PR_CREATED
102
- private readonly pr: string;
103
- private readonly outputAmount: bigint;
104
- private readonly recipient: string;
105
- private readonly token: string;
106
-
107
- //State: FINISHED
108
- /**
109
- * Destination transaction ID on the smart chain side
110
- * @private
111
- */
112
- private scTxId?: string;
113
-
114
- constructor(wrapper: LnForGasWrapper<T>, init: LnForGasSwapInit);
115
- constructor(wrapper: LnForGasWrapper<T>, obj: any);
116
- constructor(
117
- wrapper: LnForGasWrapper<T>,
118
- initOrObj: LnForGasSwapInit | any
119
- ) {
120
- if(isLnForGasSwapInit(initOrObj) && initOrObj.url!=null) initOrObj.url += "/lnforgas";
121
- super(wrapper, initOrObj);
122
- if(isLnForGasSwapInit(initOrObj)) {
123
- this.pr = initOrObj.pr;
124
- this.outputAmount = initOrObj.outputAmount;
125
- this.recipient = initOrObj.recipient;
126
- this.token = initOrObj.token;
127
- this._state = LnForGasSwapState.PR_CREATED;
128
- } else {
129
- this.pr = initOrObj.pr;
130
- this.outputAmount = toBigInt(initOrObj.outputAmount);
131
- this.recipient = initOrObj.recipient;
132
- this.token = initOrObj.token;
133
- this.scTxId = initOrObj.scTxId;
134
- }
135
- this.tryRecomputeSwapPrice();
136
- if(this.pr!=null) {
137
- const decoded = bolt11Decode(this.pr);
138
- if(decoded.timeExpireDate!=null) this.expiry = decoded.timeExpireDate*1000;
139
- }
140
- this.logger = getLogger("LnForGas("+this.getId()+"): ");
141
- }
142
-
143
- /**
144
- * @inheritDoc
145
- * @internal
146
- */
147
- protected upgradeVersion() {
148
- if(this.version == 1) {
149
- if(this._state===1) this._state = LnForGasSwapState.FINISHED;
150
- this.version = 2;
151
- }
152
- if(this.version == null) {
153
- //Noop
154
- this.version = 1;
155
- }
156
- }
157
-
158
- /**
159
- * @inheritDoc
160
- * @internal
161
- */
162
- protected tryRecomputeSwapPrice() {
163
- if(this.swapFeeBtc==null && this.swapFee!=null) {
164
- this.swapFeeBtc = this.swapFee * this.getInput().rawAmount / this.getOutAmountWithoutFee();
165
- }
166
- super.tryRecomputeSwapPrice();
167
- }
168
-
169
-
170
- //////////////////////////////
171
- //// Getters & utils
172
-
173
- /**
174
- * @inheritDoc
175
- * @internal
176
- */
177
- _getEscrowHash(): string {
178
- return this.getId();
179
- }
180
-
181
- /**
182
- * @inheritDoc
183
- */
184
- getOutputAddress(): string | null {
185
- return this.recipient;
186
- }
187
-
188
- /**
189
- * @inheritDoc
190
- */
191
- getInputAddress(): string | null {
192
- return this.pr;
193
- }
194
-
195
- /**
196
- * @inheritDoc
197
- */
198
- getInputTxId(): string | null {
199
- return this.getId();
200
- }
201
-
202
- /**
203
- * @inheritDoc
204
- */
205
- getOutputTxId(): string | null {
206
- return this.scTxId ?? null;
207
- }
208
-
209
- /**
210
- * @inheritDoc
211
- */
212
- getId(): string {
213
- if(this.pr==null) throw new Error("No payment request assigned to this swap!");
214
- const decodedPR = bolt11Decode(this.pr);
215
- if(decodedPR.tagsObject.payment_hash==null) throw new Error("Lightning invoice has no payment hash!");
216
- return decodedPR.tagsObject.payment_hash;
217
- }
218
-
219
- /**
220
- * Returns the lightning network BOLT11 invoice that needs to be paid as an input to the swap
221
- */
222
- getAddress(): string {
223
- return this.pr;
224
- }
225
-
226
- /**
227
- * Returns a string that can be displayed as QR code representation of the lightning invoice (with lightning: prefix)
228
- */
229
- getHyperlink(): string {
230
- return "lightning:"+this.pr.toUpperCase();
231
- }
232
-
233
- /**
234
- * @inheritDoc
235
- */
236
- requiresAction(): boolean {
237
- return false;
238
- }
239
-
240
- /**
241
- * @inheritDoc
242
- */
243
- isFinished(): boolean {
244
- return this._state===LnForGasSwapState.FINISHED || this._state===LnForGasSwapState.FAILED || this._state===LnForGasSwapState.EXPIRED;
245
- }
246
-
247
- /**
248
- * @inheritDoc
249
- */
250
- isQuoteExpired(): boolean {
251
- return this._state===LnForGasSwapState.EXPIRED;
252
- }
253
-
254
- /**
255
- * @inheritDoc
256
- */
257
- isQuoteSoftExpired(): boolean {
258
- return this.expiry<Date.now();
259
- }
260
-
261
- /**
262
- * @inheritDoc
263
- */
264
- isFailed(): boolean {
265
- return this._state===LnForGasSwapState.FAILED;
266
- }
267
-
268
- /**
269
- * @inheritDoc
270
- */
271
- isSuccessful(): boolean {
272
- return this._state===LnForGasSwapState.FINISHED;
273
- }
274
-
275
- /**
276
- * @inheritDoc
277
- */
278
- isInProgress(): boolean {
279
- return (this._state===LnForGasSwapState.PR_CREATED && this.initiated) || this._state===LnForGasSwapState.PR_PAID;
280
- }
281
-
282
- /**
283
- * @inheritDoc
284
- * @internal
285
- */
286
- _verifyQuoteDefinitelyExpired(): Promise<boolean> {
287
- return Promise.resolve(this.expiry<Date.now());
288
- }
289
-
290
- /**
291
- * @inheritDoc
292
- * @internal
293
- */
294
- _verifyQuoteValid(): Promise<boolean> {
295
- return Promise.resolve(this.expiry>Date.now());
296
- }
297
-
298
- //////////////////////////////
299
- //// Amounts & fees
300
-
301
- /**
302
- * Returns an output amount in base units without a swap fee included, hence this value
303
- * is larger than the actual output amount
304
- *
305
- * @internal
306
- */
307
- protected getOutAmountWithoutFee(): bigint {
308
- return this.outputAmount + (this.swapFee ?? 0n);
309
- }
310
-
311
- /**
312
- * @inheritDoc
313
- */
314
- getOutputToken(): SCToken<T["ChainId"]> {
315
- return this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()];
316
- }
317
-
318
- /**
319
- * @inheritDoc
320
- */
321
- getOutput(): TokenAmount<SCToken<T["ChainId"]>, true> {
322
- return toTokenAmount(
323
- this.outputAmount, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()],
324
- this.wrapper._prices, this.pricingInfo
325
- );
326
- }
327
-
328
- /**
329
- * @inheritDoc
330
- */
331
- getInputToken(): BtcToken<true> {
332
- return BitcoinTokens.BTCLN;
333
- }
334
-
335
- /**
336
- * @inheritDoc
337
- */
338
- getInput(): TokenAmount<BtcToken<true>, true> {
339
- const parsed = bolt11Decode(this.pr);
340
- const msats = parsed.millisatoshis;
341
- if(msats==null) throw new Error("Swap lightning invoice has no msat amount field!");
342
- const amount = (BigInt(msats) + 999n) / 1000n;
343
- return toTokenAmount(amount, BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
344
- }
345
-
346
- /**
347
- * @inheritDoc
348
- */
349
- getInputWithoutFee(): TokenAmount<BtcToken<true>, true> {
350
- const parsed = bolt11Decode(this.pr);
351
- const msats = parsed.millisatoshis;
352
- if(msats==null) throw new Error("Swap lightning invoice has no msat amount field!");
353
- const amount = (BigInt(msats) + 999n) / 1000n;
354
- return toTokenAmount(
355
- amount - (this.swapFeeBtc ?? 0n), BitcoinTokens.BTCLN,
356
- this.wrapper._prices, this.pricingInfo
357
- );
358
- }
359
-
360
- /**
361
- * Returns the swap fee charged by the intermediary (LP) on this swap
362
- *
363
- * @internal
364
- */
365
- protected getSwapFee(): Fee<T["ChainId"], BtcToken<true>, SCToken<T["ChainId"]>> {
366
- if(this.pricingInfo==null) throw new Error("No pricing info known, cannot estimate swap fee!");
367
- const feeWithoutBaseFee = this.swapFeeBtc==null ? 0n : this.swapFeeBtc - this.pricingInfo.satsBaseFee;
368
- const swapFeePPM = feeWithoutBaseFee * 1000000n / this.getInputWithoutFee().rawAmount;
369
-
370
- const amountInSrcToken = toTokenAmount(this.swapFeeBtc ?? 0n, BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
371
- return {
372
- amountInSrcToken,
373
- amountInDstToken: toTokenAmount(this.swapFee ?? 0n, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo),
374
- currentUsdValue: amountInSrcToken.currentUsdValue,
375
- usdValue: amountInSrcToken.usdValue,
376
- pastUsdValue: amountInSrcToken.pastUsdValue,
377
- composition: {
378
- base: toTokenAmount(this.pricingInfo.satsBaseFee, BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo),
379
- percentage: ppmToPercentage(swapFeePPM)
380
- }
381
- };
382
- }
383
-
384
- /**
385
- * @inheritDoc
386
- */
387
- getFee(): Fee<T["ChainId"], BtcToken<true>, SCToken<T["ChainId"]>> {
388
- return this.getSwapFee();
389
- }
390
-
391
- /**
392
- * @inheritDoc
393
- */
394
- getFeeBreakdown(): [{type: FeeType.SWAP, fee: Fee<T["ChainId"], BtcToken<true>, SCToken<T["ChainId"]>>}] {
395
- return [{
396
- type: FeeType.SWAP,
397
- fee: this.getSwapFee()
398
- }];
399
- }
400
-
401
-
402
- //////////////////////////////
403
- //// Payment
404
-
405
- /**
406
- * @inheritDoc
407
- */
408
- async txsExecute(): Promise<[SwapExecutionActionLightning]> {
409
- if(this._state===LnForGasSwapState.PR_CREATED) {
410
- if (!await this._verifyQuoteValid()) throw new Error("Quote already expired or close to expiry!");
411
- return [
412
- {
413
- name: "Payment" as const,
414
- description: "Initiates the swap by paying up the lightning network invoice",
415
- chain: "LIGHTNING",
416
- txs: [
417
- {
418
- type: "BOLT11_PAYMENT_REQUEST",
419
- address: this.pr,
420
- hyperlink: this.getHyperlink()
421
- }
422
- ]
423
- }
424
- ];
425
- }
426
-
427
- throw new Error("Invalid swap state to obtain execution txns, required PR_CREATED");
428
- }
429
-
430
- /**
431
- * @remark Not supported
432
- */
433
- async execute(): Promise<boolean> {
434
- throw new Error("Not supported");
435
- }
436
-
437
- /**
438
- * @inheritDoc
439
- */
440
- async getCurrentActions(): Promise<SwapExecutionAction<T>[]> {
441
- try {
442
- return await this.txsExecute();
443
- } catch (e) {
444
- return [];
445
- }
446
- }
447
-
448
- /**
449
- * Queries the intermediary (LP) node for the state of the swap
450
- *
451
- * @param save Whether the save the result or not
452
- *
453
- * @returns Whether the swap was successful as `boolean` or `null` if the swap is still pending
454
- * @internal
455
- */
456
- protected async checkInvoicePaid(save: boolean = true): Promise<boolean | null> {
457
- if(this._state===LnForGasSwapState.FAILED || this._state===LnForGasSwapState.EXPIRED) return false;
458
- if(this._state===LnForGasSwapState.FINISHED) return true;
459
- if(this.url==null) return false;
460
-
461
- const decodedPR = bolt11Decode(this.pr);
462
- const paymentHash = decodedPR.tagsObject.payment_hash;
463
- if(paymentHash==null) throw new Error("Invalid swap invoice, payment hash not found!");
464
-
465
- const response = await TrustedIntermediaryAPI.getInvoiceStatus(
466
- this.url, paymentHash, this.wrapper._options.getRequestTimeout
467
- );
468
- this.logger.debug("checkInvoicePaid(): LP response: ", response);
469
- switch(response.code) {
470
- case InvoiceStatusResponseCodes.PAID:
471
- this.scTxId = response.data.txId;
472
- const txStatus = await this.wrapper._chain.getTxIdStatus(this.scTxId);
473
- if(txStatus==="success") {
474
- this._state = LnForGasSwapState.FINISHED;
475
- if(save) await this._saveAndEmit();
476
- return true;
477
- }
478
- return null;
479
- case InvoiceStatusResponseCodes.EXPIRED:
480
- if(this._state===LnForGasSwapState.PR_CREATED) {
481
- this._state = LnForGasSwapState.EXPIRED;
482
- } else {
483
- this._state = LnForGasSwapState.FAILED;
484
- }
485
- if(save) await this._saveAndEmit();
486
- return false;
487
- case InvoiceStatusResponseCodes.TX_SENT:
488
- this.scTxId = response.data.txId;
489
- if(this._state===LnForGasSwapState.PR_CREATED) {
490
- this._state = LnForGasSwapState.PR_PAID;
491
- if(save) await this._saveAndEmit();
492
- }
493
- return null;
494
- case InvoiceStatusResponseCodes.PENDING:
495
- if(this._state===LnForGasSwapState.PR_CREATED) {
496
- this._state = LnForGasSwapState.PR_PAID;
497
- if(save) await this._saveAndEmit();
498
- }
499
- return null;
500
- case InvoiceStatusResponseCodes.AWAIT_PAYMENT:
501
- return null;
502
- default:
503
- this._state = LnForGasSwapState.FAILED;
504
- if(save) await this._saveAndEmit();
505
- return false;
506
- }
507
- }
508
-
509
- /**
510
- * A blocking promise resolving when payment was received by the intermediary and client can continue,
511
- * rejecting in case of failure. The swap must be in {@link LnForGasSwapState.PR_CREATED} state!
512
- *
513
- * @param checkIntervalSeconds How often to poll the intermediary for answer (default 5 seconds)
514
- * @param abortSignal Abort signal
515
- * @throws {Error} When in invalid state (not PR_CREATED)
516
- */
517
- async waitForPayment(checkIntervalSeconds?: number, abortSignal?: AbortSignal): Promise<boolean> {
518
- if(this._state!==LnForGasSwapState.PR_CREATED) throw new Error("Must be in PR_CREATED state!");
519
-
520
- if(!this.initiated) {
521
- this.initiated = true;
522
- await this._saveAndEmit();
523
- }
524
-
525
- while(!abortSignal?.aborted && (this._state===LnForGasSwapState.PR_CREATED || this._state===LnForGasSwapState.PR_PAID)) {
526
- await this.checkInvoicePaid(true);
527
- if(this._state===LnForGasSwapState.PR_CREATED || this._state===LnForGasSwapState.PR_PAID) await timeoutPromise((checkIntervalSeconds ?? 5)*1000, abortSignal);
528
- }
529
-
530
- if(this.isFailed()) throw new Error("Swap failed");
531
- return !this.isQuoteExpired();
532
-
533
- }
534
-
535
-
536
- //////////////////////////////
537
- //// Storage
538
-
539
- /**
540
- * @inheritDoc
541
- */
542
- serialize(): any{
543
- return {
544
- ...super.serialize(),
545
- pr: this.pr,
546
- outputAmount: this.outputAmount==null ? null : this.outputAmount.toString(10),
547
- recipient: this.recipient,
548
- token: this.token,
549
- scTxId: this.scTxId
550
- };
551
- }
552
-
553
- /**
554
- * @inheritDoc
555
- * @internal
556
- */
557
- _getInitiator(): string {
558
- return this.recipient;
559
- }
560
-
561
-
562
- //////////////////////////////
563
- //// Swap ticks & sync
564
-
565
- /**
566
- * @inheritDoc
567
- * @internal
568
- */
569
- async _sync(save?: boolean): Promise<boolean> {
570
- if(this._state===LnForGasSwapState.PR_CREATED) {
571
- //Check if it's maybe already paid
572
- const res = await this.checkInvoicePaid(false);
573
- if(res!==null) {
574
- if(save) await this._saveAndEmit();
575
- return true;
576
- }
577
- }
578
- return false;
579
- }
580
-
581
- /**
582
- * @inheritDoc
583
- * @internal
584
- */
585
- _tick(save?: boolean): Promise<boolean> {
586
- return Promise.resolve(false);
587
- }
588
-
589
- }
1
+ import {decode as bolt11Decode} from "@atomiqlabs/bolt11";
2
+ import {SwapType} from "../../../enums/SwapType";
3
+ import {ChainType} from "@atomiqlabs/base";
4
+ import {LnForGasSwapTypeDefinition, LnForGasWrapper} from "./LnForGasWrapper";
5
+ import {toBigInt} from "../../../utils/Utils";
6
+ import {isISwapInit, ISwap, ISwapInit} from "../../ISwap";
7
+ import {InvoiceStatusResponseCodes, TrustedIntermediaryAPI} from "../../../intermediaries/apis/TrustedIntermediaryAPI";
8
+ import {Fee} from "../../../types/fees/Fee";
9
+ import {IAddressSwap} from "../../IAddressSwap";
10
+ import {FeeType} from "../../../enums/FeeType";
11
+ import {ppmToPercentage} from "../../../types/fees/PercentagePPM";
12
+ import {TokenAmount, toTokenAmount} from "../../../types/TokenAmount";
13
+ import {BitcoinTokens, BtcToken, SCToken} from "../../../types/Token";
14
+ import {getLogger, LoggerType} from "../../../utils/Logger";
15
+ import {timeoutPromise} from "../../../utils/TimeoutUtils";
16
+ import {SwapExecutionAction, SwapExecutionActionLightning} from "../../../types/SwapExecutionAction";
17
+
18
+ /**
19
+ * State enum for trusted Lightning gas swaps
20
+ *
21
+ * @category Swaps/Trusted Gas Swaps
22
+ */
23
+ export enum LnForGasSwapState {
24
+ /**
25
+ * The swap quote expired without user sending in the lightning network payment
26
+ */
27
+ EXPIRED = -2,
28
+ /**
29
+ * The swap has failed after the intermediary already received a lightning network payment on the source
30
+ */
31
+ FAILED = -1,
32
+ /**
33
+ * Swap was created, pay the provided lightning network invoice
34
+ */
35
+ PR_CREATED = 0,
36
+ /**
37
+ * User paid the lightning network invoice on the source
38
+ */
39
+ PR_PAID = 1,
40
+ /**
41
+ * The swap is finished after the intermediary sent funds on the destination chain
42
+ */
43
+ FINISHED = 2
44
+ }
45
+
46
+ const LnForGasSwapStateDescription = {
47
+ [LnForGasSwapState.EXPIRED]:
48
+ "The swap quote expired without user sending in the lightning network payment",
49
+ [LnForGasSwapState.FAILED]:
50
+ "The swap has failed after the intermediary already received a lightning network payment on the source",
51
+ [LnForGasSwapState.PR_CREATED]:
52
+ "Swap was created, pay the provided lightning network invoice",
53
+ [LnForGasSwapState.PR_PAID]:
54
+ "User paid the lightning network invoice on the source",
55
+ [LnForGasSwapState.FINISHED]:
56
+ "The swap is finished after the intermediary sent funds on the destination chain"
57
+ }
58
+
59
+ export type LnForGasSwapInit = ISwapInit & {
60
+ pr: string;
61
+ outputAmount: bigint;
62
+ recipient: string;
63
+ token: string;
64
+ };
65
+
66
+ export function isLnForGasSwapInit(obj: any): obj is LnForGasSwapInit {
67
+ return typeof(obj.pr)==="string" &&
68
+ typeof(obj.outputAmount) === "bigint" &&
69
+ typeof(obj.recipient)==="string" &&
70
+ typeof(obj.token)==="string" &&
71
+ isISwapInit(obj);
72
+ }
73
+
74
+ /**
75
+ * Trusted swap for Bitcoin Lightning -> Smart chains, to be used for minor amounts to get gas tokens on
76
+ * the destination chain, which is only needed for Solana, which still uses legacy swaps
77
+ *
78
+ * @category Swaps/Trusted Gas Swaps
79
+ */
80
+ export class LnForGasSwap<T extends ChainType = ChainType> extends ISwap<T, LnForGasSwapTypeDefinition<T>, LnForGasSwapState> implements IAddressSwap {
81
+ protected readonly TYPE: SwapType.TRUSTED_FROM_BTCLN = SwapType.TRUSTED_FROM_BTCLN;
82
+
83
+ /**
84
+ * @internal
85
+ */
86
+ protected readonly swapStateDescription = LnForGasSwapStateDescription;
87
+ /**
88
+ * @internal
89
+ */
90
+ protected readonly swapStateName = (state: number) => LnForGasSwapState[state];
91
+
92
+ /**
93
+ * @internal
94
+ */
95
+ protected readonly currentVersion: number = 2;
96
+ /**
97
+ * @internal
98
+ */
99
+ protected readonly logger: LoggerType;
100
+
101
+ //State: PR_CREATED
102
+ private readonly pr: string;
103
+ private readonly outputAmount: bigint;
104
+ private readonly recipient: string;
105
+ private readonly token: string;
106
+
107
+ //State: FINISHED
108
+ /**
109
+ * Destination transaction ID on the smart chain side
110
+ * @private
111
+ */
112
+ private scTxId?: string;
113
+
114
+ constructor(wrapper: LnForGasWrapper<T>, init: LnForGasSwapInit);
115
+ constructor(wrapper: LnForGasWrapper<T>, obj: any);
116
+ constructor(
117
+ wrapper: LnForGasWrapper<T>,
118
+ initOrObj: LnForGasSwapInit | any
119
+ ) {
120
+ if(isLnForGasSwapInit(initOrObj) && initOrObj.url!=null) initOrObj.url += "/lnforgas";
121
+ super(wrapper, initOrObj);
122
+ if(isLnForGasSwapInit(initOrObj)) {
123
+ this.pr = initOrObj.pr;
124
+ this.outputAmount = initOrObj.outputAmount;
125
+ this.recipient = initOrObj.recipient;
126
+ this.token = initOrObj.token;
127
+ this._state = LnForGasSwapState.PR_CREATED;
128
+ } else {
129
+ this.pr = initOrObj.pr;
130
+ this.outputAmount = toBigInt(initOrObj.outputAmount);
131
+ this.recipient = initOrObj.recipient;
132
+ this.token = initOrObj.token;
133
+ this.scTxId = initOrObj.scTxId;
134
+ }
135
+ this.tryRecomputeSwapPrice();
136
+ if(this.pr!=null) {
137
+ const decoded = bolt11Decode(this.pr);
138
+ if(decoded.timeExpireDate!=null) this.expiry = decoded.timeExpireDate*1000;
139
+ }
140
+ this.logger = getLogger("LnForGas("+this.getId()+"): ");
141
+ }
142
+
143
+ /**
144
+ * @inheritDoc
145
+ * @internal
146
+ */
147
+ protected upgradeVersion() {
148
+ if(this.version == 1) {
149
+ if(this._state===1) this._state = LnForGasSwapState.FINISHED;
150
+ this.version = 2;
151
+ }
152
+ if(this.version == null) {
153
+ //Noop
154
+ this.version = 1;
155
+ }
156
+ }
157
+
158
+ /**
159
+ * @inheritDoc
160
+ * @internal
161
+ */
162
+ protected tryRecomputeSwapPrice() {
163
+ if(this.swapFeeBtc==null && this.swapFee!=null) {
164
+ this.swapFeeBtc = this.swapFee * this.getInput().rawAmount / this.getOutAmountWithoutFee();
165
+ }
166
+ super.tryRecomputeSwapPrice();
167
+ }
168
+
169
+
170
+ //////////////////////////////
171
+ //// Getters & utils
172
+
173
+ /**
174
+ * @inheritDoc
175
+ * @internal
176
+ */
177
+ _getEscrowHash(): string {
178
+ return this.getId();
179
+ }
180
+
181
+ /**
182
+ * @inheritDoc
183
+ */
184
+ getOutputAddress(): string | null {
185
+ return this.recipient;
186
+ }
187
+
188
+ /**
189
+ * @inheritDoc
190
+ */
191
+ getInputAddress(): string | null {
192
+ return this.pr;
193
+ }
194
+
195
+ /**
196
+ * @inheritDoc
197
+ */
198
+ getInputTxId(): string | null {
199
+ return this.getId();
200
+ }
201
+
202
+ /**
203
+ * @inheritDoc
204
+ */
205
+ getOutputTxId(): string | null {
206
+ return this.scTxId ?? null;
207
+ }
208
+
209
+ /**
210
+ * @inheritDoc
211
+ */
212
+ getId(): string {
213
+ if(this.pr==null) throw new Error("No payment request assigned to this swap!");
214
+ const decodedPR = bolt11Decode(this.pr);
215
+ if(decodedPR.tagsObject.payment_hash==null) throw new Error("Lightning invoice has no payment hash!");
216
+ return decodedPR.tagsObject.payment_hash;
217
+ }
218
+
219
+ /**
220
+ * Returns the lightning network BOLT11 invoice that needs to be paid as an input to the swap
221
+ */
222
+ getAddress(): string {
223
+ return this.pr;
224
+ }
225
+
226
+ /**
227
+ * Returns a string that can be displayed as QR code representation of the lightning invoice (with lightning: prefix)
228
+ */
229
+ getHyperlink(): string {
230
+ return "lightning:"+this.pr.toUpperCase();
231
+ }
232
+
233
+ /**
234
+ * @inheritDoc
235
+ */
236
+ requiresAction(): boolean {
237
+ return false;
238
+ }
239
+
240
+ /**
241
+ * @inheritDoc
242
+ */
243
+ isFinished(): boolean {
244
+ return this._state===LnForGasSwapState.FINISHED || this._state===LnForGasSwapState.FAILED || this._state===LnForGasSwapState.EXPIRED;
245
+ }
246
+
247
+ /**
248
+ * @inheritDoc
249
+ */
250
+ isQuoteExpired(): boolean {
251
+ return this._state===LnForGasSwapState.EXPIRED;
252
+ }
253
+
254
+ /**
255
+ * @inheritDoc
256
+ */
257
+ isQuoteSoftExpired(): boolean {
258
+ return this.expiry<Date.now();
259
+ }
260
+
261
+ /**
262
+ * @inheritDoc
263
+ */
264
+ isFailed(): boolean {
265
+ return this._state===LnForGasSwapState.FAILED;
266
+ }
267
+
268
+ /**
269
+ * @inheritDoc
270
+ */
271
+ isSuccessful(): boolean {
272
+ return this._state===LnForGasSwapState.FINISHED;
273
+ }
274
+
275
+ /**
276
+ * @inheritDoc
277
+ */
278
+ isInProgress(): boolean {
279
+ return (this._state===LnForGasSwapState.PR_CREATED && this.initiated) || this._state===LnForGasSwapState.PR_PAID;
280
+ }
281
+
282
+ /**
283
+ * @inheritDoc
284
+ * @internal
285
+ */
286
+ _verifyQuoteDefinitelyExpired(): Promise<boolean> {
287
+ return Promise.resolve(this.expiry<Date.now());
288
+ }
289
+
290
+ /**
291
+ * @inheritDoc
292
+ * @internal
293
+ */
294
+ _verifyQuoteValid(): Promise<boolean> {
295
+ return Promise.resolve(this.expiry>Date.now());
296
+ }
297
+
298
+ //////////////////////////////
299
+ //// Amounts & fees
300
+
301
+ /**
302
+ * Returns an output amount in base units without a swap fee included, hence this value
303
+ * is larger than the actual output amount
304
+ *
305
+ * @internal
306
+ */
307
+ protected getOutAmountWithoutFee(): bigint {
308
+ return this.outputAmount + (this.swapFee ?? 0n);
309
+ }
310
+
311
+ /**
312
+ * @inheritDoc
313
+ */
314
+ getOutputToken(): SCToken<T["ChainId"]> {
315
+ return this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()];
316
+ }
317
+
318
+ /**
319
+ * @inheritDoc
320
+ */
321
+ getOutput(): TokenAmount<SCToken<T["ChainId"]>, true> {
322
+ return toTokenAmount(
323
+ this.outputAmount, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()],
324
+ this.wrapper._prices, this.pricingInfo
325
+ );
326
+ }
327
+
328
+ /**
329
+ * @inheritDoc
330
+ */
331
+ getInputToken(): BtcToken<true> {
332
+ return BitcoinTokens.BTCLN;
333
+ }
334
+
335
+ /**
336
+ * @inheritDoc
337
+ */
338
+ getInput(): TokenAmount<BtcToken<true>, true> {
339
+ const parsed = bolt11Decode(this.pr);
340
+ const msats = parsed.millisatoshis;
341
+ if(msats==null) throw new Error("Swap lightning invoice has no msat amount field!");
342
+ const amount = (BigInt(msats) + 999n) / 1000n;
343
+ return toTokenAmount(amount, BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
344
+ }
345
+
346
+ /**
347
+ * @inheritDoc
348
+ */
349
+ getInputWithoutFee(): TokenAmount<BtcToken<true>, true> {
350
+ const parsed = bolt11Decode(this.pr);
351
+ const msats = parsed.millisatoshis;
352
+ if(msats==null) throw new Error("Swap lightning invoice has no msat amount field!");
353
+ const amount = (BigInt(msats) + 999n) / 1000n;
354
+ return toTokenAmount(
355
+ amount - (this.swapFeeBtc ?? 0n), BitcoinTokens.BTCLN,
356
+ this.wrapper._prices, this.pricingInfo
357
+ );
358
+ }
359
+
360
+ /**
361
+ * Returns the swap fee charged by the intermediary (LP) on this swap
362
+ *
363
+ * @internal
364
+ */
365
+ protected getSwapFee(): Fee<T["ChainId"], BtcToken<true>, SCToken<T["ChainId"]>> {
366
+ if(this.pricingInfo==null) throw new Error("No pricing info known, cannot estimate swap fee!");
367
+ const feeWithoutBaseFee = this.swapFeeBtc==null ? 0n : this.swapFeeBtc - this.pricingInfo.satsBaseFee;
368
+ const swapFeePPM = feeWithoutBaseFee * 1000000n / this.getInputWithoutFee().rawAmount;
369
+
370
+ const amountInSrcToken = toTokenAmount(this.swapFeeBtc ?? 0n, BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
371
+ return {
372
+ amountInSrcToken,
373
+ amountInDstToken: toTokenAmount(this.swapFee ?? 0n, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo),
374
+ currentUsdValue: amountInSrcToken.currentUsdValue,
375
+ usdValue: amountInSrcToken.usdValue,
376
+ pastUsdValue: amountInSrcToken.pastUsdValue,
377
+ composition: {
378
+ base: toTokenAmount(this.pricingInfo.satsBaseFee, BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo),
379
+ percentage: ppmToPercentage(swapFeePPM)
380
+ }
381
+ };
382
+ }
383
+
384
+ /**
385
+ * @inheritDoc
386
+ */
387
+ getFee(): Fee<T["ChainId"], BtcToken<true>, SCToken<T["ChainId"]>> {
388
+ return this.getSwapFee();
389
+ }
390
+
391
+ /**
392
+ * @inheritDoc
393
+ */
394
+ getFeeBreakdown(): [{type: FeeType.SWAP, fee: Fee<T["ChainId"], BtcToken<true>, SCToken<T["ChainId"]>>}] {
395
+ return [{
396
+ type: FeeType.SWAP,
397
+ fee: this.getSwapFee()
398
+ }];
399
+ }
400
+
401
+
402
+ //////////////////////////////
403
+ //// Payment
404
+
405
+ /**
406
+ * @inheritDoc
407
+ */
408
+ async txsExecute(): Promise<[SwapExecutionActionLightning]> {
409
+ if(this._state===LnForGasSwapState.PR_CREATED) {
410
+ if (!await this._verifyQuoteValid()) throw new Error("Quote already expired or close to expiry!");
411
+ return [
412
+ {
413
+ name: "Payment" as const,
414
+ description: "Initiates the swap by paying up the lightning network invoice",
415
+ chain: "LIGHTNING",
416
+ txs: [
417
+ {
418
+ type: "BOLT11_PAYMENT_REQUEST",
419
+ address: this.pr,
420
+ hyperlink: this.getHyperlink()
421
+ }
422
+ ]
423
+ }
424
+ ];
425
+ }
426
+
427
+ throw new Error("Invalid swap state to obtain execution txns, required PR_CREATED");
428
+ }
429
+
430
+ /**
431
+ * @remark Not supported
432
+ */
433
+ async execute(): Promise<boolean> {
434
+ throw new Error("Not supported");
435
+ }
436
+
437
+ /**
438
+ * @inheritDoc
439
+ */
440
+ async getCurrentActions(): Promise<SwapExecutionAction<T>[]> {
441
+ try {
442
+ return await this.txsExecute();
443
+ } catch (e) {
444
+ return [];
445
+ }
446
+ }
447
+
448
+ /**
449
+ * Queries the intermediary (LP) node for the state of the swap
450
+ *
451
+ * @param save Whether the save the result or not
452
+ *
453
+ * @returns Whether the swap was successful as `boolean` or `null` if the swap is still pending
454
+ * @internal
455
+ */
456
+ protected async checkInvoicePaid(save: boolean = true): Promise<boolean | null> {
457
+ if(this._state===LnForGasSwapState.FAILED || this._state===LnForGasSwapState.EXPIRED) return false;
458
+ if(this._state===LnForGasSwapState.FINISHED) return true;
459
+ if(this.url==null) return false;
460
+
461
+ const decodedPR = bolt11Decode(this.pr);
462
+ const paymentHash = decodedPR.tagsObject.payment_hash;
463
+ if(paymentHash==null) throw new Error("Invalid swap invoice, payment hash not found!");
464
+
465
+ const response = await TrustedIntermediaryAPI.getInvoiceStatus(
466
+ this.url, paymentHash, this.wrapper._options.getRequestTimeout
467
+ );
468
+ this.logger.debug("checkInvoicePaid(): LP response: ", response);
469
+ switch(response.code) {
470
+ case InvoiceStatusResponseCodes.PAID:
471
+ this.scTxId = response.data.txId;
472
+ const txStatus = await this.wrapper._chain.getTxIdStatus(this.scTxId);
473
+ if(txStatus==="success") {
474
+ this._state = LnForGasSwapState.FINISHED;
475
+ if(save) await this._saveAndEmit();
476
+ return true;
477
+ }
478
+ return null;
479
+ case InvoiceStatusResponseCodes.EXPIRED:
480
+ if(this._state===LnForGasSwapState.PR_CREATED) {
481
+ this._state = LnForGasSwapState.EXPIRED;
482
+ } else {
483
+ this._state = LnForGasSwapState.FAILED;
484
+ }
485
+ if(save) await this._saveAndEmit();
486
+ return false;
487
+ case InvoiceStatusResponseCodes.TX_SENT:
488
+ this.scTxId = response.data.txId;
489
+ if(this._state===LnForGasSwapState.PR_CREATED) {
490
+ this._state = LnForGasSwapState.PR_PAID;
491
+ if(save) await this._saveAndEmit();
492
+ }
493
+ return null;
494
+ case InvoiceStatusResponseCodes.PENDING:
495
+ if(this._state===LnForGasSwapState.PR_CREATED) {
496
+ this._state = LnForGasSwapState.PR_PAID;
497
+ if(save) await this._saveAndEmit();
498
+ }
499
+ return null;
500
+ case InvoiceStatusResponseCodes.AWAIT_PAYMENT:
501
+ return null;
502
+ default:
503
+ this._state = LnForGasSwapState.FAILED;
504
+ if(save) await this._saveAndEmit();
505
+ return false;
506
+ }
507
+ }
508
+
509
+ /**
510
+ * A blocking promise resolving when payment was received by the intermediary and client can continue,
511
+ * rejecting in case of failure. The swap must be in {@link LnForGasSwapState.PR_CREATED} state!
512
+ *
513
+ * @param checkIntervalSeconds How often to poll the intermediary for answer (default 5 seconds)
514
+ * @param abortSignal Abort signal
515
+ * @throws {Error} When in invalid state (not PR_CREATED)
516
+ */
517
+ async waitForPayment(checkIntervalSeconds?: number, abortSignal?: AbortSignal): Promise<boolean> {
518
+ if(this._state!==LnForGasSwapState.PR_CREATED) throw new Error("Must be in PR_CREATED state!");
519
+
520
+ if(!this.initiated) {
521
+ this.initiated = true;
522
+ await this._saveAndEmit();
523
+ }
524
+
525
+ while(!abortSignal?.aborted && (this._state===LnForGasSwapState.PR_CREATED || this._state===LnForGasSwapState.PR_PAID)) {
526
+ await this.checkInvoicePaid(true);
527
+ if(this._state===LnForGasSwapState.PR_CREATED || this._state===LnForGasSwapState.PR_PAID) await timeoutPromise((checkIntervalSeconds ?? 5)*1000, abortSignal);
528
+ }
529
+
530
+ if(this.isFailed()) throw new Error("Swap failed");
531
+ return !this.isQuoteExpired();
532
+
533
+ }
534
+
535
+
536
+ //////////////////////////////
537
+ //// Storage
538
+
539
+ /**
540
+ * @inheritDoc
541
+ */
542
+ serialize(): any{
543
+ return {
544
+ ...super.serialize(),
545
+ pr: this.pr,
546
+ outputAmount: this.outputAmount==null ? null : this.outputAmount.toString(10),
547
+ recipient: this.recipient,
548
+ token: this.token,
549
+ scTxId: this.scTxId
550
+ };
551
+ }
552
+
553
+ /**
554
+ * @inheritDoc
555
+ * @internal
556
+ */
557
+ _getInitiator(): string {
558
+ return this.recipient;
559
+ }
560
+
561
+
562
+ //////////////////////////////
563
+ //// Swap ticks & sync
564
+
565
+ /**
566
+ * @inheritDoc
567
+ * @internal
568
+ */
569
+ async _sync(save?: boolean): Promise<boolean> {
570
+ if(this._state===LnForGasSwapState.PR_CREATED) {
571
+ //Check if it's maybe already paid
572
+ const res = await this.checkInvoicePaid(false);
573
+ if(res!==null) {
574
+ if(save) await this._saveAndEmit();
575
+ return true;
576
+ }
577
+ }
578
+ return false;
579
+ }
580
+
581
+ /**
582
+ * @inheritDoc
583
+ * @internal
584
+ */
585
+ _tick(save?: boolean): Promise<boolean> {
586
+ return Promise.resolve(false);
587
+ }
588
+
589
+ }