@atomiqlabs/sdk 8.7.7 → 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 -6
  6. package/dist/bitcoin/coinselect2/accumulative.js +52 -52
  7. package/dist/bitcoin/coinselect2/blackjack.d.ts +7 -6
  8. package/dist/bitcoin/coinselect2/blackjack.js +38 -38
  9. package/dist/bitcoin/coinselect2/index.d.ts +20 -19
  10. package/dist/bitcoin/coinselect2/index.js +69 -69
  11. package/dist/bitcoin/coinselect2/utils.d.ts +82 -77
  12. package/dist/bitcoin/coinselect2/utils.js +158 -123
  13. package/dist/bitcoin/wallet/BitcoinWallet.d.ts +113 -130
  14. package/dist/bitcoin/wallet/BitcoinWallet.js +335 -322
  15. package/dist/bitcoin/wallet/IBitcoinWallet.d.ts +116 -78
  16. package/dist/bitcoin/wallet/IBitcoinWallet.js +21 -21
  17. package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.d.ts +106 -101
  18. package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.js +196 -190
  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 -440
  60. package/dist/intermediaries/apis/IntermediaryAPI.js +618 -603
  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 -692
  102. package/dist/swapper/Swapper.js +1713 -1657
  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 -631
  164. package/dist/swaps/spv_swaps/SpvFromBTCSwap.js +1448 -1444
  165. package/dist/swaps/spv_swaps/SpvFromBTCWrapper.d.ts +257 -225
  166. package/dist/swaps/spv_swaps/SpvFromBTCWrapper.js +947 -822
  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 -14
  210. package/dist/utils/BitcoinUtils.js +141 -102
  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 -68
  230. package/src/bitcoin/coinselect2/blackjack.ts +50 -49
  231. package/src/bitcoin/coinselect2/index.ts +93 -92
  232. package/src/bitcoin/coinselect2/utils.ts +236 -195
  233. package/src/bitcoin/wallet/BitcoinWallet.ts +439 -427
  234. package/src/bitcoin/wallet/IBitcoinWallet.ts +140 -99
  235. package/src/bitcoin/wallet/SingleAddressBitcoinWallet.ts +225 -217
  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 -947
  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 -2416
  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 -1799
  309. package/src/swaps/spv_swaps/SpvFromBTCWrapper.ts +1236 -1060
  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 -91
  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,403 +1,403 @@
1
- import {RequestError} from "../errors/RequestError";
2
- import {decode as bolt11Decode, PaymentRequestObject, TagsObject} from "@atomiqlabs/bolt11";
3
- import {UserError} from "../errors/UserError";
4
- import {bech32} from "@scure/base";
5
- import {cbc} from "@noble/ciphers/aes";
6
- import {Buffer} from "buffer";
7
- import {sha256} from "@noble/hashes/sha2";
8
- import {httpGet} from "../http/HttpUtils";
9
- import {
10
- isLNURLWithdrawParams,
11
- LNURLWithdraw,
12
- LNURLWithdrawParams,
13
- LNURLWithdrawParamsWithUrl
14
- } from "../types/lnurl/LNURLWithdraw";
15
- import {
16
- isLNURLPayParams,
17
- LNURLDecodedSuccessAction,
18
- LNURLPay,
19
- LNURLPayParams,
20
- LNURLPayParamsWithUrl
21
- } from "../types/lnurl/LNURLPay";
22
- import {tryWithRetries} from "../utils/RetryUtils";
23
-
24
- type LNURLOk = {
25
- status: "OK"
26
- };
27
-
28
- type LNURLError = {
29
- status: "ERROR",
30
- reason?: string
31
- };
32
-
33
- function isLNURLError(obj: any): obj is LNURLError {
34
- return obj.status==="ERROR" &&
35
- (obj.reason==null || typeof obj.reason==="string");
36
- }
37
-
38
- type LNURLPayResult = {
39
- pr: string;
40
- successAction: LNURLPaySuccessAction | null;
41
- disposable: boolean | null;
42
- routes: [];
43
- }
44
-
45
- function isLNURLPayResult(obj: any, domain?: string): obj is LNURLPayResult {
46
- return typeof obj.pr === "string" &&
47
- (obj.routes == null || Array.isArray(obj.routes)) &&
48
- (obj.disposable === null || obj.disposable === undefined || typeof obj.disposable === "boolean") &&
49
- (obj.successAction == null || isLNURLPaySuccessAction(obj.successAction, domain));
50
- }
51
-
52
- export type LNURLPaySuccessAction = {
53
- tag: string;
54
- description: string | null;
55
- url: string | null;
56
- message: string | null;
57
- ciphertext: string | null;
58
- iv: string | null;
59
- };
60
-
61
- export function isLNURLPaySuccessAction(obj: any, domain?: string): obj is LNURLPaySuccessAction {
62
- if(obj==null || typeof obj !== 'object' || typeof obj.tag !== 'string') return false;
63
- switch(obj.tag) {
64
- case "message":
65
- return obj.message!=null && obj.message.length<=144;
66
- case "url":
67
- return obj.description!=null && obj.description.length<=144 &&
68
- obj.url!=null &&
69
- (domain==null || new URL(obj.url).hostname===domain);
70
- case "aes":
71
- return obj.description!=null && obj.description.length<=144 &&
72
- obj.ciphertext!=null && obj.ciphertext.length<=4096 && BASE64_REGEX.test(obj.ciphertext) &&
73
- obj.iv!=null && obj.iv.length<=24 && BASE64_REGEX.test(obj.iv);
74
- default:
75
- //Unsupported action
76
- return false;
77
- }
78
- }
79
-
80
- const BASE64_REGEX = /^(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{2}==|[A-Za-z0-9+/]{3}=)?$/;
81
- const MAIL_REGEX = /(?:[A-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[A-z0-9!#$%&'*+/=?^_`{|}~-]+)*|"(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21\x23-\x5b\x5d-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])*")@(?:(?:[A-z0-9](?:[A-z0-9-]*[A-z0-9])?\.)+[A-z0-9](?:[A-z0-9-]*[A-z0-9])?|\[(?:(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9]))\.){3}(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9])|[A-z0-9-]*[A-z0-9]:(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21-\x5a\x53-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])+)\])/;
82
-
83
- /**
84
- * LNURL encoding/decoding and callback handling
85
- * @category Bitcoin
86
- */
87
- export class LNURL {
88
-
89
- private static findBech32LNURL(str: string) {
90
- const arr = /,*?((lnurl)([0-9]{1,}[a-z0-9]+){1})/.exec(str.toLowerCase());
91
- if(arr==null) return null;
92
- return arr[1];
93
- }
94
-
95
- private static isBech32LNURL(str: string): boolean {
96
- return this.findBech32LNURL(str)!=null;
97
- }
98
-
99
- /**
100
- * Checks whether a provided string is bare (non bech32 encoded) lnurl
101
- * @param str
102
- * @private
103
- */
104
- private static isBareLNURL(str: string): boolean {
105
- try {
106
- return str.startsWith("lnurlw://") || str.startsWith("lnurlp://");
107
- } catch(e) {}
108
- return false;
109
- }
110
-
111
- /**
112
- * Checks if the provided string is a lightning network address (e.g. satoshi@nakamoto.com)
113
- * @param str
114
- * @private
115
- */
116
- private static isLightningAddress(str: string): boolean {
117
- return MAIL_REGEX.test(str);
118
- }
119
-
120
- /**
121
- * Checks whether a given string is a LNURL or lightning address
122
- * @param str
123
- */
124
- static isLNURL(str: string): boolean {
125
- return LNURL.isBech32LNURL(str) || LNURL.isLightningAddress(str) || LNURL.isBareLNURL(str);
126
- }
127
-
128
- /**
129
- * Extracts the URL that needs to be request from LNURL or lightning address
130
- * @param str
131
- * @private
132
- * @returns An URL to send the request to, or null if it cannot be parsed
133
- */
134
- private static extractCallUrl(str: string): string | null {
135
- if(MAIL_REGEX.test(str)) {
136
- //lightning e-mail like address
137
- const arr = str.split("@");
138
- const username = arr[0];
139
- const domain = arr[1];
140
- let scheme = "https";
141
- if(domain.endsWith(".onion")) {
142
- scheme = "http";
143
- }
144
-
145
- return scheme+"://"+domain+"/.well-known/lnurlp/"+username;
146
- } else if(LNURL.isBareLNURL(str)) {
147
- //non-bech32m encoded lnurl
148
- const data = str.substring("lnurlw://".length);
149
- const httpUrl = new URL("http://"+data);
150
-
151
- let scheme = "https";
152
- if(httpUrl.hostname.endsWith(".onion")) {
153
- scheme = "http";
154
- }
155
-
156
- return scheme+"://"+data;
157
- } else {
158
- const lnurl = LNURL.findBech32LNURL(str);
159
-
160
- if(lnurl!=null) {
161
- let { prefix: hrp, words: dataPart } = bech32.decode(lnurl as any, 2000);
162
- let requestByteArray = bech32.fromWords(dataPart);
163
-
164
- return Buffer.from(requestByteArray).toString();
165
- }
166
- }
167
- return null;
168
- }
169
-
170
- /**
171
- * Sends a request to obtain data about a specific LNURL or lightning address
172
- *
173
- * @param str A lnurl or lightning address
174
- * @param shouldRetry Whether we should retry in case of network failure
175
- * @param timeout Request timeout in milliseconds
176
- * @param abortSignal
177
- */
178
- static async getLNURL(
179
- str: string,
180
- shouldRetry: boolean = true,
181
- timeout?: number,
182
- abortSignal?: AbortSignal
183
- ) : Promise<LNURLPayParamsWithUrl | LNURLWithdrawParamsWithUrl | null> {
184
- if(shouldRetry==null) shouldRetry = true;
185
-
186
- const url = LNURL.extractCallUrl(str);
187
- if(url==null) return null;
188
-
189
- const sendRequest =
190
- () => httpGet<LNURLPayParams | LNURLWithdrawParams | LNURLError>(url, timeout, abortSignal, true);
191
-
192
- let response = shouldRetry ?
193
- await tryWithRetries(sendRequest, undefined, RequestError, abortSignal) :
194
- await sendRequest();
195
-
196
- if(isLNURLError(response)) return null;
197
-
198
- if(response.tag==="payRequest") try {
199
- response.decodedMetadata = JSON.parse(response.metadata)
200
- } catch (err) {
201
- response.decodedMetadata = []
202
- }
203
-
204
- if(!isLNURLPayParams(response) && !isLNURLWithdrawParams(response)) return null;
205
-
206
- return {
207
- ...response,
208
- url: str
209
- };
210
- }
211
-
212
- /**
213
- * Sends a request to obtain data about a specific LNURL or lightning address
214
- *
215
- * @param str A lnurl or lightning address
216
- * @param shouldRetry Whether we should retry in case of network failure
217
- * @param timeout Request timeout in milliseconds
218
- * @param abortSignal
219
- */
220
- static async getLNURLType(str: string, shouldRetry?: boolean, timeout?: number, abortSignal?: AbortSignal): Promise<LNURLPay | LNURLWithdraw | null> {
221
- let res: any = await LNURL.getLNURL(str, shouldRetry, timeout, abortSignal);
222
-
223
- if(res.tag==="payRequest") {
224
- const payRequest: LNURLPayParamsWithUrl = res;
225
- let shortDescription: string | undefined = undefined;
226
- let longDescription: string | undefined = undefined;
227
- let icon: string | undefined = undefined;
228
- payRequest.decodedMetadata.forEach(data => {
229
- switch(data[0]) {
230
- case "text/plain":
231
- shortDescription = data[1];
232
- break;
233
- case "text/long-desc":
234
- longDescription = data[1];
235
- break;
236
- case "image/png;base64":
237
- icon = "data:"+data[0]+","+data[1];
238
- break;
239
- case "image/jpeg;base64":
240
- icon = "data:"+data[0]+","+data[1];
241
- break;
242
- }
243
- });
244
- return {
245
- type: "pay",
246
- min: BigInt(payRequest.minSendable) / 1000n,
247
- max: BigInt(payRequest.maxSendable) / 1000n,
248
- commentMaxLength: payRequest.commentAllowed || 0,
249
- shortDescription,
250
- longDescription,
251
- icon,
252
- params: payRequest
253
- }
254
- }
255
- if(res.tag==="withdrawRequest") {
256
- const payRequest: LNURLWithdrawParamsWithUrl = res;
257
- return {
258
- type: "withdraw",
259
- min: BigInt(payRequest.minWithdrawable) / 1000n,
260
- max: BigInt(payRequest.maxWithdrawable) / 1000n,
261
- params: payRequest
262
- }
263
- }
264
- return null;
265
- }
266
-
267
- /**
268
- * Uses a LNURL-pay request by obtaining a lightning network invoice from it
269
- *
270
- * @param payRequest LNURL params as returned from the getLNURL call
271
- * @param amount Amount of sats (BTC) to pay
272
- * @param comment Optional comment for the payment request
273
- * @param timeout Request timeout in milliseconds
274
- * @param abortSignal
275
- * @throws {RequestError} If the response is non-200, status: ERROR, or invalid format
276
- */
277
- static async useLNURLPay(
278
- payRequest: LNURLPayParamsWithUrl,
279
- amount: bigint,
280
- comment?: string,
281
- timeout?: number,
282
- abortSignal?: AbortSignal
283
- ): Promise<{
284
- invoice: string,
285
- parsedInvoice: PaymentRequestObject & { tagsObject: TagsObject; },
286
- successAction?: LNURLPaySuccessAction
287
- }> {
288
- const params = ["amount="+(amount * 1000n).toString(10)];
289
- if(comment!=null) {
290
- params.push("comment="+encodeURIComponent(comment));
291
- }
292
-
293
- const queryParams = (payRequest.callback.includes("?") ? "&" : "?")+params.join("&");
294
-
295
- const response = await tryWithRetries(
296
- () => httpGet<LNURLPayResult | LNURLError>(payRequest.callback+queryParams, timeout, abortSignal, true),
297
- undefined, RequestError, abortSignal
298
- );
299
-
300
- if(isLNURLError(response)) throw new RequestError("LNURL callback error: "+response.reason, 200);
301
- if(!isLNURLPayResult(response)) throw new RequestError("Invalid LNURL response!", 200);
302
-
303
- const parsedPR = bolt11Decode(response.pr);
304
-
305
- const descHash = Buffer.from(sha256(payRequest.metadata)).toString("hex");
306
- if(parsedPR.tagsObject.purpose_commit_hash!==descHash)
307
- throw new RequestError("Invalid invoice received (description hash)!", 200);
308
-
309
- const msats = parsedPR.millisatoshis;
310
- if(msats==null)
311
- throw new RequestError("Invalid invoice received (amount msats not defined)", 200);
312
- const invoiceMSats = BigInt(msats);
313
- if(invoiceMSats !== (amount * 1000n))
314
- throw new RequestError("Invalid invoice received (amount)!", 200);
315
-
316
- return {
317
- invoice: response.pr,
318
- parsedInvoice: parsedPR,
319
- successAction: response.successAction ?? undefined
320
- }
321
- }
322
-
323
- /**
324
- * Submits the bolt11 lightning invoice to the lnurl withdraw url
325
- *
326
- * @param withdrawRequest Withdraw request to use
327
- * @param withdrawRequest.k1 K1 parameter
328
- * @param withdrawRequest.callback A URL to call
329
- * @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
330
- * @throws {RequestError} If the response is non-200 or status: ERROR
331
- */
332
- static async postInvoiceToLNURLWithdraw(
333
- withdrawRequest: {k1: string, callback: string},
334
- lnpr: string
335
- ): Promise<void> {
336
- const params = [
337
- "pr="+lnpr,
338
- "k1="+withdrawRequest.k1
339
- ];
340
- const queryParams = (withdrawRequest.callback.includes("?") ? "&" : "?")+params.join("&");
341
-
342
- const response = await tryWithRetries(
343
- () => httpGet<LNURLOk | LNURLError>(withdrawRequest.callback+queryParams, undefined, undefined, true),
344
- undefined, RequestError
345
- );
346
-
347
- if(isLNURLError(response)) throw new RequestError("LNURL callback error: " + response.reason, 200);
348
- }
349
-
350
- /**
351
- * Uses a LNURL-withdraw request by submitting a lightning network invoice to it
352
- *
353
- * @param withdrawRequest Withdrawal request as returned from getLNURL call
354
- * @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
355
- * @throws {UserError} In case the provided bolt11 lightning invoice has an amount that is out of bounds for
356
- * the specified LNURL-withdraw request
357
- */
358
- static async useLNURLWithdraw(
359
- withdrawRequest: LNURLWithdrawParamsWithUrl,
360
- lnpr: string
361
- ): Promise<void> {
362
- const min = BigInt(withdrawRequest.minWithdrawable) / 1000n;
363
- const max = BigInt(withdrawRequest.maxWithdrawable) / 1000n;
364
-
365
- const parsedPR = bolt11Decode(lnpr);
366
- const msats = parsedPR.millisatoshis;
367
- if(msats==null) throw new UserError("Invoice without msats value field!");
368
- const amount = (BigInt(msats) + 999n) / 1000n;
369
- if(amount < min) throw new UserError("Invoice amount less than minimum LNURL-withdraw limit");
370
- if(amount > max) throw new UserError("Invoice amount more than maximum LNURL-withdraw limit");
371
-
372
- return await LNURL.postInvoiceToLNURLWithdraw(withdrawRequest, lnpr);
373
- }
374
-
375
- static decodeSuccessAction(successAction?: LNURLPaySuccessAction | null, secret?: string | null): LNURLDecodedSuccessAction | null {
376
- if(secret==null) return null;
377
- if(successAction==null) return null;
378
- if(successAction.tag==="message" && successAction.message!=null) {
379
- return {
380
- description: successAction.message
381
- };
382
- }
383
- if(successAction.tag==="url" && successAction.description!=null && successAction.url!=null) {
384
- return {
385
- description: successAction.description,
386
- url: successAction.url
387
- };
388
- }
389
- if(successAction.tag==="aes" && successAction.iv!=null && successAction.ciphertext!=null && successAction.description!=null) {
390
- const CBC = cbc(Buffer.from(secret, "hex"), Buffer.from(successAction.iv, "hex"));
391
- let plaintext = CBC.decrypt(Buffer.from(successAction.ciphertext, "base64"));
392
- // remove padding
393
- const size = plaintext.length;
394
- const pad = plaintext[size - 1];
395
- return {
396
- description: successAction.description,
397
- text: Buffer.from(plaintext).toString("utf8", 0, size - pad)
398
- };
399
- }
400
- return null;
401
- }
402
-
1
+ import {RequestError} from "../errors/RequestError";
2
+ import {decode as bolt11Decode, PaymentRequestObject, TagsObject} from "@atomiqlabs/bolt11";
3
+ import {UserError} from "../errors/UserError";
4
+ import {bech32} from "@scure/base";
5
+ import {cbc} from "@noble/ciphers/aes";
6
+ import {Buffer} from "buffer";
7
+ import {sha256} from "@noble/hashes/sha2";
8
+ import {httpGet} from "../http/HttpUtils";
9
+ import {
10
+ isLNURLWithdrawParams,
11
+ LNURLWithdraw,
12
+ LNURLWithdrawParams,
13
+ LNURLWithdrawParamsWithUrl
14
+ } from "../types/lnurl/LNURLWithdraw";
15
+ import {
16
+ isLNURLPayParams,
17
+ LNURLDecodedSuccessAction,
18
+ LNURLPay,
19
+ LNURLPayParams,
20
+ LNURLPayParamsWithUrl
21
+ } from "../types/lnurl/LNURLPay";
22
+ import {tryWithRetries} from "../utils/RetryUtils";
23
+
24
+ type LNURLOk = {
25
+ status: "OK"
26
+ };
27
+
28
+ type LNURLError = {
29
+ status: "ERROR",
30
+ reason?: string
31
+ };
32
+
33
+ function isLNURLError(obj: any): obj is LNURLError {
34
+ return obj.status==="ERROR" &&
35
+ (obj.reason==null || typeof obj.reason==="string");
36
+ }
37
+
38
+ type LNURLPayResult = {
39
+ pr: string;
40
+ successAction: LNURLPaySuccessAction | null;
41
+ disposable: boolean | null;
42
+ routes: [];
43
+ }
44
+
45
+ function isLNURLPayResult(obj: any, domain?: string): obj is LNURLPayResult {
46
+ return typeof obj.pr === "string" &&
47
+ (obj.routes == null || Array.isArray(obj.routes)) &&
48
+ (obj.disposable === null || obj.disposable === undefined || typeof obj.disposable === "boolean") &&
49
+ (obj.successAction == null || isLNURLPaySuccessAction(obj.successAction, domain));
50
+ }
51
+
52
+ export type LNURLPaySuccessAction = {
53
+ tag: string;
54
+ description: string | null;
55
+ url: string | null;
56
+ message: string | null;
57
+ ciphertext: string | null;
58
+ iv: string | null;
59
+ };
60
+
61
+ export function isLNURLPaySuccessAction(obj: any, domain?: string): obj is LNURLPaySuccessAction {
62
+ if(obj==null || typeof obj !== 'object' || typeof obj.tag !== 'string') return false;
63
+ switch(obj.tag) {
64
+ case "message":
65
+ return obj.message!=null && obj.message.length<=144;
66
+ case "url":
67
+ return obj.description!=null && obj.description.length<=144 &&
68
+ obj.url!=null &&
69
+ (domain==null || new URL(obj.url).hostname===domain);
70
+ case "aes":
71
+ return obj.description!=null && obj.description.length<=144 &&
72
+ obj.ciphertext!=null && obj.ciphertext.length<=4096 && BASE64_REGEX.test(obj.ciphertext) &&
73
+ obj.iv!=null && obj.iv.length<=24 && BASE64_REGEX.test(obj.iv);
74
+ default:
75
+ //Unsupported action
76
+ return false;
77
+ }
78
+ }
79
+
80
+ const BASE64_REGEX = /^(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{2}==|[A-Za-z0-9+/]{3}=)?$/;
81
+ const MAIL_REGEX = /(?:[A-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[A-z0-9!#$%&'*+/=?^_`{|}~-]+)*|"(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21\x23-\x5b\x5d-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])*")@(?:(?:[A-z0-9](?:[A-z0-9-]*[A-z0-9])?\.)+[A-z0-9](?:[A-z0-9-]*[A-z0-9])?|\[(?:(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9]))\.){3}(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9])|[A-z0-9-]*[A-z0-9]:(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21-\x5a\x53-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])+)\])/;
82
+
83
+ /**
84
+ * LNURL encoding/decoding and callback handling
85
+ * @category Bitcoin
86
+ */
87
+ export class LNURL {
88
+
89
+ private static findBech32LNURL(str: string) {
90
+ const arr = /,*?((lnurl)([0-9]{1,}[a-z0-9]+){1})/.exec(str.toLowerCase());
91
+ if(arr==null) return null;
92
+ return arr[1];
93
+ }
94
+
95
+ private static isBech32LNURL(str: string): boolean {
96
+ return this.findBech32LNURL(str)!=null;
97
+ }
98
+
99
+ /**
100
+ * Checks whether a provided string is bare (non bech32 encoded) lnurl
101
+ * @param str
102
+ * @private
103
+ */
104
+ private static isBareLNURL(str: string): boolean {
105
+ try {
106
+ return str.startsWith("lnurlw://") || str.startsWith("lnurlp://");
107
+ } catch(e) {}
108
+ return false;
109
+ }
110
+
111
+ /**
112
+ * Checks if the provided string is a lightning network address (e.g. satoshi@nakamoto.com)
113
+ * @param str
114
+ * @private
115
+ */
116
+ private static isLightningAddress(str: string): boolean {
117
+ return MAIL_REGEX.test(str);
118
+ }
119
+
120
+ /**
121
+ * Checks whether a given string is a LNURL or lightning address
122
+ * @param str
123
+ */
124
+ static isLNURL(str: string): boolean {
125
+ return LNURL.isBech32LNURL(str) || LNURL.isLightningAddress(str) || LNURL.isBareLNURL(str);
126
+ }
127
+
128
+ /**
129
+ * Extracts the URL that needs to be request from LNURL or lightning address
130
+ * @param str
131
+ * @private
132
+ * @returns An URL to send the request to, or null if it cannot be parsed
133
+ */
134
+ private static extractCallUrl(str: string): string | null {
135
+ if(MAIL_REGEX.test(str)) {
136
+ //lightning e-mail like address
137
+ const arr = str.split("@");
138
+ const username = arr[0];
139
+ const domain = arr[1];
140
+ let scheme = "https";
141
+ if(domain.endsWith(".onion")) {
142
+ scheme = "http";
143
+ }
144
+
145
+ return scheme+"://"+domain+"/.well-known/lnurlp/"+username;
146
+ } else if(LNURL.isBareLNURL(str)) {
147
+ //non-bech32m encoded lnurl
148
+ const data = str.substring("lnurlw://".length);
149
+ const httpUrl = new URL("http://"+data);
150
+
151
+ let scheme = "https";
152
+ if(httpUrl.hostname.endsWith(".onion")) {
153
+ scheme = "http";
154
+ }
155
+
156
+ return scheme+"://"+data;
157
+ } else {
158
+ const lnurl = LNURL.findBech32LNURL(str);
159
+
160
+ if(lnurl!=null) {
161
+ let { prefix: hrp, words: dataPart } = bech32.decode(lnurl as any, 2000);
162
+ let requestByteArray = bech32.fromWords(dataPart);
163
+
164
+ return Buffer.from(requestByteArray).toString();
165
+ }
166
+ }
167
+ return null;
168
+ }
169
+
170
+ /**
171
+ * Sends a request to obtain data about a specific LNURL or lightning address
172
+ *
173
+ * @param str A lnurl or lightning address
174
+ * @param shouldRetry Whether we should retry in case of network failure
175
+ * @param timeout Request timeout in milliseconds
176
+ * @param abortSignal
177
+ */
178
+ static async getLNURL(
179
+ str: string,
180
+ shouldRetry: boolean = true,
181
+ timeout?: number,
182
+ abortSignal?: AbortSignal
183
+ ) : Promise<LNURLPayParamsWithUrl | LNURLWithdrawParamsWithUrl | null> {
184
+ if(shouldRetry==null) shouldRetry = true;
185
+
186
+ const url = LNURL.extractCallUrl(str);
187
+ if(url==null) return null;
188
+
189
+ const sendRequest =
190
+ () => httpGet<LNURLPayParams | LNURLWithdrawParams | LNURLError>(url, timeout, abortSignal, true);
191
+
192
+ let response = shouldRetry ?
193
+ await tryWithRetries(sendRequest, undefined, RequestError, abortSignal) :
194
+ await sendRequest();
195
+
196
+ if(isLNURLError(response)) return null;
197
+
198
+ if(response.tag==="payRequest") try {
199
+ response.decodedMetadata = JSON.parse(response.metadata)
200
+ } catch (err) {
201
+ response.decodedMetadata = []
202
+ }
203
+
204
+ if(!isLNURLPayParams(response) && !isLNURLWithdrawParams(response)) return null;
205
+
206
+ return {
207
+ ...response,
208
+ url: str
209
+ };
210
+ }
211
+
212
+ /**
213
+ * Sends a request to obtain data about a specific LNURL or lightning address
214
+ *
215
+ * @param str A lnurl or lightning address
216
+ * @param shouldRetry Whether we should retry in case of network failure
217
+ * @param timeout Request timeout in milliseconds
218
+ * @param abortSignal
219
+ */
220
+ static async getLNURLType(str: string, shouldRetry?: boolean, timeout?: number, abortSignal?: AbortSignal): Promise<LNURLPay | LNURLWithdraw | null> {
221
+ let res: any = await LNURL.getLNURL(str, shouldRetry, timeout, abortSignal);
222
+
223
+ if(res.tag==="payRequest") {
224
+ const payRequest: LNURLPayParamsWithUrl = res;
225
+ let shortDescription: string | undefined = undefined;
226
+ let longDescription: string | undefined = undefined;
227
+ let icon: string | undefined = undefined;
228
+ payRequest.decodedMetadata.forEach(data => {
229
+ switch(data[0]) {
230
+ case "text/plain":
231
+ shortDescription = data[1];
232
+ break;
233
+ case "text/long-desc":
234
+ longDescription = data[1];
235
+ break;
236
+ case "image/png;base64":
237
+ icon = "data:"+data[0]+","+data[1];
238
+ break;
239
+ case "image/jpeg;base64":
240
+ icon = "data:"+data[0]+","+data[1];
241
+ break;
242
+ }
243
+ });
244
+ return {
245
+ type: "pay",
246
+ min: BigInt(payRequest.minSendable) / 1000n,
247
+ max: BigInt(payRequest.maxSendable) / 1000n,
248
+ commentMaxLength: payRequest.commentAllowed || 0,
249
+ shortDescription,
250
+ longDescription,
251
+ icon,
252
+ params: payRequest
253
+ }
254
+ }
255
+ if(res.tag==="withdrawRequest") {
256
+ const payRequest: LNURLWithdrawParamsWithUrl = res;
257
+ return {
258
+ type: "withdraw",
259
+ min: BigInt(payRequest.minWithdrawable) / 1000n,
260
+ max: BigInt(payRequest.maxWithdrawable) / 1000n,
261
+ params: payRequest
262
+ }
263
+ }
264
+ return null;
265
+ }
266
+
267
+ /**
268
+ * Uses a LNURL-pay request by obtaining a lightning network invoice from it
269
+ *
270
+ * @param payRequest LNURL params as returned from the getLNURL call
271
+ * @param amount Amount of sats (BTC) to pay
272
+ * @param comment Optional comment for the payment request
273
+ * @param timeout Request timeout in milliseconds
274
+ * @param abortSignal
275
+ * @throws {RequestError} If the response is non-200, status: ERROR, or invalid format
276
+ */
277
+ static async useLNURLPay(
278
+ payRequest: LNURLPayParamsWithUrl,
279
+ amount: bigint,
280
+ comment?: string,
281
+ timeout?: number,
282
+ abortSignal?: AbortSignal
283
+ ): Promise<{
284
+ invoice: string,
285
+ parsedInvoice: PaymentRequestObject & { tagsObject: TagsObject; },
286
+ successAction?: LNURLPaySuccessAction
287
+ }> {
288
+ const params = ["amount="+(amount * 1000n).toString(10)];
289
+ if(comment!=null) {
290
+ params.push("comment="+encodeURIComponent(comment));
291
+ }
292
+
293
+ const queryParams = (payRequest.callback.includes("?") ? "&" : "?")+params.join("&");
294
+
295
+ const response = await tryWithRetries(
296
+ () => httpGet<LNURLPayResult | LNURLError>(payRequest.callback+queryParams, timeout, abortSignal, true),
297
+ undefined, RequestError, abortSignal
298
+ );
299
+
300
+ if(isLNURLError(response)) throw new RequestError("LNURL callback error: "+response.reason, 200);
301
+ if(!isLNURLPayResult(response)) throw new RequestError("Invalid LNURL response!", 200);
302
+
303
+ const parsedPR = bolt11Decode(response.pr);
304
+
305
+ const descHash = Buffer.from(sha256(payRequest.metadata)).toString("hex");
306
+ if(parsedPR.tagsObject.purpose_commit_hash!==descHash)
307
+ throw new RequestError("Invalid invoice received (description hash)!", 200);
308
+
309
+ const msats = parsedPR.millisatoshis;
310
+ if(msats==null)
311
+ throw new RequestError("Invalid invoice received (amount msats not defined)", 200);
312
+ const invoiceMSats = BigInt(msats);
313
+ if(invoiceMSats !== (amount * 1000n))
314
+ throw new RequestError("Invalid invoice received (amount)!", 200);
315
+
316
+ return {
317
+ invoice: response.pr,
318
+ parsedInvoice: parsedPR,
319
+ successAction: response.successAction ?? undefined
320
+ }
321
+ }
322
+
323
+ /**
324
+ * Submits the bolt11 lightning invoice to the lnurl withdraw url
325
+ *
326
+ * @param withdrawRequest Withdraw request to use
327
+ * @param withdrawRequest.k1 K1 parameter
328
+ * @param withdrawRequest.callback A URL to call
329
+ * @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
330
+ * @throws {RequestError} If the response is non-200 or status: ERROR
331
+ */
332
+ static async postInvoiceToLNURLWithdraw(
333
+ withdrawRequest: {k1: string, callback: string},
334
+ lnpr: string
335
+ ): Promise<void> {
336
+ const params = [
337
+ "pr="+lnpr,
338
+ "k1="+withdrawRequest.k1
339
+ ];
340
+ const queryParams = (withdrawRequest.callback.includes("?") ? "&" : "?")+params.join("&");
341
+
342
+ const response = await tryWithRetries(
343
+ () => httpGet<LNURLOk | LNURLError>(withdrawRequest.callback+queryParams, undefined, undefined, true),
344
+ undefined, RequestError
345
+ );
346
+
347
+ if(isLNURLError(response)) throw new RequestError("LNURL callback error: " + response.reason, 200);
348
+ }
349
+
350
+ /**
351
+ * Uses a LNURL-withdraw request by submitting a lightning network invoice to it
352
+ *
353
+ * @param withdrawRequest Withdrawal request as returned from getLNURL call
354
+ * @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
355
+ * @throws {UserError} In case the provided bolt11 lightning invoice has an amount that is out of bounds for
356
+ * the specified LNURL-withdraw request
357
+ */
358
+ static async useLNURLWithdraw(
359
+ withdrawRequest: LNURLWithdrawParamsWithUrl,
360
+ lnpr: string
361
+ ): Promise<void> {
362
+ const min = BigInt(withdrawRequest.minWithdrawable) / 1000n;
363
+ const max = BigInt(withdrawRequest.maxWithdrawable) / 1000n;
364
+
365
+ const parsedPR = bolt11Decode(lnpr);
366
+ const msats = parsedPR.millisatoshis;
367
+ if(msats==null) throw new UserError("Invoice without msats value field!");
368
+ const amount = (BigInt(msats) + 999n) / 1000n;
369
+ if(amount < min) throw new UserError("Invoice amount less than minimum LNURL-withdraw limit");
370
+ if(amount > max) throw new UserError("Invoice amount more than maximum LNURL-withdraw limit");
371
+
372
+ return await LNURL.postInvoiceToLNURLWithdraw(withdrawRequest, lnpr);
373
+ }
374
+
375
+ static decodeSuccessAction(successAction?: LNURLPaySuccessAction | null, secret?: string | null): LNURLDecodedSuccessAction | null {
376
+ if(secret==null) return null;
377
+ if(successAction==null) return null;
378
+ if(successAction.tag==="message" && successAction.message!=null) {
379
+ return {
380
+ description: successAction.message
381
+ };
382
+ }
383
+ if(successAction.tag==="url" && successAction.description!=null && successAction.url!=null) {
384
+ return {
385
+ description: successAction.description,
386
+ url: successAction.url
387
+ };
388
+ }
389
+ if(successAction.tag==="aes" && successAction.iv!=null && successAction.ciphertext!=null && successAction.description!=null) {
390
+ const CBC = cbc(Buffer.from(secret, "hex"), Buffer.from(successAction.iv, "hex"));
391
+ let plaintext = CBC.decrypt(Buffer.from(successAction.ciphertext, "base64"));
392
+ // remove padding
393
+ const size = plaintext.length;
394
+ const pad = plaintext[size - 1];
395
+ return {
396
+ description: successAction.description,
397
+ text: Buffer.from(plaintext).toString("utf8", 0, size - pad)
398
+ };
399
+ }
400
+ return null;
401
+ }
402
+
403
403
  }