@atomiqlabs/sdk 8.9.1 → 8.9.3
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.
- package/LICENSE +201 -201
- package/README.md +1760 -1760
- package/api/index.d.ts +1 -1
- package/api/index.js +3 -3
- package/dist/ApiList.d.ts +37 -37
- package/dist/ApiList.js +30 -30
- package/dist/SmartChainAssets.d.ts +181 -181
- package/dist/SmartChainAssets.js +181 -181
- package/dist/api/ApiEndpoints.d.ts +393 -393
- package/dist/api/ApiEndpoints.js +2 -2
- package/dist/api/ApiParser.d.ts +10 -10
- package/dist/api/ApiParser.js +134 -134
- package/dist/api/ApiTypes.d.ts +157 -157
- package/dist/api/ApiTypes.js +75 -75
- package/dist/api/SerializedAction.d.ts +40 -40
- package/dist/api/SerializedAction.js +59 -59
- package/dist/api/SwapperApi.d.ts +50 -50
- package/dist/api/SwapperApi.js +431 -431
- package/dist/api/index.d.ts +5 -5
- package/dist/api/index.js +24 -24
- package/dist/bitcoin/coinselect2/accumulative.d.ts +7 -7
- package/dist/bitcoin/coinselect2/accumulative.js +52 -52
- package/dist/bitcoin/coinselect2/blackjack.d.ts +7 -7
- package/dist/bitcoin/coinselect2/blackjack.js +38 -38
- package/dist/bitcoin/coinselect2/index.d.ts +20 -20
- package/dist/bitcoin/coinselect2/index.js +69 -69
- package/dist/bitcoin/coinselect2/utils.d.ts +82 -82
- package/dist/bitcoin/coinselect2/utils.js +158 -158
- package/dist/bitcoin/wallet/BitcoinWallet.d.ts +113 -113
- package/dist/bitcoin/wallet/BitcoinWallet.js +335 -335
- package/dist/bitcoin/wallet/IBitcoinWallet.d.ts +116 -116
- package/dist/bitcoin/wallet/IBitcoinWallet.js +21 -21
- package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.d.ts +106 -106
- package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.js +196 -196
- package/dist/enums/FeeType.d.ts +15 -15
- package/dist/enums/FeeType.js +19 -19
- package/dist/enums/SwapAmountType.d.ts +15 -15
- package/dist/enums/SwapAmountType.js +19 -19
- package/dist/enums/SwapDirection.d.ts +15 -15
- package/dist/enums/SwapDirection.js +19 -19
- package/dist/enums/SwapSide.d.ts +15 -15
- package/dist/enums/SwapSide.js +19 -19
- package/dist/enums/SwapType.d.ts +75 -75
- package/dist/enums/SwapType.js +79 -79
- package/dist/errors/IntermediaryError.d.ts +13 -13
- package/dist/errors/IntermediaryError.js +27 -27
- package/dist/errors/RequestError.d.ts +32 -32
- package/dist/errors/RequestError.js +54 -54
- package/dist/errors/UserError.d.ts +8 -8
- package/dist/errors/UserError.js +16 -16
- package/dist/events/UnifiedSwapEventListener.d.ts +24 -24
- package/dist/events/UnifiedSwapEventListener.js +138 -138
- package/dist/http/HttpUtils.d.ts +29 -29
- package/dist/http/HttpUtils.js +97 -97
- package/dist/http/paramcoders/IParamReader.d.ts +8 -8
- package/dist/http/paramcoders/IParamReader.js +2 -2
- package/dist/http/paramcoders/ParamDecoder.d.ts +44 -44
- package/dist/http/paramcoders/ParamDecoder.js +137 -137
- package/dist/http/paramcoders/ParamEncoder.d.ts +20 -20
- package/dist/http/paramcoders/ParamEncoder.js +36 -36
- package/dist/http/paramcoders/SchemaVerifier.d.ts +26 -26
- package/dist/http/paramcoders/SchemaVerifier.js +145 -145
- package/dist/http/paramcoders/client/ResponseParamDecoder.d.ts +11 -11
- package/dist/http/paramcoders/client/ResponseParamDecoder.js +57 -57
- package/dist/http/paramcoders/client/StreamParamEncoder.d.ts +13 -13
- package/dist/http/paramcoders/client/StreamParamEncoder.js +26 -26
- package/dist/http/paramcoders/client/StreamingFetchPromise.d.ts +17 -17
- package/dist/http/paramcoders/client/StreamingFetchPromise.js +175 -175
- package/dist/index.d.ts +86 -86
- package/dist/index.js +159 -159
- package/dist/intermediaries/Intermediary.d.ts +178 -178
- package/dist/intermediaries/Intermediary.js +166 -166
- package/dist/intermediaries/IntermediaryDiscovery.d.ts +216 -216
- package/dist/intermediaries/IntermediaryDiscovery.js +424 -424
- package/dist/intermediaries/apis/IntermediaryAPI.d.ts +607 -607
- package/dist/intermediaries/apis/IntermediaryAPI.js +764 -764
- package/dist/intermediaries/apis/TrustedIntermediaryAPI.d.ts +155 -155
- package/dist/intermediaries/apis/TrustedIntermediaryAPI.js +137 -137
- package/dist/intermediaries/auth/SignedKeyBasedAuth.d.ts +14 -14
- package/dist/intermediaries/auth/SignedKeyBasedAuth.js +68 -68
- package/dist/lnurl/LNURL.d.ts +102 -102
- package/dist/lnurl/LNURL.js +321 -321
- package/dist/prices/RedundantSwapPrice.d.ts +110 -110
- package/dist/prices/RedundantSwapPrice.js +222 -222
- package/dist/prices/SingleSwapPrice.d.ts +34 -34
- package/dist/prices/SingleSwapPrice.js +44 -44
- package/dist/prices/SwapPriceWithChain.d.ts +107 -107
- package/dist/prices/SwapPriceWithChain.js +128 -128
- package/dist/prices/abstract/ICachedSwapPrice.d.ts +28 -28
- package/dist/prices/abstract/ICachedSwapPrice.js +62 -62
- package/dist/prices/abstract/IPriceProvider.d.ts +81 -81
- package/dist/prices/abstract/IPriceProvider.js +74 -74
- package/dist/prices/abstract/ISwapPrice.d.ts +168 -168
- package/dist/prices/abstract/ISwapPrice.js +279 -279
- package/dist/prices/providers/BinancePriceProvider.d.ts +23 -23
- package/dist/prices/providers/BinancePriceProvider.js +30 -30
- package/dist/prices/providers/CoinGeckoPriceProvider.d.ts +23 -23
- package/dist/prices/providers/CoinGeckoPriceProvider.js +29 -29
- package/dist/prices/providers/CoinPaprikaPriceProvider.d.ts +25 -25
- package/dist/prices/providers/CoinPaprikaPriceProvider.js +29 -29
- package/dist/prices/providers/CustomPriceProvider.d.ts +24 -24
- package/dist/prices/providers/CustomPriceProvider.js +35 -35
- package/dist/prices/providers/KrakenPriceProvider.d.ts +38 -38
- package/dist/prices/providers/KrakenPriceProvider.js +45 -45
- package/dist/prices/providers/OKXPriceProvider.d.ts +34 -34
- package/dist/prices/providers/OKXPriceProvider.js +29 -29
- package/dist/prices/providers/abstract/ExchangePriceProvider.d.ts +17 -17
- package/dist/prices/providers/abstract/ExchangePriceProvider.js +21 -21
- package/dist/prices/providers/abstract/HttpPriceProvider.d.ts +7 -7
- package/dist/prices/providers/abstract/HttpPriceProvider.js +12 -12
- package/dist/storage/IUnifiedStorage.d.ts +127 -127
- package/dist/storage/IUnifiedStorage.js +2 -2
- package/dist/storage/UnifiedSwapStorage.d.ts +120 -120
- package/dist/storage/UnifiedSwapStorage.js +154 -154
- package/dist/storage-browser/IndexedDBUnifiedStorage.d.ts +63 -63
- package/dist/storage-browser/IndexedDBUnifiedStorage.js +298 -298
- package/dist/storage-browser/LocalStorageManager.d.ts +49 -49
- package/dist/storage-browser/LocalStorageManager.js +93 -93
- package/dist/swapper/Swapper.d.ts +770 -770
- package/dist/swapper/Swapper.js +1758 -1758
- package/dist/swapper/SwapperFactory.d.ts +135 -135
- package/dist/swapper/SwapperFactory.js +162 -162
- package/dist/swapper/SwapperUtils.d.ts +222 -222
- package/dist/swapper/SwapperUtils.js +519 -519
- package/dist/swapper/SwapperWithChain.d.ts +404 -404
- package/dist/swapper/SwapperWithChain.js +469 -469
- package/dist/swapper/SwapperWithSigner.d.ts +322 -322
- package/dist/swapper/SwapperWithSigner.js +318 -318
- package/dist/swaps/IAddressSwap.d.ts +22 -22
- package/dist/swaps/IAddressSwap.js +14 -14
- package/dist/swaps/IBTCWalletSwap.d.ts +73 -73
- package/dist/swaps/IBTCWalletSwap.js +18 -18
- package/dist/swaps/IClaimableSwap.d.ts +49 -49
- package/dist/swaps/IClaimableSwap.js +15 -15
- package/dist/swaps/IClaimableSwapWrapper.d.ts +15 -15
- package/dist/swaps/IClaimableSwapWrapper.js +2 -2
- package/dist/swaps/IRefundableSwap.d.ts +43 -43
- package/dist/swaps/IRefundableSwap.js +14 -14
- package/dist/swaps/ISwap.d.ts +453 -453
- package/dist/swaps/ISwap.js +371 -371
- package/dist/swaps/ISwapWithGasDrop.d.ts +21 -21
- package/dist/swaps/ISwapWithGasDrop.js +12 -12
- package/dist/swaps/ISwapWrapper.d.ts +295 -295
- package/dist/swaps/ISwapWrapper.js +373 -373
- package/dist/swaps/escrow_swaps/IEscrowSelfInitSwap.d.ts +98 -98
- package/dist/swaps/escrow_swaps/IEscrowSelfInitSwap.js +126 -126
- package/dist/swaps/escrow_swaps/IEscrowSwap.d.ts +139 -139
- package/dist/swaps/escrow_swaps/IEscrowSwap.js +172 -172
- package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.d.ts +129 -129
- package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.js +167 -167
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.d.ts +107 -107
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.js +130 -130
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCSelfInitSwap.d.ts +162 -162
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCSelfInitSwap.js +190 -190
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCWrapper.d.ts +64 -64
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCWrapper.js +82 -82
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.d.ts +547 -547
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.js +1419 -1419
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.d.ts +192 -192
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.js +432 -432
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.d.ts +650 -650
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.js +1577 -1577
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.d.ts +237 -237
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.js +525 -525
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.d.ts +491 -491
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.js +1463 -1463
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.d.ts +204 -204
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.js +406 -406
- package/dist/swaps/escrow_swaps/tobtc/IToBTCSwap.d.ts +446 -446
- package/dist/swaps/escrow_swaps/tobtc/IToBTCSwap.js +1097 -1097
- package/dist/swaps/escrow_swaps/tobtc/IToBTCWrapper.d.ts +68 -68
- package/dist/swaps/escrow_swaps/tobtc/IToBTCWrapper.js +117 -117
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.d.ts +127 -127
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.js +256 -256
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.d.ts +252 -252
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.js +535 -535
- package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCSwap.d.ts +73 -73
- package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCSwap.js +155 -155
- package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCWrapper.d.ts +134 -134
- package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCWrapper.js +286 -286
- package/dist/swaps/spv_swaps/SpvFromBTCSwap.d.ts +694 -694
- package/dist/swaps/spv_swaps/SpvFromBTCSwap.js +1687 -1687
- package/dist/swaps/spv_swaps/SpvFromBTCWrapper.d.ts +259 -259
- package/dist/swaps/spv_swaps/SpvFromBTCWrapper.js +947 -947
- package/dist/swaps/trusted/ln/LnForGasSwap.d.ts +302 -302
- package/dist/swaps/trusted/ln/LnForGasSwap.js +625 -625
- package/dist/swaps/trusted/ln/LnForGasWrapper.d.ts +40 -40
- package/dist/swaps/trusted/ln/LnForGasWrapper.js +82 -82
- package/dist/swaps/trusted/onchain/OnchainForGasSwap.d.ts +343 -343
- package/dist/swaps/trusted/onchain/OnchainForGasSwap.js +698 -698
- package/dist/swaps/trusted/onchain/OnchainForGasWrapper.d.ts +71 -71
- package/dist/swaps/trusted/onchain/OnchainForGasWrapper.js +93 -93
- package/dist/types/AmountData.d.ts +10 -10
- package/dist/types/AmountData.js +2 -2
- package/dist/types/CustomPriceFunction.d.ts +11 -11
- package/dist/types/CustomPriceFunction.js +2 -2
- package/dist/types/PriceInfoType.d.ts +28 -28
- package/dist/types/PriceInfoType.js +57 -57
- package/dist/types/SwapExecutionAction.d.ts +195 -195
- package/dist/types/SwapExecutionAction.js +106 -106
- package/dist/types/SwapExecutionStep.d.ts +144 -144
- package/dist/types/SwapExecutionStep.js +87 -87
- package/dist/types/SwapStateInfo.d.ts +5 -5
- package/dist/types/SwapStateInfo.js +2 -2
- package/dist/types/SwapWithSigner.d.ts +17 -17
- package/dist/types/SwapWithSigner.js +43 -43
- package/dist/types/Token.d.ts +99 -99
- package/dist/types/Token.js +76 -76
- package/dist/types/TokenAmount.d.ts +75 -75
- package/dist/types/TokenAmount.js +85 -85
- package/dist/types/fees/Fee.d.ts +50 -50
- package/dist/types/fees/Fee.js +2 -2
- package/dist/types/fees/FeeBreakdown.d.ts +11 -11
- package/dist/types/fees/FeeBreakdown.js +2 -2
- package/dist/types/fees/PercentagePPM.d.ts +17 -17
- package/dist/types/fees/PercentagePPM.js +18 -18
- package/dist/types/lnurl/LNURLPay.d.ts +61 -61
- package/dist/types/lnurl/LNURLPay.js +31 -31
- package/dist/types/lnurl/LNURLWithdraw.d.ts +48 -48
- package/dist/types/lnurl/LNURLWithdraw.js +27 -27
- package/dist/types/wallets/LightningInvoiceCreateService.d.ts +24 -24
- package/dist/types/wallets/LightningInvoiceCreateService.js +15 -15
- package/dist/types/wallets/MinimalBitcoinWalletInterface.d.ts +23 -23
- package/dist/types/wallets/MinimalBitcoinWalletInterface.js +2 -2
- package/dist/types/wallets/MinimalLightningNetworkWalletInterface.d.ts +9 -9
- package/dist/types/wallets/MinimalLightningNetworkWalletInterface.js +2 -2
- package/dist/utils/AutomaticClockDriftCorrection.d.ts +1 -1
- package/dist/utils/AutomaticClockDriftCorrection.js +70 -70
- package/dist/utils/BitcoinUtils.d.ts +18 -18
- package/dist/utils/BitcoinUtils.js +174 -174
- package/dist/utils/BitcoinWalletUtils.d.ts +7 -7
- package/dist/utils/BitcoinWalletUtils.js +14 -14
- package/dist/utils/Logger.d.ts +7 -7
- package/dist/utils/Logger.js +12 -12
- package/dist/utils/RetryUtils.d.ts +22 -22
- package/dist/utils/RetryUtils.js +67 -67
- package/dist/utils/SwapUtils.d.ts +88 -88
- package/dist/utils/SwapUtils.js +72 -72
- package/dist/utils/TimeoutUtils.d.ts +17 -17
- package/dist/utils/TimeoutUtils.js +55 -55
- package/dist/utils/TokenUtils.d.ts +19 -19
- package/dist/utils/TokenUtils.js +37 -37
- package/dist/utils/TypeUtils.d.ts +7 -7
- package/dist/utils/TypeUtils.js +2 -2
- package/dist/utils/Utils.d.ts +69 -69
- package/dist/utils/Utils.js +214 -214
- package/package.json +46 -46
- package/src/SmartChainAssets.ts +186 -186
- package/src/api/ApiEndpoints.ts +427 -427
- package/src/api/ApiParser.ts +138 -138
- package/src/api/ApiTypes.ts +229 -229
- package/src/api/SerializedAction.ts +97 -97
- package/src/api/SwapperApi.ts +545 -545
- package/src/api/index.ts +5 -5
- package/src/bitcoin/coinselect2/accumulative.ts +69 -69
- package/src/bitcoin/coinselect2/blackjack.ts +50 -50
- package/src/bitcoin/coinselect2/index.ts +93 -93
- package/src/bitcoin/coinselect2/utils.ts +236 -236
- package/src/bitcoin/wallet/BitcoinWallet.ts +439 -439
- package/src/bitcoin/wallet/IBitcoinWallet.ts +140 -140
- package/src/bitcoin/wallet/SingleAddressBitcoinWallet.ts +225 -225
- package/src/enums/FeeType.ts +15 -15
- package/src/enums/SwapAmountType.ts +16 -16
- package/src/enums/SwapDirection.ts +15 -15
- package/src/enums/SwapSide.ts +16 -16
- package/src/enums/SwapType.ts +75 -75
- package/src/errors/IntermediaryError.ts +28 -28
- package/src/errors/RequestError.ts +64 -64
- package/src/errors/UserError.ts +15 -15
- package/src/events/UnifiedSwapEventListener.ts +181 -181
- package/src/http/HttpUtils.ts +97 -97
- package/src/http/paramcoders/IParamReader.ts +9 -9
- package/src/http/paramcoders/ParamDecoder.ts +145 -145
- package/src/http/paramcoders/ParamEncoder.ts +40 -40
- package/src/http/paramcoders/SchemaVerifier.ts +153 -153
- package/src/http/paramcoders/client/ResponseParamDecoder.ts +57 -57
- package/src/http/paramcoders/client/StreamParamEncoder.ts +28 -28
- package/src/http/paramcoders/client/StreamingFetchPromise.ts +194 -194
- package/src/index.ts +141 -141
- package/src/intermediaries/Intermediary.ts +280 -280
- package/src/intermediaries/IntermediaryDiscovery.ts +548 -548
- package/src/intermediaries/apis/IntermediaryAPI.ts +1247 -1247
- package/src/intermediaries/auth/SignedKeyBasedAuth.ts +69 -69
- package/src/lnurl/LNURL.ts +402 -402
- package/src/prices/RedundantSwapPrice.ts +264 -264
- package/src/prices/SingleSwapPrice.ts +50 -50
- package/src/prices/SwapPriceWithChain.ts +194 -194
- package/src/prices/abstract/ICachedSwapPrice.ts +85 -85
- package/src/prices/abstract/IPriceProvider.ts +127 -127
- package/src/prices/abstract/ISwapPrice.ts +390 -390
- package/src/prices/providers/BinancePriceProvider.ts +48 -48
- package/src/prices/providers/CoinGeckoPriceProvider.ts +46 -46
- package/src/prices/providers/CoinPaprikaPriceProvider.ts +49 -49
- package/src/prices/providers/CustomPriceProvider.ts +40 -40
- package/src/prices/providers/KrakenPriceProvider.ts +83 -83
- package/src/prices/providers/OKXPriceProvider.ts +59 -59
- package/src/prices/providers/abstract/ExchangePriceProvider.ts +31 -31
- package/src/prices/providers/abstract/HttpPriceProvider.ts +14 -14
- package/src/storage/IUnifiedStorage.ts +136 -136
- package/src/storage/UnifiedSwapStorage.ts +175 -175
- package/src/storage-browser/IndexedDBUnifiedStorage.ts +350 -350
- package/src/storage-browser/LocalStorageManager.ts +106 -106
- package/src/swapper/Swapper.ts +2570 -2570
- package/src/swapper/SwapperFactory.ts +307 -307
- package/src/swapper/SwapperUtils.ts +610 -610
- package/src/swapper/SwapperWithChain.ts +707 -707
- package/src/swapper/SwapperWithSigner.ts +511 -511
- package/src/swaps/IAddressSwap.ts +30 -30
- package/src/swaps/IBTCWalletSwap.ts +92 -92
- package/src/swaps/IClaimableSwap.ts +65 -65
- package/src/swaps/IClaimableSwapWrapper.ts +17 -17
- package/src/swaps/IRefundableSwap.ts +58 -58
- package/src/swaps/ISwap.ts +775 -775
- package/src/swaps/ISwapWithGasDrop.ts +25 -25
- package/src/swaps/ISwapWrapper.ts +564 -564
- package/src/swaps/escrow_swaps/IEscrowSelfInitSwap.ts +217 -217
- package/src/swaps/escrow_swaps/IEscrowSwap.ts +271 -271
- package/src/swaps/escrow_swaps/IEscrowSwapWrapper.ts +284 -284
- package/src/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.ts +172 -172
- package/src/swaps/escrow_swaps/frombtc/IFromBTCSelfInitSwap.ts +300 -300
- package/src/swaps/escrow_swaps/frombtc/IFromBTCWrapper.ts +107 -107
- package/src/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.ts +1670 -1671
- package/src/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.ts +603 -603
- package/src/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.ts +1883 -1883
- package/src/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.ts +752 -752
- package/src/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.ts +1753 -1753
- package/src/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.ts +612 -612
- package/src/swaps/escrow_swaps/tobtc/IToBTCSwap.ts +1327 -1327
- package/src/swaps/escrow_swaps/tobtc/IToBTCWrapper.ts +138 -138
- package/src/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.ts +304 -304
- package/src/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.ts +787 -787
- package/src/swaps/escrow_swaps/tobtc/onchain/ToBTCSwap.ts +206 -206
- package/src/swaps/escrow_swaps/tobtc/onchain/ToBTCWrapper.ts +403 -403
- package/src/swaps/spv_swaps/SpvFromBTCSwap.ts +2148 -2148
- package/src/swaps/spv_swaps/SpvFromBTCWrapper.ts +1238 -1238
- package/src/swaps/trusted/ln/LnForGasSwap.ts +753 -753
- package/src/swaps/trusted/ln/LnForGasWrapper.ts +90 -90
- package/src/swaps/trusted/onchain/OnchainForGasSwap.ts +843 -843
- package/src/swaps/trusted/onchain/OnchainForGasWrapper.ts +133 -133
- package/src/types/AmountData.ts +9 -9
- package/src/types/CustomPriceFunction.ts +11 -11
- package/src/types/PriceInfoType.ts +66 -66
- package/src/types/SwapExecutionAction.ts +323 -323
- package/src/types/SwapExecutionStep.ts +224 -224
- package/src/types/SwapStateInfo.ts +6 -6
- package/src/types/SwapWithSigner.ts +61 -61
- package/src/types/Token.ts +163 -163
- package/src/types/TokenAmount.ts +167 -167
- package/src/types/fees/Fee.ts +56 -56
- package/src/types/fees/FeeBreakdown.ts +11 -11
- package/src/types/fees/PercentagePPM.ts +26 -26
- package/src/types/lnurl/LNURLPay.ts +79 -79
- package/src/types/lnurl/LNURLWithdraw.ts +61 -61
- package/src/types/wallets/LightningInvoiceCreateService.ts +30 -30
- package/src/types/wallets/MinimalBitcoinWalletInterface.ts +21 -21
- package/src/types/wallets/MinimalLightningNetworkWalletInterface.ts +9 -9
- package/src/utils/AutomaticClockDriftCorrection.ts +71 -71
- package/src/utils/BitcoinUtils.ts +164 -164
- package/src/utils/BitcoinWalletUtils.ts +15 -15
- package/src/utils/Logger.ts +14 -14
- package/src/utils/RetryUtils.ts +78 -78
- package/src/utils/SwapUtils.ts +99 -99
- package/src/utils/TimeoutUtils.ts +49 -49
- package/src/utils/TokenUtils.ts +33 -33
- package/src/utils/TypeUtils.ts +8 -8
- package/src/utils/Utils.ts +221 -221
|
@@ -1,625 +1,625 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.LnForGasSwap = exports.isLnForGasSwapInit = exports.LnForGasSwapState = void 0;
|
|
4
|
-
const bolt11_1 = require("@atomiqlabs/bolt11");
|
|
5
|
-
const SwapType_1 = require("../../../enums/SwapType");
|
|
6
|
-
const Utils_1 = require("../../../utils/Utils");
|
|
7
|
-
const ISwap_1 = require("../../ISwap");
|
|
8
|
-
const IntermediaryAPI_1 = require("../../../intermediaries/apis/IntermediaryAPI");
|
|
9
|
-
const FeeType_1 = require("../../../enums/FeeType");
|
|
10
|
-
const PercentagePPM_1 = require("../../../types/fees/PercentagePPM");
|
|
11
|
-
const TokenAmount_1 = require("../../../types/TokenAmount");
|
|
12
|
-
const Token_1 = require("../../../types/Token");
|
|
13
|
-
const Logger_1 = require("../../../utils/Logger");
|
|
14
|
-
const TimeoutUtils_1 = require("../../../utils/TimeoutUtils");
|
|
15
|
-
/**
|
|
16
|
-
* State enum for trusted Lightning gas swaps
|
|
17
|
-
*
|
|
18
|
-
* @category Swaps/Trusted Gas Swaps
|
|
19
|
-
*/
|
|
20
|
-
var LnForGasSwapState;
|
|
21
|
-
(function (LnForGasSwapState) {
|
|
22
|
-
/**
|
|
23
|
-
* The swap quote expired before the user paid the Lightning invoice
|
|
24
|
-
*/
|
|
25
|
-
LnForGasSwapState[LnForGasSwapState["EXPIRED"] = -2] = "EXPIRED";
|
|
26
|
-
/**
|
|
27
|
-
* The swap has failed before the destination payout completed, and the held Lightning invoice was released
|
|
28
|
-
*/
|
|
29
|
-
LnForGasSwapState[LnForGasSwapState["FAILED"] = -1] = "FAILED";
|
|
30
|
-
/**
|
|
31
|
-
* Swap was created, pay the provided Lightning invoice which will remain held until destination payout succeeds
|
|
32
|
-
*/
|
|
33
|
-
LnForGasSwapState[LnForGasSwapState["PR_CREATED"] = 0] = "PR_CREATED";
|
|
34
|
-
/**
|
|
35
|
-
* The Lightning invoice was paid and is currently held until the user receives the destination funds
|
|
36
|
-
*/
|
|
37
|
-
LnForGasSwapState[LnForGasSwapState["PR_PAID"] = 1] = "PR_PAID";
|
|
38
|
-
/**
|
|
39
|
-
* The swap is finished after the destination payout succeeded and the held Lightning invoice was settled
|
|
40
|
-
*/
|
|
41
|
-
LnForGasSwapState[LnForGasSwapState["FINISHED"] = 2] = "FINISHED";
|
|
42
|
-
})(LnForGasSwapState = exports.LnForGasSwapState || (exports.LnForGasSwapState = {}));
|
|
43
|
-
const LnForGasSwapStateDescription = {
|
|
44
|
-
[LnForGasSwapState.EXPIRED]: "The swap quote expired before the user paid the Lightning invoice",
|
|
45
|
-
[LnForGasSwapState.FAILED]: "The swap failed before destination payout completed, and the held Lightning invoice was released back to the user",
|
|
46
|
-
[LnForGasSwapState.PR_CREATED]: "Swap was created, pay the provided Lightning invoice. The invoice will remain held until destination payout succeeds",
|
|
47
|
-
[LnForGasSwapState.PR_PAID]: "The Lightning invoice was paid and is currently held. It will only settle once the user receives the destination funds",
|
|
48
|
-
[LnForGasSwapState.FINISHED]: "The swap is finished after the destination payout succeeded and the held Lightning invoice was settled"
|
|
49
|
-
};
|
|
50
|
-
function isLnForGasSwapInit(obj) {
|
|
51
|
-
return typeof (obj.pr) === "string" &&
|
|
52
|
-
typeof (obj.outputAmount) === "bigint" &&
|
|
53
|
-
typeof (obj.recipient) === "string" &&
|
|
54
|
-
typeof (obj.token) === "string" &&
|
|
55
|
-
(0, ISwap_1.isISwapInit)(obj);
|
|
56
|
-
}
|
|
57
|
-
exports.isLnForGasSwapInit = isLnForGasSwapInit;
|
|
58
|
-
/**
|
|
59
|
-
* Trusted swap for Bitcoin Lightning -> Smart chains, to be used for minor amounts to get gas tokens on
|
|
60
|
-
* the destination chain, which is only needed for Solana, which still uses legacy swaps
|
|
61
|
-
*
|
|
62
|
-
* @category Swaps/Trusted Gas Swaps
|
|
63
|
-
*/
|
|
64
|
-
class LnForGasSwap extends ISwap_1.ISwap {
|
|
65
|
-
constructor(wrapper, initOrObj) {
|
|
66
|
-
if (isLnForGasSwapInit(initOrObj) && initOrObj.url != null)
|
|
67
|
-
initOrObj.url += "/lnforgas";
|
|
68
|
-
super(wrapper, initOrObj);
|
|
69
|
-
this.TYPE = SwapType_1.SwapType.TRUSTED_FROM_BTCLN;
|
|
70
|
-
/**
|
|
71
|
-
* @internal
|
|
72
|
-
*/
|
|
73
|
-
this.swapStateDescription = LnForGasSwapStateDescription;
|
|
74
|
-
/**
|
|
75
|
-
* @internal
|
|
76
|
-
*/
|
|
77
|
-
this.swapStateName = (state) => LnForGasSwapState[state];
|
|
78
|
-
/**
|
|
79
|
-
* @internal
|
|
80
|
-
*/
|
|
81
|
-
this.currentVersion = 2;
|
|
82
|
-
if (isLnForGasSwapInit(initOrObj)) {
|
|
83
|
-
this.pr = initOrObj.pr;
|
|
84
|
-
this.outputAmount = initOrObj.outputAmount;
|
|
85
|
-
this.recipient = initOrObj.recipient;
|
|
86
|
-
this.token = initOrObj.token;
|
|
87
|
-
this._state = LnForGasSwapState.PR_CREATED;
|
|
88
|
-
}
|
|
89
|
-
else {
|
|
90
|
-
this.pr = initOrObj.pr;
|
|
91
|
-
this.outputAmount = (0, Utils_1.toBigInt)(initOrObj.outputAmount);
|
|
92
|
-
this.recipient = initOrObj.recipient;
|
|
93
|
-
this.token = initOrObj.token;
|
|
94
|
-
this.scTxId = initOrObj.scTxId;
|
|
95
|
-
}
|
|
96
|
-
this.tryRecomputeSwapPrice();
|
|
97
|
-
if (this.pr != null) {
|
|
98
|
-
const decoded = (0, bolt11_1.decode)(this.pr);
|
|
99
|
-
if (decoded.timeExpireDate != null)
|
|
100
|
-
this.expiry = decoded.timeExpireDate * 1000;
|
|
101
|
-
}
|
|
102
|
-
this.logger = (0, Logger_1.getLogger)("LnForGas(" + this.getId() + "): ");
|
|
103
|
-
}
|
|
104
|
-
/**
|
|
105
|
-
* @inheritDoc
|
|
106
|
-
* @internal
|
|
107
|
-
*/
|
|
108
|
-
upgradeVersion() {
|
|
109
|
-
if (this.version == 1) {
|
|
110
|
-
if (this._state === 1)
|
|
111
|
-
this._state = LnForGasSwapState.FINISHED;
|
|
112
|
-
this.version = 2;
|
|
113
|
-
}
|
|
114
|
-
if (this.version == null) {
|
|
115
|
-
//Noop
|
|
116
|
-
this.version = 1;
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
/**
|
|
120
|
-
* @inheritDoc
|
|
121
|
-
* @internal
|
|
122
|
-
*/
|
|
123
|
-
tryRecomputeSwapPrice() {
|
|
124
|
-
if (this.swapFeeBtc == null && this.swapFee != null) {
|
|
125
|
-
this.swapFeeBtc = this.swapFee * this.getInput().rawAmount / this.getOutAmountWithoutFee();
|
|
126
|
-
}
|
|
127
|
-
super.tryRecomputeSwapPrice();
|
|
128
|
-
}
|
|
129
|
-
//////////////////////////////
|
|
130
|
-
//// Getters & utils
|
|
131
|
-
/**
|
|
132
|
-
* @inheritDoc
|
|
133
|
-
* @internal
|
|
134
|
-
*/
|
|
135
|
-
_getEscrowHash() {
|
|
136
|
-
return this.getId();
|
|
137
|
-
}
|
|
138
|
-
/**
|
|
139
|
-
* @inheritDoc
|
|
140
|
-
*/
|
|
141
|
-
getOutputAddress() {
|
|
142
|
-
return this.recipient;
|
|
143
|
-
}
|
|
144
|
-
/**
|
|
145
|
-
* @inheritDoc
|
|
146
|
-
*/
|
|
147
|
-
getInputAddress() {
|
|
148
|
-
return this.pr;
|
|
149
|
-
}
|
|
150
|
-
/**
|
|
151
|
-
* @inheritDoc
|
|
152
|
-
*/
|
|
153
|
-
getInputTxId() {
|
|
154
|
-
return this.getId();
|
|
155
|
-
}
|
|
156
|
-
/**
|
|
157
|
-
* @inheritDoc
|
|
158
|
-
*/
|
|
159
|
-
getOutputTxId() {
|
|
160
|
-
return this.scTxId ?? null;
|
|
161
|
-
}
|
|
162
|
-
/**
|
|
163
|
-
* @inheritDoc
|
|
164
|
-
*/
|
|
165
|
-
getId() {
|
|
166
|
-
if (this.pr == null)
|
|
167
|
-
throw new Error("No payment request assigned to this swap!");
|
|
168
|
-
const decodedPR = (0, bolt11_1.decode)(this.pr);
|
|
169
|
-
if (decodedPR.tagsObject.payment_hash == null)
|
|
170
|
-
throw new Error("Lightning invoice has no payment hash!");
|
|
171
|
-
return decodedPR.tagsObject.payment_hash;
|
|
172
|
-
}
|
|
173
|
-
/**
|
|
174
|
-
* Returns the lightning network BOLT11 invoice that needs to be paid as an input to the swap
|
|
175
|
-
*/
|
|
176
|
-
getAddress() {
|
|
177
|
-
return this.pr;
|
|
178
|
-
}
|
|
179
|
-
/**
|
|
180
|
-
* Returns a string that can be displayed as QR code representation of the lightning invoice (with lightning: prefix)
|
|
181
|
-
*/
|
|
182
|
-
getHyperlink() {
|
|
183
|
-
return "lightning:" + this.pr.toUpperCase();
|
|
184
|
-
}
|
|
185
|
-
/**
|
|
186
|
-
* @inheritDoc
|
|
187
|
-
*/
|
|
188
|
-
requiresAction() {
|
|
189
|
-
return false;
|
|
190
|
-
}
|
|
191
|
-
/**
|
|
192
|
-
* @inheritDoc
|
|
193
|
-
*/
|
|
194
|
-
isFinished() {
|
|
195
|
-
return this._state === LnForGasSwapState.FINISHED || this._state === LnForGasSwapState.FAILED || this._state === LnForGasSwapState.EXPIRED;
|
|
196
|
-
}
|
|
197
|
-
/**
|
|
198
|
-
* @inheritDoc
|
|
199
|
-
*/
|
|
200
|
-
isQuoteExpired() {
|
|
201
|
-
return this._state === LnForGasSwapState.EXPIRED;
|
|
202
|
-
}
|
|
203
|
-
/**
|
|
204
|
-
* @inheritDoc
|
|
205
|
-
*/
|
|
206
|
-
isQuoteSoftExpired() {
|
|
207
|
-
return this.expiry < Date.now();
|
|
208
|
-
}
|
|
209
|
-
/**
|
|
210
|
-
* @inheritDoc
|
|
211
|
-
*/
|
|
212
|
-
isFailed() {
|
|
213
|
-
return this._state === LnForGasSwapState.FAILED;
|
|
214
|
-
}
|
|
215
|
-
/**
|
|
216
|
-
* @inheritDoc
|
|
217
|
-
*/
|
|
218
|
-
isSuccessful() {
|
|
219
|
-
return this._state === LnForGasSwapState.FINISHED;
|
|
220
|
-
}
|
|
221
|
-
/**
|
|
222
|
-
* @inheritDoc
|
|
223
|
-
*/
|
|
224
|
-
isInProgress() {
|
|
225
|
-
return (this._state === LnForGasSwapState.PR_CREATED && this.initiated) || this._state === LnForGasSwapState.PR_PAID;
|
|
226
|
-
}
|
|
227
|
-
/**
|
|
228
|
-
* @inheritDoc
|
|
229
|
-
* @internal
|
|
230
|
-
*/
|
|
231
|
-
_verifyQuoteDefinitelyExpired() {
|
|
232
|
-
return Promise.resolve(this.expiry < Date.now());
|
|
233
|
-
}
|
|
234
|
-
/**
|
|
235
|
-
* @inheritDoc
|
|
236
|
-
* @internal
|
|
237
|
-
*/
|
|
238
|
-
_verifyQuoteValid() {
|
|
239
|
-
return Promise.resolve(this.expiry > Date.now());
|
|
240
|
-
}
|
|
241
|
-
//////////////////////////////
|
|
242
|
-
//// Amounts & fees
|
|
243
|
-
/**
|
|
244
|
-
* Returns an output amount in base units without a swap fee included, hence this value
|
|
245
|
-
* is larger than the actual output amount
|
|
246
|
-
*
|
|
247
|
-
* @internal
|
|
248
|
-
*/
|
|
249
|
-
getOutAmountWithoutFee() {
|
|
250
|
-
return this.outputAmount + (this.swapFee ?? 0n);
|
|
251
|
-
}
|
|
252
|
-
/**
|
|
253
|
-
* @inheritDoc
|
|
254
|
-
*/
|
|
255
|
-
getOutputToken() {
|
|
256
|
-
return this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()];
|
|
257
|
-
}
|
|
258
|
-
/**
|
|
259
|
-
* @inheritDoc
|
|
260
|
-
*/
|
|
261
|
-
getOutput() {
|
|
262
|
-
return (0, TokenAmount_1.toTokenAmount)(this.outputAmount, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo);
|
|
263
|
-
}
|
|
264
|
-
/**
|
|
265
|
-
* @inheritDoc
|
|
266
|
-
*/
|
|
267
|
-
getInputToken() {
|
|
268
|
-
return Token_1.BitcoinTokens.BTCLN;
|
|
269
|
-
}
|
|
270
|
-
/**
|
|
271
|
-
* @inheritDoc
|
|
272
|
-
*/
|
|
273
|
-
getInput() {
|
|
274
|
-
const parsed = (0, bolt11_1.decode)(this.pr);
|
|
275
|
-
const msats = parsed.millisatoshis;
|
|
276
|
-
if (msats == null)
|
|
277
|
-
throw new Error("Swap lightning invoice has no msat amount field!");
|
|
278
|
-
const amount = (BigInt(msats) + 999n) / 1000n;
|
|
279
|
-
return (0, TokenAmount_1.toTokenAmount)(amount, Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
|
|
280
|
-
}
|
|
281
|
-
/**
|
|
282
|
-
* @inheritDoc
|
|
283
|
-
*/
|
|
284
|
-
getInputWithoutFee() {
|
|
285
|
-
const parsed = (0, bolt11_1.decode)(this.pr);
|
|
286
|
-
const msats = parsed.millisatoshis;
|
|
287
|
-
if (msats == null)
|
|
288
|
-
throw new Error("Swap lightning invoice has no msat amount field!");
|
|
289
|
-
const amount = (BigInt(msats) + 999n) / 1000n;
|
|
290
|
-
return (0, TokenAmount_1.toTokenAmount)(amount - (this.swapFeeBtc ?? 0n), Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
|
|
291
|
-
}
|
|
292
|
-
/**
|
|
293
|
-
* Returns the swap fee charged by the intermediary (LP) on this swap
|
|
294
|
-
*
|
|
295
|
-
* @internal
|
|
296
|
-
*/
|
|
297
|
-
getSwapFee() {
|
|
298
|
-
if (this.pricingInfo == null)
|
|
299
|
-
throw new Error("No pricing info known, cannot estimate swap fee!");
|
|
300
|
-
const feeWithoutBaseFee = this.swapFeeBtc == null ? 0n : this.swapFeeBtc - this.pricingInfo.satsBaseFee;
|
|
301
|
-
const swapFeePPM = feeWithoutBaseFee * 1000000n / this.getInputWithoutFee().rawAmount;
|
|
302
|
-
const amountInSrcToken = (0, TokenAmount_1.toTokenAmount)(this.swapFeeBtc ?? 0n, Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
|
|
303
|
-
return {
|
|
304
|
-
amountInSrcToken,
|
|
305
|
-
amountInDstToken: (0, TokenAmount_1.toTokenAmount)(this.swapFee ?? 0n, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo),
|
|
306
|
-
currentUsdValue: amountInSrcToken.currentUsdValue,
|
|
307
|
-
usdValue: amountInSrcToken.usdValue,
|
|
308
|
-
pastUsdValue: amountInSrcToken.pastUsdValue,
|
|
309
|
-
composition: {
|
|
310
|
-
base: (0, TokenAmount_1.toTokenAmount)(this.pricingInfo.satsBaseFee, Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo),
|
|
311
|
-
percentage: (0, PercentagePPM_1.ppmToPercentage)(swapFeePPM)
|
|
312
|
-
}
|
|
313
|
-
};
|
|
314
|
-
}
|
|
315
|
-
/**
|
|
316
|
-
* @inheritDoc
|
|
317
|
-
*/
|
|
318
|
-
getFee() {
|
|
319
|
-
return this.getSwapFee();
|
|
320
|
-
}
|
|
321
|
-
/**
|
|
322
|
-
* @inheritDoc
|
|
323
|
-
*/
|
|
324
|
-
getFeeBreakdown() {
|
|
325
|
-
return [{
|
|
326
|
-
type: FeeType_1.FeeType.SWAP,
|
|
327
|
-
fee: this.getSwapFee()
|
|
328
|
-
}];
|
|
329
|
-
}
|
|
330
|
-
//////////////////////////////
|
|
331
|
-
//// Payment
|
|
332
|
-
/**
|
|
333
|
-
* @remarks Not supported
|
|
334
|
-
*/
|
|
335
|
-
async execute() {
|
|
336
|
-
throw new Error("Not supported");
|
|
337
|
-
}
|
|
338
|
-
/**
|
|
339
|
-
* @internal
|
|
340
|
-
*/
|
|
341
|
-
async _getExecutionStatus() {
|
|
342
|
-
const state = this._state;
|
|
343
|
-
let lightningPaymentStatus = "inactive";
|
|
344
|
-
let destinationSettlementStatus = "inactive";
|
|
345
|
-
let buildCurrentAction = async () => undefined;
|
|
346
|
-
switch (state) {
|
|
347
|
-
case LnForGasSwapState.PR_CREATED: {
|
|
348
|
-
const quoteValid = await this._verifyQuoteValid();
|
|
349
|
-
lightningPaymentStatus = quoteValid ? "awaiting" : "soft_expired";
|
|
350
|
-
if (quoteValid) {
|
|
351
|
-
buildCurrentAction = this._buildLightningPaymentAction.bind(this);
|
|
352
|
-
}
|
|
353
|
-
break;
|
|
354
|
-
}
|
|
355
|
-
case LnForGasSwapState.EXPIRED:
|
|
356
|
-
lightningPaymentStatus = "expired";
|
|
357
|
-
break;
|
|
358
|
-
case LnForGasSwapState.PR_PAID:
|
|
359
|
-
lightningPaymentStatus = "received";
|
|
360
|
-
destinationSettlementStatus = "waiting_lp";
|
|
361
|
-
buildCurrentAction = this._buildWaitLpAction.bind(this);
|
|
362
|
-
break;
|
|
363
|
-
case LnForGasSwapState.FAILED:
|
|
364
|
-
lightningPaymentStatus = "expired";
|
|
365
|
-
destinationSettlementStatus = "expired";
|
|
366
|
-
break;
|
|
367
|
-
case LnForGasSwapState.FINISHED:
|
|
368
|
-
lightningPaymentStatus = "confirmed";
|
|
369
|
-
destinationSettlementStatus = "settled";
|
|
370
|
-
break;
|
|
371
|
-
}
|
|
372
|
-
return {
|
|
373
|
-
steps: [
|
|
374
|
-
{
|
|
375
|
-
type: "Payment",
|
|
376
|
-
side: "source",
|
|
377
|
-
chain: "LIGHTNING",
|
|
378
|
-
title: "Lightning payment",
|
|
379
|
-
description: "Pay the Lightning network invoice to initiate the swap",
|
|
380
|
-
status: lightningPaymentStatus
|
|
381
|
-
},
|
|
382
|
-
{
|
|
383
|
-
type: "Settlement",
|
|
384
|
-
side: "destination",
|
|
385
|
-
chain: this.chainIdentifier,
|
|
386
|
-
title: "Destination payout",
|
|
387
|
-
description: "Wait for the intermediary to send the gas tokens on the destination smart chain",
|
|
388
|
-
status: destinationSettlementStatus
|
|
389
|
-
}
|
|
390
|
-
],
|
|
391
|
-
buildCurrentAction,
|
|
392
|
-
state
|
|
393
|
-
};
|
|
394
|
-
}
|
|
395
|
-
/**
|
|
396
|
-
* @internal
|
|
397
|
-
* @inheritDoc
|
|
398
|
-
*/
|
|
399
|
-
_submitExecutionTransactions() {
|
|
400
|
-
throw new Error("Invalid swap state for transaction submission!");
|
|
401
|
-
}
|
|
402
|
-
/**
|
|
403
|
-
* @internal
|
|
404
|
-
*/
|
|
405
|
-
async _buildLightningPaymentAction() {
|
|
406
|
-
return {
|
|
407
|
-
type: "SendToAddress",
|
|
408
|
-
name: "Deposit on Lightning",
|
|
409
|
-
description: "Pay the lightning network invoice to initiate the swap",
|
|
410
|
-
chain: "LIGHTNING",
|
|
411
|
-
txs: [{
|
|
412
|
-
type: "BOLT11_PAYMENT_REQUEST",
|
|
413
|
-
address: this.pr,
|
|
414
|
-
hyperlink: this.getHyperlink(),
|
|
415
|
-
amount: this.getInput()
|
|
416
|
-
}],
|
|
417
|
-
waitForTransactions: async (maxWaitTimeSeconds, pollIntervalSeconds, abortSignal) => {
|
|
418
|
-
const abortController = (0, Utils_1.extendAbortController)(abortSignal, maxWaitTimeSeconds, "Timed out waiting for lightning payment");
|
|
419
|
-
let lightningTxId;
|
|
420
|
-
try {
|
|
421
|
-
const success = await this.waitForPayment(pollIntervalSeconds, abortController.signal, (txId) => {
|
|
422
|
-
lightningTxId = txId;
|
|
423
|
-
abortController.abort();
|
|
424
|
-
});
|
|
425
|
-
if (!success)
|
|
426
|
-
throw new Error("Quote expired while waiting for lightning payment");
|
|
427
|
-
}
|
|
428
|
-
catch (e) {
|
|
429
|
-
if (lightningTxId != null)
|
|
430
|
-
return lightningTxId;
|
|
431
|
-
throw e;
|
|
432
|
-
}
|
|
433
|
-
return this.getInputTxId();
|
|
434
|
-
}
|
|
435
|
-
};
|
|
436
|
-
}
|
|
437
|
-
/**
|
|
438
|
-
* @internal
|
|
439
|
-
*/
|
|
440
|
-
async _buildWaitLpAction() {
|
|
441
|
-
return {
|
|
442
|
-
type: "Wait",
|
|
443
|
-
name: "Awaiting LP payout",
|
|
444
|
-
description: "Wait for the intermediary to send the gas tokens on the destination smart chain",
|
|
445
|
-
pollTimeSeconds: 5,
|
|
446
|
-
expectedTimeSeconds: 10,
|
|
447
|
-
wait: async (maxWaitTimeSeconds, pollIntervalSeconds, abortSignal) => {
|
|
448
|
-
const abortController = (0, Utils_1.extendAbortController)(abortSignal, maxWaitTimeSeconds, "Timed out waiting for LP payout");
|
|
449
|
-
await this.waitForPayment(pollIntervalSeconds, abortController.signal);
|
|
450
|
-
}
|
|
451
|
-
};
|
|
452
|
-
}
|
|
453
|
-
/**
|
|
454
|
-
* @inheritDoc
|
|
455
|
-
*/
|
|
456
|
-
async getExecutionAction() {
|
|
457
|
-
const executionStatus = await this._getExecutionStatus();
|
|
458
|
-
return executionStatus.buildCurrentAction();
|
|
459
|
-
}
|
|
460
|
-
/**
|
|
461
|
-
* @inheritDoc
|
|
462
|
-
*/
|
|
463
|
-
async getExecutionStatus(options) {
|
|
464
|
-
const executionStatus = await this._getExecutionStatus();
|
|
465
|
-
return {
|
|
466
|
-
steps: executionStatus.steps,
|
|
467
|
-
currentAction: options?.skipBuildingAction ? undefined : await executionStatus.buildCurrentAction(),
|
|
468
|
-
stateInfo: this._getStateInfo(executionStatus.state)
|
|
469
|
-
};
|
|
470
|
-
}
|
|
471
|
-
/**
|
|
472
|
-
* @inheritDoc
|
|
473
|
-
*/
|
|
474
|
-
async getExecutionSteps() {
|
|
475
|
-
return (await this._getExecutionStatus()).steps;
|
|
476
|
-
}
|
|
477
|
-
/**
|
|
478
|
-
* Queries the intermediary (LP) node for the state of the swap
|
|
479
|
-
*
|
|
480
|
-
* @param save Whether the save the result or not
|
|
481
|
-
*
|
|
482
|
-
* @returns Whether the swap was successful as `boolean` or `null` if the swap is still pending
|
|
483
|
-
* @internal
|
|
484
|
-
*/
|
|
485
|
-
async checkInvoicePaid(save = true) {
|
|
486
|
-
if (this._state === LnForGasSwapState.FAILED || this._state === LnForGasSwapState.EXPIRED)
|
|
487
|
-
return false;
|
|
488
|
-
if (this._state === LnForGasSwapState.FINISHED)
|
|
489
|
-
return true;
|
|
490
|
-
if (this.url == null)
|
|
491
|
-
return false;
|
|
492
|
-
const decodedPR = (0, bolt11_1.decode)(this.pr);
|
|
493
|
-
const paymentHash = decodedPR.tagsObject.payment_hash;
|
|
494
|
-
if (paymentHash == null)
|
|
495
|
-
throw new Error("Invalid swap invoice, payment hash not found!");
|
|
496
|
-
const response = await this.wrapper._lpApi.getTrustedInvoiceStatus(this.url, paymentHash, this.wrapper._options.getRequestTimeout);
|
|
497
|
-
this.logger.debug("checkInvoicePaid(): LP response: ", response);
|
|
498
|
-
switch (response.code) {
|
|
499
|
-
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.PAID:
|
|
500
|
-
this.scTxId = response.data.txId;
|
|
501
|
-
const txStatus = await this.wrapper._chain.getTxIdStatus(this.scTxId);
|
|
502
|
-
if (txStatus === "success") {
|
|
503
|
-
this._state = LnForGasSwapState.FINISHED;
|
|
504
|
-
if (save)
|
|
505
|
-
await this._saveAndEmit();
|
|
506
|
-
return true;
|
|
507
|
-
}
|
|
508
|
-
return null;
|
|
509
|
-
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.EXPIRED:
|
|
510
|
-
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
511
|
-
this._state = LnForGasSwapState.EXPIRED;
|
|
512
|
-
}
|
|
513
|
-
else {
|
|
514
|
-
this._state = LnForGasSwapState.FAILED;
|
|
515
|
-
}
|
|
516
|
-
if (save)
|
|
517
|
-
await this._saveAndEmit();
|
|
518
|
-
return false;
|
|
519
|
-
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.TX_SENT:
|
|
520
|
-
this.scTxId = response.data.txId;
|
|
521
|
-
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
522
|
-
this._state = LnForGasSwapState.PR_PAID;
|
|
523
|
-
if (save)
|
|
524
|
-
await this._saveAndEmit();
|
|
525
|
-
}
|
|
526
|
-
return null;
|
|
527
|
-
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.PENDING:
|
|
528
|
-
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
529
|
-
this._state = LnForGasSwapState.PR_PAID;
|
|
530
|
-
if (save)
|
|
531
|
-
await this._saveAndEmit();
|
|
532
|
-
}
|
|
533
|
-
return null;
|
|
534
|
-
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.AWAIT_PAYMENT:
|
|
535
|
-
return null;
|
|
536
|
-
default:
|
|
537
|
-
this._state = LnForGasSwapState.FAILED;
|
|
538
|
-
if (save)
|
|
539
|
-
await this._saveAndEmit();
|
|
540
|
-
return false;
|
|
541
|
-
}
|
|
542
|
-
}
|
|
543
|
-
/**
|
|
544
|
-
* A blocking promise resolving when payment was received by the intermediary and client can continue,
|
|
545
|
-
* rejecting in case of failure. The swap must be in {@link LnForGasSwapState.PR_CREATED} or
|
|
546
|
-
* {@link LnForGasSwapState.PR_PAID} state!
|
|
547
|
-
*
|
|
548
|
-
* @param checkIntervalSeconds How often to poll the intermediary for answer (default 5 seconds)
|
|
549
|
-
* @param abortSignal Abort signal
|
|
550
|
-
* @param onPaymentReceived Callback as for when the LP reports having received the ln payment
|
|
551
|
-
* @throws {Error} When in invalid state (not PR_CREATED)
|
|
552
|
-
*/
|
|
553
|
-
async waitForPayment(checkIntervalSeconds, abortSignal, onPaymentReceived) {
|
|
554
|
-
if (this._state !== LnForGasSwapState.PR_CREATED && this._state !== LnForGasSwapState.PR_PAID)
|
|
555
|
-
throw new Error("Must be in PR_CREATED or PR_PAID state!");
|
|
556
|
-
if (!this.initiated) {
|
|
557
|
-
this.initiated = true;
|
|
558
|
-
await this._saveAndEmit();
|
|
559
|
-
}
|
|
560
|
-
while (!abortSignal?.aborted && (this._state === LnForGasSwapState.PR_CREATED || this._state === LnForGasSwapState.PR_PAID)) {
|
|
561
|
-
await this.checkInvoicePaid(true);
|
|
562
|
-
if (this._state === LnForGasSwapState.PR_PAID) {
|
|
563
|
-
if (onPaymentReceived != null) {
|
|
564
|
-
onPaymentReceived(this.getInputTxId());
|
|
565
|
-
onPaymentReceived = undefined; // Set to null so it only triggers once
|
|
566
|
-
}
|
|
567
|
-
}
|
|
568
|
-
if (this._state === LnForGasSwapState.PR_CREATED || this._state === LnForGasSwapState.PR_PAID)
|
|
569
|
-
await (0, TimeoutUtils_1.timeoutPromise)((checkIntervalSeconds ?? 5) * 1000, abortSignal);
|
|
570
|
-
}
|
|
571
|
-
if (abortSignal != null)
|
|
572
|
-
abortSignal.throwIfAborted();
|
|
573
|
-
if (this.isFailed())
|
|
574
|
-
throw new Error("Swap failed");
|
|
575
|
-
return !this.isQuoteExpired();
|
|
576
|
-
}
|
|
577
|
-
//////////////////////////////
|
|
578
|
-
//// Storage
|
|
579
|
-
/**
|
|
580
|
-
* @inheritDoc
|
|
581
|
-
*/
|
|
582
|
-
serialize() {
|
|
583
|
-
return {
|
|
584
|
-
...super.serialize(),
|
|
585
|
-
pr: this.pr,
|
|
586
|
-
outputAmount: this.outputAmount == null ? null : this.outputAmount.toString(10),
|
|
587
|
-
recipient: this.recipient,
|
|
588
|
-
token: this.token,
|
|
589
|
-
scTxId: this.scTxId
|
|
590
|
-
};
|
|
591
|
-
}
|
|
592
|
-
/**
|
|
593
|
-
* @inheritDoc
|
|
594
|
-
* @internal
|
|
595
|
-
*/
|
|
596
|
-
_getInitiator() {
|
|
597
|
-
return this.recipient;
|
|
598
|
-
}
|
|
599
|
-
//////////////////////////////
|
|
600
|
-
//// Swap ticks & sync
|
|
601
|
-
/**
|
|
602
|
-
* @inheritDoc
|
|
603
|
-
* @internal
|
|
604
|
-
*/
|
|
605
|
-
async _sync(save) {
|
|
606
|
-
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
607
|
-
//Check if it's maybe already paid
|
|
608
|
-
const res = await this.checkInvoicePaid(false);
|
|
609
|
-
if (res !== null) {
|
|
610
|
-
if (save)
|
|
611
|
-
await this._saveAndEmit();
|
|
612
|
-
return true;
|
|
613
|
-
}
|
|
614
|
-
}
|
|
615
|
-
return false;
|
|
616
|
-
}
|
|
617
|
-
/**
|
|
618
|
-
* @inheritDoc
|
|
619
|
-
* @internal
|
|
620
|
-
*/
|
|
621
|
-
_tick(save) {
|
|
622
|
-
return Promise.resolve(false);
|
|
623
|
-
}
|
|
624
|
-
}
|
|
625
|
-
exports.LnForGasSwap = LnForGasSwap;
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.LnForGasSwap = exports.isLnForGasSwapInit = exports.LnForGasSwapState = void 0;
|
|
4
|
+
const bolt11_1 = require("@atomiqlabs/bolt11");
|
|
5
|
+
const SwapType_1 = require("../../../enums/SwapType");
|
|
6
|
+
const Utils_1 = require("../../../utils/Utils");
|
|
7
|
+
const ISwap_1 = require("../../ISwap");
|
|
8
|
+
const IntermediaryAPI_1 = require("../../../intermediaries/apis/IntermediaryAPI");
|
|
9
|
+
const FeeType_1 = require("../../../enums/FeeType");
|
|
10
|
+
const PercentagePPM_1 = require("../../../types/fees/PercentagePPM");
|
|
11
|
+
const TokenAmount_1 = require("../../../types/TokenAmount");
|
|
12
|
+
const Token_1 = require("../../../types/Token");
|
|
13
|
+
const Logger_1 = require("../../../utils/Logger");
|
|
14
|
+
const TimeoutUtils_1 = require("../../../utils/TimeoutUtils");
|
|
15
|
+
/**
|
|
16
|
+
* State enum for trusted Lightning gas swaps
|
|
17
|
+
*
|
|
18
|
+
* @category Swaps/Trusted Gas Swaps
|
|
19
|
+
*/
|
|
20
|
+
var LnForGasSwapState;
|
|
21
|
+
(function (LnForGasSwapState) {
|
|
22
|
+
/**
|
|
23
|
+
* The swap quote expired before the user paid the Lightning invoice
|
|
24
|
+
*/
|
|
25
|
+
LnForGasSwapState[LnForGasSwapState["EXPIRED"] = -2] = "EXPIRED";
|
|
26
|
+
/**
|
|
27
|
+
* The swap has failed before the destination payout completed, and the held Lightning invoice was released
|
|
28
|
+
*/
|
|
29
|
+
LnForGasSwapState[LnForGasSwapState["FAILED"] = -1] = "FAILED";
|
|
30
|
+
/**
|
|
31
|
+
* Swap was created, pay the provided Lightning invoice which will remain held until destination payout succeeds
|
|
32
|
+
*/
|
|
33
|
+
LnForGasSwapState[LnForGasSwapState["PR_CREATED"] = 0] = "PR_CREATED";
|
|
34
|
+
/**
|
|
35
|
+
* The Lightning invoice was paid and is currently held until the user receives the destination funds
|
|
36
|
+
*/
|
|
37
|
+
LnForGasSwapState[LnForGasSwapState["PR_PAID"] = 1] = "PR_PAID";
|
|
38
|
+
/**
|
|
39
|
+
* The swap is finished after the destination payout succeeded and the held Lightning invoice was settled
|
|
40
|
+
*/
|
|
41
|
+
LnForGasSwapState[LnForGasSwapState["FINISHED"] = 2] = "FINISHED";
|
|
42
|
+
})(LnForGasSwapState = exports.LnForGasSwapState || (exports.LnForGasSwapState = {}));
|
|
43
|
+
const LnForGasSwapStateDescription = {
|
|
44
|
+
[LnForGasSwapState.EXPIRED]: "The swap quote expired before the user paid the Lightning invoice",
|
|
45
|
+
[LnForGasSwapState.FAILED]: "The swap failed before destination payout completed, and the held Lightning invoice was released back to the user",
|
|
46
|
+
[LnForGasSwapState.PR_CREATED]: "Swap was created, pay the provided Lightning invoice. The invoice will remain held until destination payout succeeds",
|
|
47
|
+
[LnForGasSwapState.PR_PAID]: "The Lightning invoice was paid and is currently held. It will only settle once the user receives the destination funds",
|
|
48
|
+
[LnForGasSwapState.FINISHED]: "The swap is finished after the destination payout succeeded and the held Lightning invoice was settled"
|
|
49
|
+
};
|
|
50
|
+
function isLnForGasSwapInit(obj) {
|
|
51
|
+
return typeof (obj.pr) === "string" &&
|
|
52
|
+
typeof (obj.outputAmount) === "bigint" &&
|
|
53
|
+
typeof (obj.recipient) === "string" &&
|
|
54
|
+
typeof (obj.token) === "string" &&
|
|
55
|
+
(0, ISwap_1.isISwapInit)(obj);
|
|
56
|
+
}
|
|
57
|
+
exports.isLnForGasSwapInit = isLnForGasSwapInit;
|
|
58
|
+
/**
|
|
59
|
+
* Trusted swap for Bitcoin Lightning -> Smart chains, to be used for minor amounts to get gas tokens on
|
|
60
|
+
* the destination chain, which is only needed for Solana, which still uses legacy swaps
|
|
61
|
+
*
|
|
62
|
+
* @category Swaps/Trusted Gas Swaps
|
|
63
|
+
*/
|
|
64
|
+
class LnForGasSwap extends ISwap_1.ISwap {
|
|
65
|
+
constructor(wrapper, initOrObj) {
|
|
66
|
+
if (isLnForGasSwapInit(initOrObj) && initOrObj.url != null)
|
|
67
|
+
initOrObj.url += "/lnforgas";
|
|
68
|
+
super(wrapper, initOrObj);
|
|
69
|
+
this.TYPE = SwapType_1.SwapType.TRUSTED_FROM_BTCLN;
|
|
70
|
+
/**
|
|
71
|
+
* @internal
|
|
72
|
+
*/
|
|
73
|
+
this.swapStateDescription = LnForGasSwapStateDescription;
|
|
74
|
+
/**
|
|
75
|
+
* @internal
|
|
76
|
+
*/
|
|
77
|
+
this.swapStateName = (state) => LnForGasSwapState[state];
|
|
78
|
+
/**
|
|
79
|
+
* @internal
|
|
80
|
+
*/
|
|
81
|
+
this.currentVersion = 2;
|
|
82
|
+
if (isLnForGasSwapInit(initOrObj)) {
|
|
83
|
+
this.pr = initOrObj.pr;
|
|
84
|
+
this.outputAmount = initOrObj.outputAmount;
|
|
85
|
+
this.recipient = initOrObj.recipient;
|
|
86
|
+
this.token = initOrObj.token;
|
|
87
|
+
this._state = LnForGasSwapState.PR_CREATED;
|
|
88
|
+
}
|
|
89
|
+
else {
|
|
90
|
+
this.pr = initOrObj.pr;
|
|
91
|
+
this.outputAmount = (0, Utils_1.toBigInt)(initOrObj.outputAmount);
|
|
92
|
+
this.recipient = initOrObj.recipient;
|
|
93
|
+
this.token = initOrObj.token;
|
|
94
|
+
this.scTxId = initOrObj.scTxId;
|
|
95
|
+
}
|
|
96
|
+
this.tryRecomputeSwapPrice();
|
|
97
|
+
if (this.pr != null) {
|
|
98
|
+
const decoded = (0, bolt11_1.decode)(this.pr);
|
|
99
|
+
if (decoded.timeExpireDate != null)
|
|
100
|
+
this.expiry = decoded.timeExpireDate * 1000;
|
|
101
|
+
}
|
|
102
|
+
this.logger = (0, Logger_1.getLogger)("LnForGas(" + this.getId() + "): ");
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* @inheritDoc
|
|
106
|
+
* @internal
|
|
107
|
+
*/
|
|
108
|
+
upgradeVersion() {
|
|
109
|
+
if (this.version == 1) {
|
|
110
|
+
if (this._state === 1)
|
|
111
|
+
this._state = LnForGasSwapState.FINISHED;
|
|
112
|
+
this.version = 2;
|
|
113
|
+
}
|
|
114
|
+
if (this.version == null) {
|
|
115
|
+
//Noop
|
|
116
|
+
this.version = 1;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
/**
|
|
120
|
+
* @inheritDoc
|
|
121
|
+
* @internal
|
|
122
|
+
*/
|
|
123
|
+
tryRecomputeSwapPrice() {
|
|
124
|
+
if (this.swapFeeBtc == null && this.swapFee != null) {
|
|
125
|
+
this.swapFeeBtc = this.swapFee * this.getInput().rawAmount / this.getOutAmountWithoutFee();
|
|
126
|
+
}
|
|
127
|
+
super.tryRecomputeSwapPrice();
|
|
128
|
+
}
|
|
129
|
+
//////////////////////////////
|
|
130
|
+
//// Getters & utils
|
|
131
|
+
/**
|
|
132
|
+
* @inheritDoc
|
|
133
|
+
* @internal
|
|
134
|
+
*/
|
|
135
|
+
_getEscrowHash() {
|
|
136
|
+
return this.getId();
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* @inheritDoc
|
|
140
|
+
*/
|
|
141
|
+
getOutputAddress() {
|
|
142
|
+
return this.recipient;
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* @inheritDoc
|
|
146
|
+
*/
|
|
147
|
+
getInputAddress() {
|
|
148
|
+
return this.pr;
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* @inheritDoc
|
|
152
|
+
*/
|
|
153
|
+
getInputTxId() {
|
|
154
|
+
return this.getId();
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* @inheritDoc
|
|
158
|
+
*/
|
|
159
|
+
getOutputTxId() {
|
|
160
|
+
return this.scTxId ?? null;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* @inheritDoc
|
|
164
|
+
*/
|
|
165
|
+
getId() {
|
|
166
|
+
if (this.pr == null)
|
|
167
|
+
throw new Error("No payment request assigned to this swap!");
|
|
168
|
+
const decodedPR = (0, bolt11_1.decode)(this.pr);
|
|
169
|
+
if (decodedPR.tagsObject.payment_hash == null)
|
|
170
|
+
throw new Error("Lightning invoice has no payment hash!");
|
|
171
|
+
return decodedPR.tagsObject.payment_hash;
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Returns the lightning network BOLT11 invoice that needs to be paid as an input to the swap
|
|
175
|
+
*/
|
|
176
|
+
getAddress() {
|
|
177
|
+
return this.pr;
|
|
178
|
+
}
|
|
179
|
+
/**
|
|
180
|
+
* Returns a string that can be displayed as QR code representation of the lightning invoice (with lightning: prefix)
|
|
181
|
+
*/
|
|
182
|
+
getHyperlink() {
|
|
183
|
+
return "lightning:" + this.pr.toUpperCase();
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* @inheritDoc
|
|
187
|
+
*/
|
|
188
|
+
requiresAction() {
|
|
189
|
+
return false;
|
|
190
|
+
}
|
|
191
|
+
/**
|
|
192
|
+
* @inheritDoc
|
|
193
|
+
*/
|
|
194
|
+
isFinished() {
|
|
195
|
+
return this._state === LnForGasSwapState.FINISHED || this._state === LnForGasSwapState.FAILED || this._state === LnForGasSwapState.EXPIRED;
|
|
196
|
+
}
|
|
197
|
+
/**
|
|
198
|
+
* @inheritDoc
|
|
199
|
+
*/
|
|
200
|
+
isQuoteExpired() {
|
|
201
|
+
return this._state === LnForGasSwapState.EXPIRED;
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* @inheritDoc
|
|
205
|
+
*/
|
|
206
|
+
isQuoteSoftExpired() {
|
|
207
|
+
return this.expiry < Date.now();
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* @inheritDoc
|
|
211
|
+
*/
|
|
212
|
+
isFailed() {
|
|
213
|
+
return this._state === LnForGasSwapState.FAILED;
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* @inheritDoc
|
|
217
|
+
*/
|
|
218
|
+
isSuccessful() {
|
|
219
|
+
return this._state === LnForGasSwapState.FINISHED;
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* @inheritDoc
|
|
223
|
+
*/
|
|
224
|
+
isInProgress() {
|
|
225
|
+
return (this._state === LnForGasSwapState.PR_CREATED && this.initiated) || this._state === LnForGasSwapState.PR_PAID;
|
|
226
|
+
}
|
|
227
|
+
/**
|
|
228
|
+
* @inheritDoc
|
|
229
|
+
* @internal
|
|
230
|
+
*/
|
|
231
|
+
_verifyQuoteDefinitelyExpired() {
|
|
232
|
+
return Promise.resolve(this.expiry < Date.now());
|
|
233
|
+
}
|
|
234
|
+
/**
|
|
235
|
+
* @inheritDoc
|
|
236
|
+
* @internal
|
|
237
|
+
*/
|
|
238
|
+
_verifyQuoteValid() {
|
|
239
|
+
return Promise.resolve(this.expiry > Date.now());
|
|
240
|
+
}
|
|
241
|
+
//////////////////////////////
|
|
242
|
+
//// Amounts & fees
|
|
243
|
+
/**
|
|
244
|
+
* Returns an output amount in base units without a swap fee included, hence this value
|
|
245
|
+
* is larger than the actual output amount
|
|
246
|
+
*
|
|
247
|
+
* @internal
|
|
248
|
+
*/
|
|
249
|
+
getOutAmountWithoutFee() {
|
|
250
|
+
return this.outputAmount + (this.swapFee ?? 0n);
|
|
251
|
+
}
|
|
252
|
+
/**
|
|
253
|
+
* @inheritDoc
|
|
254
|
+
*/
|
|
255
|
+
getOutputToken() {
|
|
256
|
+
return this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()];
|
|
257
|
+
}
|
|
258
|
+
/**
|
|
259
|
+
* @inheritDoc
|
|
260
|
+
*/
|
|
261
|
+
getOutput() {
|
|
262
|
+
return (0, TokenAmount_1.toTokenAmount)(this.outputAmount, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo);
|
|
263
|
+
}
|
|
264
|
+
/**
|
|
265
|
+
* @inheritDoc
|
|
266
|
+
*/
|
|
267
|
+
getInputToken() {
|
|
268
|
+
return Token_1.BitcoinTokens.BTCLN;
|
|
269
|
+
}
|
|
270
|
+
/**
|
|
271
|
+
* @inheritDoc
|
|
272
|
+
*/
|
|
273
|
+
getInput() {
|
|
274
|
+
const parsed = (0, bolt11_1.decode)(this.pr);
|
|
275
|
+
const msats = parsed.millisatoshis;
|
|
276
|
+
if (msats == null)
|
|
277
|
+
throw new Error("Swap lightning invoice has no msat amount field!");
|
|
278
|
+
const amount = (BigInt(msats) + 999n) / 1000n;
|
|
279
|
+
return (0, TokenAmount_1.toTokenAmount)(amount, Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
|
|
280
|
+
}
|
|
281
|
+
/**
|
|
282
|
+
* @inheritDoc
|
|
283
|
+
*/
|
|
284
|
+
getInputWithoutFee() {
|
|
285
|
+
const parsed = (0, bolt11_1.decode)(this.pr);
|
|
286
|
+
const msats = parsed.millisatoshis;
|
|
287
|
+
if (msats == null)
|
|
288
|
+
throw new Error("Swap lightning invoice has no msat amount field!");
|
|
289
|
+
const amount = (BigInt(msats) + 999n) / 1000n;
|
|
290
|
+
return (0, TokenAmount_1.toTokenAmount)(amount - (this.swapFeeBtc ?? 0n), Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
|
|
291
|
+
}
|
|
292
|
+
/**
|
|
293
|
+
* Returns the swap fee charged by the intermediary (LP) on this swap
|
|
294
|
+
*
|
|
295
|
+
* @internal
|
|
296
|
+
*/
|
|
297
|
+
getSwapFee() {
|
|
298
|
+
if (this.pricingInfo == null)
|
|
299
|
+
throw new Error("No pricing info known, cannot estimate swap fee!");
|
|
300
|
+
const feeWithoutBaseFee = this.swapFeeBtc == null ? 0n : this.swapFeeBtc - this.pricingInfo.satsBaseFee;
|
|
301
|
+
const swapFeePPM = feeWithoutBaseFee * 1000000n / this.getInputWithoutFee().rawAmount;
|
|
302
|
+
const amountInSrcToken = (0, TokenAmount_1.toTokenAmount)(this.swapFeeBtc ?? 0n, Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo);
|
|
303
|
+
return {
|
|
304
|
+
amountInSrcToken,
|
|
305
|
+
amountInDstToken: (0, TokenAmount_1.toTokenAmount)(this.swapFee ?? 0n, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo),
|
|
306
|
+
currentUsdValue: amountInSrcToken.currentUsdValue,
|
|
307
|
+
usdValue: amountInSrcToken.usdValue,
|
|
308
|
+
pastUsdValue: amountInSrcToken.pastUsdValue,
|
|
309
|
+
composition: {
|
|
310
|
+
base: (0, TokenAmount_1.toTokenAmount)(this.pricingInfo.satsBaseFee, Token_1.BitcoinTokens.BTCLN, this.wrapper._prices, this.pricingInfo),
|
|
311
|
+
percentage: (0, PercentagePPM_1.ppmToPercentage)(swapFeePPM)
|
|
312
|
+
}
|
|
313
|
+
};
|
|
314
|
+
}
|
|
315
|
+
/**
|
|
316
|
+
* @inheritDoc
|
|
317
|
+
*/
|
|
318
|
+
getFee() {
|
|
319
|
+
return this.getSwapFee();
|
|
320
|
+
}
|
|
321
|
+
/**
|
|
322
|
+
* @inheritDoc
|
|
323
|
+
*/
|
|
324
|
+
getFeeBreakdown() {
|
|
325
|
+
return [{
|
|
326
|
+
type: FeeType_1.FeeType.SWAP,
|
|
327
|
+
fee: this.getSwapFee()
|
|
328
|
+
}];
|
|
329
|
+
}
|
|
330
|
+
//////////////////////////////
|
|
331
|
+
//// Payment
|
|
332
|
+
/**
|
|
333
|
+
* @remarks Not supported
|
|
334
|
+
*/
|
|
335
|
+
async execute() {
|
|
336
|
+
throw new Error("Not supported");
|
|
337
|
+
}
|
|
338
|
+
/**
|
|
339
|
+
* @internal
|
|
340
|
+
*/
|
|
341
|
+
async _getExecutionStatus() {
|
|
342
|
+
const state = this._state;
|
|
343
|
+
let lightningPaymentStatus = "inactive";
|
|
344
|
+
let destinationSettlementStatus = "inactive";
|
|
345
|
+
let buildCurrentAction = async () => undefined;
|
|
346
|
+
switch (state) {
|
|
347
|
+
case LnForGasSwapState.PR_CREATED: {
|
|
348
|
+
const quoteValid = await this._verifyQuoteValid();
|
|
349
|
+
lightningPaymentStatus = quoteValid ? "awaiting" : "soft_expired";
|
|
350
|
+
if (quoteValid) {
|
|
351
|
+
buildCurrentAction = this._buildLightningPaymentAction.bind(this);
|
|
352
|
+
}
|
|
353
|
+
break;
|
|
354
|
+
}
|
|
355
|
+
case LnForGasSwapState.EXPIRED:
|
|
356
|
+
lightningPaymentStatus = "expired";
|
|
357
|
+
break;
|
|
358
|
+
case LnForGasSwapState.PR_PAID:
|
|
359
|
+
lightningPaymentStatus = "received";
|
|
360
|
+
destinationSettlementStatus = "waiting_lp";
|
|
361
|
+
buildCurrentAction = this._buildWaitLpAction.bind(this);
|
|
362
|
+
break;
|
|
363
|
+
case LnForGasSwapState.FAILED:
|
|
364
|
+
lightningPaymentStatus = "expired";
|
|
365
|
+
destinationSettlementStatus = "expired";
|
|
366
|
+
break;
|
|
367
|
+
case LnForGasSwapState.FINISHED:
|
|
368
|
+
lightningPaymentStatus = "confirmed";
|
|
369
|
+
destinationSettlementStatus = "settled";
|
|
370
|
+
break;
|
|
371
|
+
}
|
|
372
|
+
return {
|
|
373
|
+
steps: [
|
|
374
|
+
{
|
|
375
|
+
type: "Payment",
|
|
376
|
+
side: "source",
|
|
377
|
+
chain: "LIGHTNING",
|
|
378
|
+
title: "Lightning payment",
|
|
379
|
+
description: "Pay the Lightning network invoice to initiate the swap",
|
|
380
|
+
status: lightningPaymentStatus
|
|
381
|
+
},
|
|
382
|
+
{
|
|
383
|
+
type: "Settlement",
|
|
384
|
+
side: "destination",
|
|
385
|
+
chain: this.chainIdentifier,
|
|
386
|
+
title: "Destination payout",
|
|
387
|
+
description: "Wait for the intermediary to send the gas tokens on the destination smart chain",
|
|
388
|
+
status: destinationSettlementStatus
|
|
389
|
+
}
|
|
390
|
+
],
|
|
391
|
+
buildCurrentAction,
|
|
392
|
+
state
|
|
393
|
+
};
|
|
394
|
+
}
|
|
395
|
+
/**
|
|
396
|
+
* @internal
|
|
397
|
+
* @inheritDoc
|
|
398
|
+
*/
|
|
399
|
+
_submitExecutionTransactions() {
|
|
400
|
+
throw new Error("Invalid swap state for transaction submission!");
|
|
401
|
+
}
|
|
402
|
+
/**
|
|
403
|
+
* @internal
|
|
404
|
+
*/
|
|
405
|
+
async _buildLightningPaymentAction() {
|
|
406
|
+
return {
|
|
407
|
+
type: "SendToAddress",
|
|
408
|
+
name: "Deposit on Lightning",
|
|
409
|
+
description: "Pay the lightning network invoice to initiate the swap",
|
|
410
|
+
chain: "LIGHTNING",
|
|
411
|
+
txs: [{
|
|
412
|
+
type: "BOLT11_PAYMENT_REQUEST",
|
|
413
|
+
address: this.pr,
|
|
414
|
+
hyperlink: this.getHyperlink(),
|
|
415
|
+
amount: this.getInput()
|
|
416
|
+
}],
|
|
417
|
+
waitForTransactions: async (maxWaitTimeSeconds, pollIntervalSeconds, abortSignal) => {
|
|
418
|
+
const abortController = (0, Utils_1.extendAbortController)(abortSignal, maxWaitTimeSeconds, "Timed out waiting for lightning payment");
|
|
419
|
+
let lightningTxId;
|
|
420
|
+
try {
|
|
421
|
+
const success = await this.waitForPayment(pollIntervalSeconds, abortController.signal, (txId) => {
|
|
422
|
+
lightningTxId = txId;
|
|
423
|
+
abortController.abort();
|
|
424
|
+
});
|
|
425
|
+
if (!success)
|
|
426
|
+
throw new Error("Quote expired while waiting for lightning payment");
|
|
427
|
+
}
|
|
428
|
+
catch (e) {
|
|
429
|
+
if (lightningTxId != null)
|
|
430
|
+
return lightningTxId;
|
|
431
|
+
throw e;
|
|
432
|
+
}
|
|
433
|
+
return this.getInputTxId();
|
|
434
|
+
}
|
|
435
|
+
};
|
|
436
|
+
}
|
|
437
|
+
/**
|
|
438
|
+
* @internal
|
|
439
|
+
*/
|
|
440
|
+
async _buildWaitLpAction() {
|
|
441
|
+
return {
|
|
442
|
+
type: "Wait",
|
|
443
|
+
name: "Awaiting LP payout",
|
|
444
|
+
description: "Wait for the intermediary to send the gas tokens on the destination smart chain",
|
|
445
|
+
pollTimeSeconds: 5,
|
|
446
|
+
expectedTimeSeconds: 10,
|
|
447
|
+
wait: async (maxWaitTimeSeconds, pollIntervalSeconds, abortSignal) => {
|
|
448
|
+
const abortController = (0, Utils_1.extendAbortController)(abortSignal, maxWaitTimeSeconds, "Timed out waiting for LP payout");
|
|
449
|
+
await this.waitForPayment(pollIntervalSeconds, abortController.signal);
|
|
450
|
+
}
|
|
451
|
+
};
|
|
452
|
+
}
|
|
453
|
+
/**
|
|
454
|
+
* @inheritDoc
|
|
455
|
+
*/
|
|
456
|
+
async getExecutionAction() {
|
|
457
|
+
const executionStatus = await this._getExecutionStatus();
|
|
458
|
+
return executionStatus.buildCurrentAction();
|
|
459
|
+
}
|
|
460
|
+
/**
|
|
461
|
+
* @inheritDoc
|
|
462
|
+
*/
|
|
463
|
+
async getExecutionStatus(options) {
|
|
464
|
+
const executionStatus = await this._getExecutionStatus();
|
|
465
|
+
return {
|
|
466
|
+
steps: executionStatus.steps,
|
|
467
|
+
currentAction: options?.skipBuildingAction ? undefined : await executionStatus.buildCurrentAction(),
|
|
468
|
+
stateInfo: this._getStateInfo(executionStatus.state)
|
|
469
|
+
};
|
|
470
|
+
}
|
|
471
|
+
/**
|
|
472
|
+
* @inheritDoc
|
|
473
|
+
*/
|
|
474
|
+
async getExecutionSteps() {
|
|
475
|
+
return (await this._getExecutionStatus()).steps;
|
|
476
|
+
}
|
|
477
|
+
/**
|
|
478
|
+
* Queries the intermediary (LP) node for the state of the swap
|
|
479
|
+
*
|
|
480
|
+
* @param save Whether the save the result or not
|
|
481
|
+
*
|
|
482
|
+
* @returns Whether the swap was successful as `boolean` or `null` if the swap is still pending
|
|
483
|
+
* @internal
|
|
484
|
+
*/
|
|
485
|
+
async checkInvoicePaid(save = true) {
|
|
486
|
+
if (this._state === LnForGasSwapState.FAILED || this._state === LnForGasSwapState.EXPIRED)
|
|
487
|
+
return false;
|
|
488
|
+
if (this._state === LnForGasSwapState.FINISHED)
|
|
489
|
+
return true;
|
|
490
|
+
if (this.url == null)
|
|
491
|
+
return false;
|
|
492
|
+
const decodedPR = (0, bolt11_1.decode)(this.pr);
|
|
493
|
+
const paymentHash = decodedPR.tagsObject.payment_hash;
|
|
494
|
+
if (paymentHash == null)
|
|
495
|
+
throw new Error("Invalid swap invoice, payment hash not found!");
|
|
496
|
+
const response = await this.wrapper._lpApi.getTrustedInvoiceStatus(this.url, paymentHash, this.wrapper._options.getRequestTimeout);
|
|
497
|
+
this.logger.debug("checkInvoicePaid(): LP response: ", response);
|
|
498
|
+
switch (response.code) {
|
|
499
|
+
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.PAID:
|
|
500
|
+
this.scTxId = response.data.txId;
|
|
501
|
+
const txStatus = await this.wrapper._chain.getTxIdStatus(this.scTxId);
|
|
502
|
+
if (txStatus === "success") {
|
|
503
|
+
this._state = LnForGasSwapState.FINISHED;
|
|
504
|
+
if (save)
|
|
505
|
+
await this._saveAndEmit();
|
|
506
|
+
return true;
|
|
507
|
+
}
|
|
508
|
+
return null;
|
|
509
|
+
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.EXPIRED:
|
|
510
|
+
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
511
|
+
this._state = LnForGasSwapState.EXPIRED;
|
|
512
|
+
}
|
|
513
|
+
else {
|
|
514
|
+
this._state = LnForGasSwapState.FAILED;
|
|
515
|
+
}
|
|
516
|
+
if (save)
|
|
517
|
+
await this._saveAndEmit();
|
|
518
|
+
return false;
|
|
519
|
+
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.TX_SENT:
|
|
520
|
+
this.scTxId = response.data.txId;
|
|
521
|
+
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
522
|
+
this._state = LnForGasSwapState.PR_PAID;
|
|
523
|
+
if (save)
|
|
524
|
+
await this._saveAndEmit();
|
|
525
|
+
}
|
|
526
|
+
return null;
|
|
527
|
+
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.PENDING:
|
|
528
|
+
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
529
|
+
this._state = LnForGasSwapState.PR_PAID;
|
|
530
|
+
if (save)
|
|
531
|
+
await this._saveAndEmit();
|
|
532
|
+
}
|
|
533
|
+
return null;
|
|
534
|
+
case IntermediaryAPI_1.TrustedInvoiceStatusResponseCodes.AWAIT_PAYMENT:
|
|
535
|
+
return null;
|
|
536
|
+
default:
|
|
537
|
+
this._state = LnForGasSwapState.FAILED;
|
|
538
|
+
if (save)
|
|
539
|
+
await this._saveAndEmit();
|
|
540
|
+
return false;
|
|
541
|
+
}
|
|
542
|
+
}
|
|
543
|
+
/**
|
|
544
|
+
* A blocking promise resolving when payment was received by the intermediary and client can continue,
|
|
545
|
+
* rejecting in case of failure. The swap must be in {@link LnForGasSwapState.PR_CREATED} or
|
|
546
|
+
* {@link LnForGasSwapState.PR_PAID} state!
|
|
547
|
+
*
|
|
548
|
+
* @param checkIntervalSeconds How often to poll the intermediary for answer (default 5 seconds)
|
|
549
|
+
* @param abortSignal Abort signal
|
|
550
|
+
* @param onPaymentReceived Callback as for when the LP reports having received the ln payment
|
|
551
|
+
* @throws {Error} When in invalid state (not PR_CREATED)
|
|
552
|
+
*/
|
|
553
|
+
async waitForPayment(checkIntervalSeconds, abortSignal, onPaymentReceived) {
|
|
554
|
+
if (this._state !== LnForGasSwapState.PR_CREATED && this._state !== LnForGasSwapState.PR_PAID)
|
|
555
|
+
throw new Error("Must be in PR_CREATED or PR_PAID state!");
|
|
556
|
+
if (!this.initiated) {
|
|
557
|
+
this.initiated = true;
|
|
558
|
+
await this._saveAndEmit();
|
|
559
|
+
}
|
|
560
|
+
while (!abortSignal?.aborted && (this._state === LnForGasSwapState.PR_CREATED || this._state === LnForGasSwapState.PR_PAID)) {
|
|
561
|
+
await this.checkInvoicePaid(true);
|
|
562
|
+
if (this._state === LnForGasSwapState.PR_PAID) {
|
|
563
|
+
if (onPaymentReceived != null) {
|
|
564
|
+
onPaymentReceived(this.getInputTxId());
|
|
565
|
+
onPaymentReceived = undefined; // Set to null so it only triggers once
|
|
566
|
+
}
|
|
567
|
+
}
|
|
568
|
+
if (this._state === LnForGasSwapState.PR_CREATED || this._state === LnForGasSwapState.PR_PAID)
|
|
569
|
+
await (0, TimeoutUtils_1.timeoutPromise)((checkIntervalSeconds ?? 5) * 1000, abortSignal);
|
|
570
|
+
}
|
|
571
|
+
if (abortSignal != null)
|
|
572
|
+
abortSignal.throwIfAborted();
|
|
573
|
+
if (this.isFailed())
|
|
574
|
+
throw new Error("Swap failed");
|
|
575
|
+
return !this.isQuoteExpired();
|
|
576
|
+
}
|
|
577
|
+
//////////////////////////////
|
|
578
|
+
//// Storage
|
|
579
|
+
/**
|
|
580
|
+
* @inheritDoc
|
|
581
|
+
*/
|
|
582
|
+
serialize() {
|
|
583
|
+
return {
|
|
584
|
+
...super.serialize(),
|
|
585
|
+
pr: this.pr,
|
|
586
|
+
outputAmount: this.outputAmount == null ? null : this.outputAmount.toString(10),
|
|
587
|
+
recipient: this.recipient,
|
|
588
|
+
token: this.token,
|
|
589
|
+
scTxId: this.scTxId
|
|
590
|
+
};
|
|
591
|
+
}
|
|
592
|
+
/**
|
|
593
|
+
* @inheritDoc
|
|
594
|
+
* @internal
|
|
595
|
+
*/
|
|
596
|
+
_getInitiator() {
|
|
597
|
+
return this.recipient;
|
|
598
|
+
}
|
|
599
|
+
//////////////////////////////
|
|
600
|
+
//// Swap ticks & sync
|
|
601
|
+
/**
|
|
602
|
+
* @inheritDoc
|
|
603
|
+
* @internal
|
|
604
|
+
*/
|
|
605
|
+
async _sync(save) {
|
|
606
|
+
if (this._state === LnForGasSwapState.PR_CREATED) {
|
|
607
|
+
//Check if it's maybe already paid
|
|
608
|
+
const res = await this.checkInvoicePaid(false);
|
|
609
|
+
if (res !== null) {
|
|
610
|
+
if (save)
|
|
611
|
+
await this._saveAndEmit();
|
|
612
|
+
return true;
|
|
613
|
+
}
|
|
614
|
+
}
|
|
615
|
+
return false;
|
|
616
|
+
}
|
|
617
|
+
/**
|
|
618
|
+
* @inheritDoc
|
|
619
|
+
* @internal
|
|
620
|
+
*/
|
|
621
|
+
_tick(save) {
|
|
622
|
+
return Promise.resolve(false);
|
|
623
|
+
}
|
|
624
|
+
}
|
|
625
|
+
exports.LnForGasSwap = LnForGasSwap;
|