@atomiqlabs/sdk 8.6.5 → 8.6.6
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/dist/SmartChainAssets.d.ts +181 -181
- package/dist/SmartChainAssets.js +181 -181
- package/dist/bitcoin/coinselect2/accumulative.d.ts +6 -6
- package/dist/bitcoin/coinselect2/accumulative.js +51 -52
- package/dist/bitcoin/coinselect2/blackjack.d.ts +6 -6
- package/dist/bitcoin/coinselect2/blackjack.js +37 -38
- package/dist/bitcoin/coinselect2/index.d.ts +17 -19
- package/dist/bitcoin/coinselect2/index.js +69 -69
- package/dist/bitcoin/coinselect2/utils.d.ts +75 -77
- package/dist/bitcoin/coinselect2/utils.js +123 -123
- package/dist/bitcoin/wallet/BitcoinWallet.d.ts +128 -130
- package/dist/bitcoin/wallet/BitcoinWallet.js +322 -322
- package/dist/bitcoin/wallet/IBitcoinWallet.d.ts +78 -78
- package/dist/bitcoin/wallet/IBitcoinWallet.js +20 -21
- package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.d.ts +99 -101
- package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.js +176 -176
- 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 +23 -23
- package/dist/events/UnifiedSwapEventListener.js +130 -130
- package/dist/http/HttpUtils.d.ts +27 -27
- package/dist/http/HttpUtils.js +90 -91
- package/dist/http/paramcoders/IParamReader.d.ts +8 -8
- package/dist/http/paramcoders/IParamReader.js +2 -2
- package/dist/http/paramcoders/ParamDecoder.d.ts +42 -44
- package/dist/http/paramcoders/ParamDecoder.js +137 -137
- package/dist/http/paramcoders/ParamEncoder.d.ts +18 -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 +11 -13
- package/dist/http/paramcoders/client/StreamParamEncoder.js +26 -26
- package/dist/http/paramcoders/client/StreamingFetchPromise.d.ts +16 -16
- package/dist/http/paramcoders/client/StreamingFetchPromise.js +173 -174
- package/dist/index.d.ts +85 -85
- package/dist/index.js +158 -158
- package/dist/intermediaries/Intermediary.d.ts +157 -157
- package/dist/intermediaries/Intermediary.js +142 -142
- package/dist/intermediaries/IntermediaryDiscovery.d.ts +198 -199
- package/dist/intermediaries/IntermediaryDiscovery.js +406 -406
- package/dist/intermediaries/apis/IntermediaryAPI.d.ts +437 -438
- package/dist/intermediaries/apis/IntermediaryAPI.js +603 -602
- package/dist/intermediaries/apis/TrustedIntermediaryAPI.d.ts +155 -155
- package/dist/intermediaries/apis/TrustedIntermediaryAPI.js +137 -137
- 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 +85 -85
- package/dist/storage/IUnifiedStorage.js +2 -2
- package/dist/storage/UnifiedSwapStorage.d.ts +114 -114
- package/dist/storage/UnifiedSwapStorage.js +116 -116
- 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 +686 -687
- package/dist/swapper/Swapper.js +1603 -1603
- package/dist/swapper/SwapperFactory.d.ts +135 -135
- package/dist/swapper/SwapperFactory.js +162 -162
- package/dist/swapper/SwapperUtils.d.ts +200 -200
- package/dist/swapper/SwapperUtils.js +467 -467
- 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 +13 -14
- package/dist/swaps/IBTCWalletSwap.d.ts +73 -73
- package/dist/swaps/IBTCWalletSwap.js +17 -18
- package/dist/swaps/IClaimableSwap.d.ts +49 -49
- package/dist/swaps/IClaimableSwap.js +14 -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 +13 -14
- package/dist/swaps/ISwap.d.ts +386 -387
- package/dist/swaps/ISwap.js +346 -346
- package/dist/swaps/ISwapWithGasDrop.d.ts +21 -21
- package/dist/swaps/ISwapWithGasDrop.js +11 -12
- package/dist/swaps/ISwapWrapper.d.ts +283 -285
- package/dist/swaps/ISwapWrapper.js +353 -353
- 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 +133 -135
- package/dist/swaps/escrow_swaps/IEscrowSwap.js +169 -169
- package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.d.ts +114 -115
- package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.js +134 -134
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.d.ts +98 -101
- 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 +58 -58
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCWrapper.js +78 -78
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.d.ts +529 -531
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.js +1285 -1285
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.d.ts +181 -184
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.js +418 -418
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.d.ts +581 -583
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.js +1371 -1371
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.d.ts +225 -228
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.js +506 -506
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.d.ts +458 -458
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCSwap.js +1126 -1126
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.d.ts +190 -191
- package/dist/swaps/escrow_swaps/frombtc/onchain/FromBTCWrapper.js +378 -378
- package/dist/swaps/escrow_swaps/tobtc/IToBTCSwap.d.ts +403 -403
- package/dist/swaps/escrow_swaps/tobtc/IToBTCSwap.js +924 -924
- package/dist/swaps/escrow_swaps/tobtc/IToBTCWrapper.d.ts +62 -62
- package/dist/swaps/escrow_swaps/tobtc/IToBTCWrapper.js +112 -112
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.d.ts +125 -127
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.js +256 -256
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.d.ts +241 -242
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.js +520 -520
- 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 +127 -128
- package/dist/swaps/escrow_swaps/tobtc/onchain/ToBTCWrapper.js +278 -278
- package/dist/swaps/spv_swaps/SpvFromBTCSwap.d.ts +630 -630
- package/dist/swaps/spv_swaps/SpvFromBTCSwap.js +1443 -1443
- package/dist/swaps/spv_swaps/SpvFromBTCWrapper.d.ts +213 -207
- package/dist/swaps/spv_swaps/SpvFromBTCWrapper.js +756 -755
- package/dist/swaps/trusted/ln/LnForGasSwap.d.ts +261 -261
- package/dist/swaps/trusted/ln/LnForGasSwap.js +511 -511
- 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 +342 -342
- package/dist/swaps/trusted/onchain/OnchainForGasSwap.js +715 -715
- package/dist/swaps/trusted/onchain/OnchainForGasWrapper.d.ts +68 -69
- package/dist/swaps/trusted/onchain/OnchainForGasWrapper.js +92 -92
- 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 +56 -57
- package/dist/types/SwapExecutionAction.d.ts +88 -88
- package/dist/types/SwapExecutionAction.js +2 -2
- 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 +42 -43
- package/dist/types/Token.d.ts +99 -99
- package/dist/types/Token.js +76 -76
- package/dist/types/TokenAmount.d.ts +69 -69
- package/dist/types/TokenAmount.js +59 -60
- 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 +17 -18
- package/dist/types/lnurl/LNURLPay.d.ts +61 -61
- package/dist/types/lnurl/LNURLPay.js +30 -31
- package/dist/types/lnurl/LNURLWithdraw.d.ts +48 -48
- package/dist/types/lnurl/LNURLWithdraw.js +26 -27
- package/dist/types/wallets/LightningInvoiceCreateService.d.ts +24 -24
- package/dist/types/wallets/LightningInvoiceCreateService.js +14 -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 +69 -70
- package/dist/utils/BitcoinUtils.d.ts +12 -14
- package/dist/utils/BitcoinUtils.js +101 -102
- package/dist/utils/BitcoinWalletUtils.d.ts +7 -7
- package/dist/utils/BitcoinWalletUtils.js +13 -14
- package/dist/utils/Logger.d.ts +7 -7
- package/dist/utils/Logger.js +11 -12
- package/dist/utils/RetryUtils.d.ts +22 -22
- package/dist/utils/RetryUtils.js +66 -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 +54 -55
- package/dist/utils/TokenUtils.d.ts +19 -19
- package/dist/utils/TokenUtils.js +36 -37
- package/dist/utils/TypeUtils.d.ts +7 -7
- package/dist/utils/TypeUtils.js +2 -2
- package/dist/utils/Utils.d.ts +56 -58
- package/dist/utils/Utils.js +193 -194
- package/package.json +1 -1
- package/src/intermediaries/apis/IntermediaryAPI.ts +4 -2
- package/src/swaps/spv_swaps/SpvFromBTCWrapper.ts +8 -0
|
@@ -1,715 +1,715 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.OnchainForGasSwap = exports.
|
|
4
|
-
|
|
5
|
-
const
|
|
6
|
-
const
|
|
7
|
-
const
|
|
8
|
-
const
|
|
9
|
-
const
|
|
10
|
-
const
|
|
11
|
-
const
|
|
12
|
-
const
|
|
13
|
-
const
|
|
14
|
-
const
|
|
15
|
-
const
|
|
16
|
-
const
|
|
17
|
-
const
|
|
18
|
-
const
|
|
19
|
-
const
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
*
|
|
23
|
-
*
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
[OnchainForGasSwapState.
|
|
55
|
-
[OnchainForGasSwapState.
|
|
56
|
-
[OnchainForGasSwapState.
|
|
57
|
-
[OnchainForGasSwapState.
|
|
58
|
-
[OnchainForGasSwapState.
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
typeof (obj.
|
|
64
|
-
typeof (obj.
|
|
65
|
-
typeof (obj.
|
|
66
|
-
typeof (obj.
|
|
67
|
-
typeof (obj.
|
|
68
|
-
|
|
69
|
-
(
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* Trusted swap for Bitcoin -> Smart chains, to be used for minor amounts to get gas tokens on the
|
|
74
|
-
* destination chain, which is only needed for Solana, which still uses legacy swaps
|
|
75
|
-
*
|
|
76
|
-
* @category Swaps/Trusted Gas Swaps
|
|
77
|
-
*/
|
|
78
|
-
class OnchainForGasSwap extends ISwap_1.ISwap {
|
|
79
|
-
constructor(wrapper, initOrObj) {
|
|
80
|
-
if (isOnchainForGasSwapInit(initOrObj) && initOrObj.url != null)
|
|
81
|
-
initOrObj.url += "/frombtc_trusted";
|
|
82
|
-
super(wrapper, initOrObj);
|
|
83
|
-
this.TYPE = SwapType_1.SwapType.TRUSTED_FROM_BTC;
|
|
84
|
-
/**
|
|
85
|
-
* @internal
|
|
86
|
-
*/
|
|
87
|
-
this.swapStateDescription = OnchainForGasSwapStateDescription;
|
|
88
|
-
/**
|
|
89
|
-
* @internal
|
|
90
|
-
*/
|
|
91
|
-
this.swapStateName = (state) => OnchainForGasSwapState[state];
|
|
92
|
-
this.wrapper = wrapper;
|
|
93
|
-
if (isOnchainForGasSwapInit(initOrObj)) {
|
|
94
|
-
this.paymentHash = initOrObj.paymentHash;
|
|
95
|
-
this.sequence = initOrObj.sequence;
|
|
96
|
-
this.address = initOrObj.address;
|
|
97
|
-
this.inputAmount = initOrObj.inputAmount;
|
|
98
|
-
this.outputAmount = initOrObj.outputAmount;
|
|
99
|
-
this.recipient = initOrObj.recipient;
|
|
100
|
-
this.token = initOrObj.token;
|
|
101
|
-
this.refundAddress = initOrObj.refundAddress;
|
|
102
|
-
this._state = OnchainForGasSwapState.PR_CREATED;
|
|
103
|
-
}
|
|
104
|
-
else {
|
|
105
|
-
this.paymentHash = initOrObj.paymentHash;
|
|
106
|
-
this.sequence = (0, Utils_1.toBigInt)(initOrObj.sequence);
|
|
107
|
-
this.address = initOrObj.address;
|
|
108
|
-
this.inputAmount = (0, Utils_1.toBigInt)(initOrObj.inputAmount);
|
|
109
|
-
this.outputAmount = (0, Utils_1.toBigInt)(initOrObj.outputAmount);
|
|
110
|
-
this.recipient = initOrObj.recipient;
|
|
111
|
-
this.token = initOrObj.token;
|
|
112
|
-
this.refundAddress = initOrObj.refundAddress;
|
|
113
|
-
this.scTxId = initOrObj.scTxId;
|
|
114
|
-
this.txId = initOrObj.txId;
|
|
115
|
-
this.refundTxId = initOrObj.refundTxId;
|
|
116
|
-
}
|
|
117
|
-
this.logger = (0, Logger_1.getLogger)("OnchainForGas(" + this.getId() + "): ");
|
|
118
|
-
this.tryRecomputeSwapPrice();
|
|
119
|
-
}
|
|
120
|
-
/**
|
|
121
|
-
* @inheritDoc
|
|
122
|
-
* @internal
|
|
123
|
-
*/
|
|
124
|
-
upgradeVersion() {
|
|
125
|
-
if (this.version == null) {
|
|
126
|
-
//Noop
|
|
127
|
-
this.version = 1;
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
/**
|
|
131
|
-
* @inheritDoc
|
|
132
|
-
* @internal
|
|
133
|
-
*/
|
|
134
|
-
tryRecomputeSwapPrice() {
|
|
135
|
-
if (this.swapFeeBtc == null && this.swapFee != null) {
|
|
136
|
-
this.swapFeeBtc = this.swapFee * this.getInput().rawAmount / this.getOutAmountWithoutFee();
|
|
137
|
-
}
|
|
138
|
-
super.tryRecomputeSwapPrice();
|
|
139
|
-
}
|
|
140
|
-
//////////////////////////////
|
|
141
|
-
//// Getters & utils
|
|
142
|
-
/**
|
|
143
|
-
* @inheritDoc
|
|
144
|
-
* @internal
|
|
145
|
-
*/
|
|
146
|
-
_getEscrowHash() {
|
|
147
|
-
return this.paymentHash;
|
|
148
|
-
}
|
|
149
|
-
/**
|
|
150
|
-
* @inheritDoc
|
|
151
|
-
*/
|
|
152
|
-
getOutputAddress() {
|
|
153
|
-
return this.recipient;
|
|
154
|
-
}
|
|
155
|
-
/**
|
|
156
|
-
* @inheritDoc
|
|
157
|
-
*/
|
|
158
|
-
getInputAddress() {
|
|
159
|
-
//TODO: Fuck this, it's not used anyway
|
|
160
|
-
return null;
|
|
161
|
-
}
|
|
162
|
-
/**
|
|
163
|
-
* @inheritDoc
|
|
164
|
-
*/
|
|
165
|
-
getInputTxId() {
|
|
166
|
-
return this.txId ?? null;
|
|
167
|
-
}
|
|
168
|
-
/**
|
|
169
|
-
* @inheritDoc
|
|
170
|
-
*/
|
|
171
|
-
getOutputTxId() {
|
|
172
|
-
return this.scTxId ?? null;
|
|
173
|
-
}
|
|
174
|
-
/**
|
|
175
|
-
* @inheritDoc
|
|
176
|
-
*/
|
|
177
|
-
getId() {
|
|
178
|
-
return this.paymentHash;
|
|
179
|
-
}
|
|
180
|
-
/**
|
|
181
|
-
* @inheritDoc
|
|
182
|
-
*/
|
|
183
|
-
getAddress() {
|
|
184
|
-
return this.address;
|
|
185
|
-
}
|
|
186
|
-
/**
|
|
187
|
-
* @inheritDoc
|
|
188
|
-
*/
|
|
189
|
-
getHyperlink() {
|
|
190
|
-
return "bitcoin:" + this.address + "?amount=" + encodeURIComponent((Number(this.inputAmount) / 100000000).toString(10));
|
|
191
|
-
}
|
|
192
|
-
/**
|
|
193
|
-
* @inheritDoc
|
|
194
|
-
*/
|
|
195
|
-
requiresAction() {
|
|
196
|
-
return this._state === OnchainForGasSwapState.REFUNDABLE;
|
|
197
|
-
}
|
|
198
|
-
/**
|
|
199
|
-
* @inheritDoc
|
|
200
|
-
*/
|
|
201
|
-
isFinished() {
|
|
202
|
-
return this._state === OnchainForGasSwapState.FINISHED || this._state === OnchainForGasSwapState.FAILED || this._state === OnchainForGasSwapState.EXPIRED || this._state === OnchainForGasSwapState.REFUNDED;
|
|
203
|
-
}
|
|
204
|
-
/**
|
|
205
|
-
* @inheritDoc
|
|
206
|
-
*/
|
|
207
|
-
isQuoteExpired() {
|
|
208
|
-
return this._state === OnchainForGasSwapState.EXPIRED;
|
|
209
|
-
}
|
|
210
|
-
/**
|
|
211
|
-
* @inheritDoc
|
|
212
|
-
*/
|
|
213
|
-
isQuoteSoftExpired() {
|
|
214
|
-
return this.expiry < Date.now();
|
|
215
|
-
}
|
|
216
|
-
/**
|
|
217
|
-
* @inheritDoc
|
|
218
|
-
*/
|
|
219
|
-
isFailed() {
|
|
220
|
-
return this._state === OnchainForGasSwapState.FAILED;
|
|
221
|
-
}
|
|
222
|
-
/**
|
|
223
|
-
* @inheritDoc
|
|
224
|
-
*/
|
|
225
|
-
isSuccessful() {
|
|
226
|
-
return this._state === OnchainForGasSwapState.FINISHED;
|
|
227
|
-
}
|
|
228
|
-
/**
|
|
229
|
-
* @inheritDoc
|
|
230
|
-
*/
|
|
231
|
-
isInProgress() {
|
|
232
|
-
return (this._state === OnchainForGasSwapState.PR_CREATED && this.txId != null) ||
|
|
233
|
-
(this._state === OnchainForGasSwapState.REFUNDABLE && this.refundAddress != null);
|
|
234
|
-
}
|
|
235
|
-
/**
|
|
236
|
-
* @inheritDoc
|
|
237
|
-
* @internal
|
|
238
|
-
*/
|
|
239
|
-
_verifyQuoteDefinitelyExpired() {
|
|
240
|
-
return Promise.resolve(this.expiry < Date.now());
|
|
241
|
-
}
|
|
242
|
-
/**
|
|
243
|
-
* @inheritDoc
|
|
244
|
-
* @internal
|
|
245
|
-
*/
|
|
246
|
-
_verifyQuoteValid() {
|
|
247
|
-
return Promise.resolve(this.expiry > Date.now());
|
|
248
|
-
}
|
|
249
|
-
//////////////////////////////
|
|
250
|
-
//// Amounts & fees
|
|
251
|
-
/**
|
|
252
|
-
* Returns an output amount in base units without a swap fee included, hence this value
|
|
253
|
-
* is larger than the actual output amount
|
|
254
|
-
*
|
|
255
|
-
* @internal
|
|
256
|
-
*/
|
|
257
|
-
getOutAmountWithoutFee() {
|
|
258
|
-
return this.outputAmount + (this.swapFee ?? 0n);
|
|
259
|
-
}
|
|
260
|
-
/**
|
|
261
|
-
* @inheritDoc
|
|
262
|
-
*/
|
|
263
|
-
getOutputToken() {
|
|
264
|
-
return this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()];
|
|
265
|
-
}
|
|
266
|
-
/**
|
|
267
|
-
* @inheritDoc
|
|
268
|
-
*/
|
|
269
|
-
getOutput() {
|
|
270
|
-
return (0, TokenAmount_1.toTokenAmount)(this.outputAmount, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo);
|
|
271
|
-
}
|
|
272
|
-
/**
|
|
273
|
-
* @inheritDoc
|
|
274
|
-
*/
|
|
275
|
-
getInputToken() {
|
|
276
|
-
return Token_1.BitcoinTokens.BTC;
|
|
277
|
-
}
|
|
278
|
-
/**
|
|
279
|
-
* @inheritDoc
|
|
280
|
-
*/
|
|
281
|
-
getInput() {
|
|
282
|
-
return (0, TokenAmount_1.toTokenAmount)(this.inputAmount, Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
283
|
-
}
|
|
284
|
-
/**
|
|
285
|
-
* @inheritDoc
|
|
286
|
-
*/
|
|
287
|
-
getInputWithoutFee() {
|
|
288
|
-
return (0, TokenAmount_1.toTokenAmount)(this.inputAmount - (this.swapFeeBtc ?? 0n), Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
289
|
-
}
|
|
290
|
-
/**
|
|
291
|
-
* Returns the swap fee charged by the intermediary (LP) on this swap
|
|
292
|
-
*
|
|
293
|
-
* @internal
|
|
294
|
-
*/
|
|
295
|
-
getSwapFee() {
|
|
296
|
-
if (this.pricingInfo == null)
|
|
297
|
-
throw new Error("No pricing info known!");
|
|
298
|
-
const feeWithoutBaseFee = this.swapFeeBtc == null ? 0n : this.swapFeeBtc - this.pricingInfo.satsBaseFee;
|
|
299
|
-
const swapFeePPM = feeWithoutBaseFee * 1000000n / this.getInputWithoutFee().rawAmount;
|
|
300
|
-
const amountInSrcToken = (0, TokenAmount_1.toTokenAmount)(this.swapFeeBtc ?? 0n, Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
301
|
-
return {
|
|
302
|
-
amountInSrcToken,
|
|
303
|
-
amountInDstToken: (0, TokenAmount_1.toTokenAmount)(this.swapFee ?? 0n, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo),
|
|
304
|
-
currentUsdValue: amountInSrcToken.currentUsdValue,
|
|
305
|
-
usdValue: amountInSrcToken.usdValue,
|
|
306
|
-
pastUsdValue: amountInSrcToken.pastUsdValue,
|
|
307
|
-
composition: {
|
|
308
|
-
base: (0, TokenAmount_1.toTokenAmount)(this.pricingInfo.satsBaseFee, Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo),
|
|
309
|
-
percentage: (0, PercentagePPM_1.ppmToPercentage)(swapFeePPM)
|
|
310
|
-
}
|
|
311
|
-
};
|
|
312
|
-
}
|
|
313
|
-
/**
|
|
314
|
-
* @inheritDoc
|
|
315
|
-
*/
|
|
316
|
-
getFee() {
|
|
317
|
-
return this.getSwapFee();
|
|
318
|
-
}
|
|
319
|
-
/**
|
|
320
|
-
* @inheritDoc
|
|
321
|
-
*/
|
|
322
|
-
getFeeBreakdown() {
|
|
323
|
-
return [{
|
|
324
|
-
type: FeeType_1.FeeType.SWAP,
|
|
325
|
-
fee: this.getSwapFee()
|
|
326
|
-
}];
|
|
327
|
-
}
|
|
328
|
-
/**
|
|
329
|
-
* @inheritDoc
|
|
330
|
-
*/
|
|
331
|
-
getRequiredConfirmationsCount() {
|
|
332
|
-
return 1;
|
|
333
|
-
}
|
|
334
|
-
/**
|
|
335
|
-
* @inheritDoc
|
|
336
|
-
*/
|
|
337
|
-
async getFundedPsbt(_bitcoinWallet, feeRate, additionalOutputs) {
|
|
338
|
-
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
339
|
-
throw new Error("Swap already paid for!");
|
|
340
|
-
let bitcoinWallet;
|
|
341
|
-
if ((0, IBitcoinWallet_1.isIBitcoinWallet)(_bitcoinWallet)) {
|
|
342
|
-
bitcoinWallet = _bitcoinWallet;
|
|
343
|
-
}
|
|
344
|
-
else {
|
|
345
|
-
bitcoinWallet = new SingleAddressBitcoinWallet_1.SingleAddressBitcoinWallet(this.wrapper._btcRpc, this.wrapper._options.bitcoinNetwork, _bitcoinWallet);
|
|
346
|
-
}
|
|
347
|
-
//TODO: Maybe re-introduce fee rate check here if passed from the user
|
|
348
|
-
if (feeRate == null) {
|
|
349
|
-
feeRate = await bitcoinWallet.getFeeRate();
|
|
350
|
-
}
|
|
351
|
-
const basePsbt = new btc_signer_1.Transaction({
|
|
352
|
-
allowUnknownOutputs: true,
|
|
353
|
-
allowLegacyWitnessUtxo: true
|
|
354
|
-
});
|
|
355
|
-
basePsbt.addOutput({
|
|
356
|
-
amount: this.outputAmount,
|
|
357
|
-
script: (0, BitcoinUtils_1.toOutputScript)(this.wrapper._options.bitcoinNetwork, this.address)
|
|
358
|
-
});
|
|
359
|
-
if (additionalOutputs != null)
|
|
360
|
-
additionalOutputs.forEach(output => {
|
|
361
|
-
basePsbt.addOutput({
|
|
362
|
-
amount: output.amount,
|
|
363
|
-
script: output.outputScript ?? (0, BitcoinUtils_1.toOutputScript)(this.wrapper._options.bitcoinNetwork, output.address)
|
|
364
|
-
});
|
|
365
|
-
});
|
|
366
|
-
const psbt = await bitcoinWallet.fundPsbt(basePsbt, feeRate);
|
|
367
|
-
//Sign every input
|
|
368
|
-
const signInputs = [];
|
|
369
|
-
for (let i = 0; i < psbt.inputsLength; i++) {
|
|
370
|
-
signInputs.push(i);
|
|
371
|
-
}
|
|
372
|
-
const serializedPsbt = buffer_1.Buffer.from(psbt.toPSBT());
|
|
373
|
-
return {
|
|
374
|
-
psbt,
|
|
375
|
-
psbtHex: serializedPsbt.toString("hex"),
|
|
376
|
-
psbtBase64: serializedPsbt.toString("base64"),
|
|
377
|
-
signInputs
|
|
378
|
-
};
|
|
379
|
-
}
|
|
380
|
-
/**
|
|
381
|
-
* @inheritDoc
|
|
382
|
-
*/
|
|
383
|
-
async submitPsbt(_psbt) {
|
|
384
|
-
const psbt = (0, BitcoinUtils_1.parsePsbtTransaction)(_psbt);
|
|
385
|
-
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
386
|
-
throw new Error("Swap already paid for!");
|
|
387
|
-
//Ensure not expired
|
|
388
|
-
if (this.expiry < Date.now()) {
|
|
389
|
-
throw new Error("Swap expired!");
|
|
390
|
-
}
|
|
391
|
-
const output0 = psbt.getOutput(0);
|
|
392
|
-
if (output0.amount !== this.outputAmount)
|
|
393
|
-
throw new Error("PSBT output amount invalid, expected: " + this.outputAmount + " got: " + output0.amount);
|
|
394
|
-
const expectedOutputScript = (0, BitcoinUtils_1.toOutputScript)(this.wrapper._options.bitcoinNetwork, this.address);
|
|
395
|
-
if (output0.script == null || !expectedOutputScript.equals(output0.script))
|
|
396
|
-
throw new Error("PSBT output script invalid!");
|
|
397
|
-
if (!psbt.isFinal)
|
|
398
|
-
psbt.finalize();
|
|
399
|
-
return await this.wrapper._btcRpc.sendRawTransaction(buffer_1.Buffer.from(psbt.toBytes(true, true)).toString("hex"));
|
|
400
|
-
}
|
|
401
|
-
/**
|
|
402
|
-
* @inheritDoc
|
|
403
|
-
*/
|
|
404
|
-
async estimateBitcoinFee(_bitcoinWallet, feeRate) {
|
|
405
|
-
const bitcoinWallet = (0, BitcoinWalletUtils_1.toBitcoinWallet)(_bitcoinWallet, this.wrapper._btcRpc, this.wrapper._options.bitcoinNetwork);
|
|
406
|
-
const txFee = await bitcoinWallet.getTransactionFee(this.address, this.inputAmount, feeRate);
|
|
407
|
-
if (txFee == null)
|
|
408
|
-
return null;
|
|
409
|
-
return (0, TokenAmount_1.toTokenAmount)(BigInt(txFee), Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
410
|
-
}
|
|
411
|
-
/**
|
|
412
|
-
* @inheritDoc
|
|
413
|
-
*/
|
|
414
|
-
async sendBitcoinTransaction(wallet, feeRate) {
|
|
415
|
-
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
416
|
-
throw new Error("Swap already paid for!");
|
|
417
|
-
//Ensure not expired
|
|
418
|
-
if (this.expiry < Date.now()) {
|
|
419
|
-
throw new Error("Swap expired!");
|
|
420
|
-
}
|
|
421
|
-
if ((0, IBitcoinWallet_1.isIBitcoinWallet)(wallet)) {
|
|
422
|
-
return await wallet.sendTransaction(this.address, this.inputAmount, feeRate);
|
|
423
|
-
}
|
|
424
|
-
else {
|
|
425
|
-
const { psbt, psbtHex, psbtBase64, signInputs } = await this.getFundedPsbt(wallet, feeRate);
|
|
426
|
-
const signedPsbt = await wallet.signPsbt({
|
|
427
|
-
psbt, psbtHex, psbtBase64
|
|
428
|
-
}, signInputs);
|
|
429
|
-
return await this.submitPsbt(signedPsbt);
|
|
430
|
-
}
|
|
431
|
-
}
|
|
432
|
-
/**
|
|
433
|
-
* @inheritDoc
|
|
434
|
-
*
|
|
435
|
-
* @param options.bitcoinWallet Optional bitcoin wallet address specification to return a funded PSBT,
|
|
436
|
-
* if not provided an address is returned instead.
|
|
437
|
-
*/
|
|
438
|
-
async txsExecute(options) {
|
|
439
|
-
if (this._state === OnchainForGasSwapState.PR_CREATED) {
|
|
440
|
-
if (!await this._verifyQuoteValid())
|
|
441
|
-
throw new Error("Quote already expired or close to expiry!");
|
|
442
|
-
return [
|
|
443
|
-
{
|
|
444
|
-
name: "Payment",
|
|
445
|
-
description: "Send funds to the bitcoin swap address",
|
|
446
|
-
chain: "BITCOIN",
|
|
447
|
-
txs: [
|
|
448
|
-
options?.bitcoinWallet == null ? {
|
|
449
|
-
address: this.address,
|
|
450
|
-
amount: Number(this.inputAmount),
|
|
451
|
-
hyperlink: this.getHyperlink(),
|
|
452
|
-
type: "ADDRESS"
|
|
453
|
-
} : {
|
|
454
|
-
...await this.getFundedPsbt(options.bitcoinWallet),
|
|
455
|
-
type: "FUNDED_PSBT"
|
|
456
|
-
}
|
|
457
|
-
]
|
|
458
|
-
}
|
|
459
|
-
];
|
|
460
|
-
}
|
|
461
|
-
throw new Error("Invalid swap state to obtain execution txns, required PR_CREATED or CLAIM_COMMITED");
|
|
462
|
-
}
|
|
463
|
-
/**
|
|
464
|
-
* @remark Not supported
|
|
465
|
-
*/
|
|
466
|
-
async execute() {
|
|
467
|
-
throw new Error("Not supported");
|
|
468
|
-
}
|
|
469
|
-
/**
|
|
470
|
-
* @inheritDoc
|
|
471
|
-
*
|
|
472
|
-
* @param options.bitcoinWallet Optional bitcoin wallet address specification to return a funded PSBT,
|
|
473
|
-
* if not provided an address is returned instead.
|
|
474
|
-
*/
|
|
475
|
-
async getCurrentActions(options) {
|
|
476
|
-
try {
|
|
477
|
-
return await this.txsExecute(options);
|
|
478
|
-
}
|
|
479
|
-
catch (e) {
|
|
480
|
-
return [];
|
|
481
|
-
}
|
|
482
|
-
}
|
|
483
|
-
//////////////////////////////
|
|
484
|
-
//// Payment
|
|
485
|
-
/**
|
|
486
|
-
* Queries the intermediary (LP) node for the state of the swap
|
|
487
|
-
*
|
|
488
|
-
* @param save Whether the save the result or not
|
|
489
|
-
*
|
|
490
|
-
* @returns Whether the swap was successful as `boolean` or `null` if the swap is still pending
|
|
491
|
-
* @internal
|
|
492
|
-
*/
|
|
493
|
-
async checkAddress(save = true) {
|
|
494
|
-
if (this._state === OnchainForGasSwapState.FAILED ||
|
|
495
|
-
this._state === OnchainForGasSwapState.EXPIRED ||
|
|
496
|
-
this._state === OnchainForGasSwapState.REFUNDED)
|
|
497
|
-
return false;
|
|
498
|
-
if (this._state === OnchainForGasSwapState.FINISHED)
|
|
499
|
-
return false;
|
|
500
|
-
if (this.url == null)
|
|
501
|
-
return false;
|
|
502
|
-
const response = await TrustedIntermediaryAPI_1.TrustedIntermediaryAPI.getAddressStatus(this.url, this.paymentHash, this.sequence, this.wrapper._options.getRequestTimeout);
|
|
503
|
-
switch (response.code) {
|
|
504
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.AWAIT_PAYMENT:
|
|
505
|
-
if (this.txId != null) {
|
|
506
|
-
this.txId = undefined;
|
|
507
|
-
if (save)
|
|
508
|
-
await this._save();
|
|
509
|
-
return true;
|
|
510
|
-
}
|
|
511
|
-
return false;
|
|
512
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.AWAIT_CONFIRMATION:
|
|
513
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.PENDING:
|
|
514
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.TX_SENT:
|
|
515
|
-
const inputAmount = BigInt(response.data.adjustedAmount);
|
|
516
|
-
const outputAmount = BigInt(response.data.adjustedTotal);
|
|
517
|
-
const adjustedFee = response.data.adjustedFee == null ? null : BigInt(response.data.adjustedFee);
|
|
518
|
-
const adjustedFeeSats = response.data.adjustedFeeSats == null ? null : BigInt(response.data.adjustedFeeSats);
|
|
519
|
-
const txId = response.data.txId;
|
|
520
|
-
if (this.txId != txId ||
|
|
521
|
-
this.inputAmount !== inputAmount ||
|
|
522
|
-
this.outputAmount !== outputAmount) {
|
|
523
|
-
this.txId = txId;
|
|
524
|
-
this.inputAmount = inputAmount;
|
|
525
|
-
this.outputAmount = outputAmount;
|
|
526
|
-
if (adjustedFee != null)
|
|
527
|
-
this.swapFee = adjustedFee;
|
|
528
|
-
if (adjustedFeeSats != null)
|
|
529
|
-
this.swapFeeBtc = adjustedFeeSats;
|
|
530
|
-
if (save)
|
|
531
|
-
await this._save();
|
|
532
|
-
return true;
|
|
533
|
-
}
|
|
534
|
-
return false;
|
|
535
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.PAID:
|
|
536
|
-
const txStatus = await this.wrapper._chain.getTxIdStatus(response.data.txId);
|
|
537
|
-
if (txStatus === "success") {
|
|
538
|
-
this._state = OnchainForGasSwapState.FINISHED;
|
|
539
|
-
this.scTxId = response.data.txId;
|
|
540
|
-
if (save)
|
|
541
|
-
await this._saveAndEmit();
|
|
542
|
-
return true;
|
|
543
|
-
}
|
|
544
|
-
return false;
|
|
545
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.EXPIRED:
|
|
546
|
-
this._state = OnchainForGasSwapState.EXPIRED;
|
|
547
|
-
if (save)
|
|
548
|
-
await this._saveAndEmit();
|
|
549
|
-
return true;
|
|
550
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.REFUNDABLE:
|
|
551
|
-
if (this._state === OnchainForGasSwapState.REFUNDABLE)
|
|
552
|
-
return null;
|
|
553
|
-
this._state = OnchainForGasSwapState.REFUNDABLE;
|
|
554
|
-
if (save)
|
|
555
|
-
await this._saveAndEmit();
|
|
556
|
-
return true;
|
|
557
|
-
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.REFUNDED:
|
|
558
|
-
this._state = OnchainForGasSwapState.REFUNDED;
|
|
559
|
-
this.refundTxId = response.data.txId;
|
|
560
|
-
if (save)
|
|
561
|
-
await this._saveAndEmit();
|
|
562
|
-
return true;
|
|
563
|
-
default:
|
|
564
|
-
this._state = OnchainForGasSwapState.FAILED;
|
|
565
|
-
if (save)
|
|
566
|
-
await this._saveAndEmit();
|
|
567
|
-
return true;
|
|
568
|
-
}
|
|
569
|
-
}
|
|
570
|
-
/**
|
|
571
|
-
* Sets the bitcoin address used for possible refunds in case something goes wrong with the swap
|
|
572
|
-
*
|
|
573
|
-
* @param refundAddress Bitcoin address to receive the refund to
|
|
574
|
-
* @internal
|
|
575
|
-
*/
|
|
576
|
-
async setRefundAddress(refundAddress) {
|
|
577
|
-
if (this.refundAddress != null) {
|
|
578
|
-
if (this.refundAddress !== refundAddress)
|
|
579
|
-
throw new Error("Different refund address already set!");
|
|
580
|
-
return;
|
|
581
|
-
}
|
|
582
|
-
if (this.url == null)
|
|
583
|
-
throw new Error("LP URL not known, cannot set refund address!");
|
|
584
|
-
await TrustedIntermediaryAPI_1.TrustedIntermediaryAPI.setRefundAddress(this.url, this.paymentHash, this.sequence, refundAddress, this.wrapper._options.getRequestTimeout);
|
|
585
|
-
this.refundAddress = refundAddress;
|
|
586
|
-
}
|
|
587
|
-
/**
|
|
588
|
-
* @inheritDoc
|
|
589
|
-
*/
|
|
590
|
-
async waitForBitcoinTransaction(updateCallback, checkIntervalSeconds = 5, abortSignal) {
|
|
591
|
-
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
592
|
-
throw new Error("Must be in PR_CREATED state!");
|
|
593
|
-
if (!this.initiated) {
|
|
594
|
-
this.initiated = true;
|
|
595
|
-
await this._saveAndEmit();
|
|
596
|
-
}
|
|
597
|
-
while (!abortSignal?.aborted &&
|
|
598
|
-
this._state === OnchainForGasSwapState.PR_CREATED) {
|
|
599
|
-
await this.checkAddress(true);
|
|
600
|
-
if (this.txId != null && updateCallback != null) {
|
|
601
|
-
const res = await this.wrapper._btcRpc.getTransaction(this.txId);
|
|
602
|
-
if (res == null) {
|
|
603
|
-
updateCallback();
|
|
604
|
-
}
|
|
605
|
-
else if (res.confirmations != null && res.confirmations > 0) {
|
|
606
|
-
updateCallback(res.txid, res.confirmations, 1, 0);
|
|
607
|
-
}
|
|
608
|
-
else {
|
|
609
|
-
const delay = await this.wrapper._btcRpc.getConfirmationDelay(res, 1);
|
|
610
|
-
updateCallback(res.txid, 0, 1, delay ?? undefined);
|
|
611
|
-
}
|
|
612
|
-
}
|
|
613
|
-
if (this._state === OnchainForGasSwapState.PR_CREATED)
|
|
614
|
-
await (0, TimeoutUtils_1.timeoutPromise)(checkIntervalSeconds * 1000, abortSignal);
|
|
615
|
-
}
|
|
616
|
-
if (this._state === OnchainForGasSwapState.REFUNDABLE ||
|
|
617
|
-
this._state === OnchainForGasSwapState.REFUNDED)
|
|
618
|
-
return this.txId;
|
|
619
|
-
if (this.isQuoteExpired())
|
|
620
|
-
throw new Error("Swap expired");
|
|
621
|
-
if (this.isFailed())
|
|
622
|
-
throw new Error("Swap failed");
|
|
623
|
-
return this.txId;
|
|
624
|
-
}
|
|
625
|
-
/**
|
|
626
|
-
* Waits till the LP processes a refund for a failed swap. The swap must be in
|
|
627
|
-
* {@link OnchainForGasSwapState.REFUNDABLE} state
|
|
628
|
-
*
|
|
629
|
-
* @param checkIntervalSeconds How often to check (default 5 seconds)
|
|
630
|
-
* @param abortSignal Abort signal
|
|
631
|
-
*/
|
|
632
|
-
async waitTillRefunded(checkIntervalSeconds, abortSignal) {
|
|
633
|
-
checkIntervalSeconds ??= 5;
|
|
634
|
-
if (this._state === OnchainForGasSwapState.REFUNDED)
|
|
635
|
-
return;
|
|
636
|
-
if (this._state !== OnchainForGasSwapState.REFUNDABLE)
|
|
637
|
-
throw new Error("Must be in REFUNDABLE state!");
|
|
638
|
-
while (!abortSignal?.aborted &&
|
|
639
|
-
this._state === OnchainForGasSwapState.REFUNDABLE) {
|
|
640
|
-
await this.checkAddress(true);
|
|
641
|
-
if (this._state === OnchainForGasSwapState.REFUNDABLE)
|
|
642
|
-
await (0, TimeoutUtils_1.timeoutPromise)(checkIntervalSeconds * 1000, abortSignal);
|
|
643
|
-
}
|
|
644
|
-
if (this.isQuoteExpired())
|
|
645
|
-
throw new Error("Swap expired");
|
|
646
|
-
if (this.isFailed())
|
|
647
|
-
throw new Error("Swap failed");
|
|
648
|
-
}
|
|
649
|
-
/**
|
|
650
|
-
* Requests a refund after the swap failed, this also waits till the refund is actually sent by the
|
|
651
|
-
* intermediary (LP). The swap must be in {@link OnchainForGasSwapState.REFUNDABLE} state
|
|
652
|
-
*
|
|
653
|
-
* @param refundAddress Bitcoin address to receive the refund to
|
|
654
|
-
* @param abortSignal Abort signal
|
|
655
|
-
*/
|
|
656
|
-
async requestRefund(refundAddress, abortSignal) {
|
|
657
|
-
if (refundAddress != null)
|
|
658
|
-
await this.setRefundAddress(refundAddress);
|
|
659
|
-
await this.waitTillRefunded(undefined, abortSignal);
|
|
660
|
-
}
|
|
661
|
-
//////////////////////////////
|
|
662
|
-
//// Storage
|
|
663
|
-
/**
|
|
664
|
-
* @inheritDoc
|
|
665
|
-
*/
|
|
666
|
-
serialize() {
|
|
667
|
-
return {
|
|
668
|
-
...super.serialize(),
|
|
669
|
-
paymentHash: this.paymentHash,
|
|
670
|
-
sequence: this.sequence == null ? null : this.sequence.toString(10),
|
|
671
|
-
address: this.address,
|
|
672
|
-
inputAmount: this.inputAmount == null ? null : this.inputAmount.toString(10),
|
|
673
|
-
outputAmount: this.outputAmount == null ? null : this.outputAmount.toString(10),
|
|
674
|
-
recipient: this.recipient,
|
|
675
|
-
token: this.token,
|
|
676
|
-
refundAddress: this.refundAddress,
|
|
677
|
-
scTxId: this.scTxId,
|
|
678
|
-
txId: this.txId,
|
|
679
|
-
refundTxId: this.refundTxId,
|
|
680
|
-
};
|
|
681
|
-
}
|
|
682
|
-
/**
|
|
683
|
-
* @inheritDoc
|
|
684
|
-
* @internal
|
|
685
|
-
*/
|
|
686
|
-
_getInitiator() {
|
|
687
|
-
return this.recipient;
|
|
688
|
-
}
|
|
689
|
-
//////////////////////////////
|
|
690
|
-
//// Swap ticks & sync
|
|
691
|
-
/**
|
|
692
|
-
* @inheritDoc
|
|
693
|
-
* @internal
|
|
694
|
-
*/
|
|
695
|
-
async _sync(save) {
|
|
696
|
-
if (this._state === OnchainForGasSwapState.PR_CREATED) {
|
|
697
|
-
//Check if it's maybe already paid
|
|
698
|
-
const result = await this.checkAddress(false);
|
|
699
|
-
if (result) {
|
|
700
|
-
if (save)
|
|
701
|
-
await this._saveAndEmit();
|
|
702
|
-
return true;
|
|
703
|
-
}
|
|
704
|
-
}
|
|
705
|
-
return false;
|
|
706
|
-
}
|
|
707
|
-
/**
|
|
708
|
-
* @inheritDoc
|
|
709
|
-
* @internal
|
|
710
|
-
*/
|
|
711
|
-
_tick(save) {
|
|
712
|
-
return Promise.resolve(false);
|
|
713
|
-
}
|
|
714
|
-
}
|
|
715
|
-
exports.OnchainForGasSwap = OnchainForGasSwap;
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.OnchainForGasSwap = exports.OnchainForGasSwapState = void 0;
|
|
4
|
+
exports.isOnchainForGasSwapInit = isOnchainForGasSwapInit;
|
|
5
|
+
const SwapType_1 = require("../../../enums/SwapType");
|
|
6
|
+
const Utils_1 = require("../../../utils/Utils");
|
|
7
|
+
const BitcoinUtils_1 = require("../../../utils/BitcoinUtils");
|
|
8
|
+
const ISwap_1 = require("../../ISwap");
|
|
9
|
+
const TrustedIntermediaryAPI_1 = require("../../../intermediaries/apis/TrustedIntermediaryAPI");
|
|
10
|
+
const IBitcoinWallet_1 = require("../../../bitcoin/wallet/IBitcoinWallet");
|
|
11
|
+
const btc_signer_1 = require("@scure/btc-signer");
|
|
12
|
+
const SingleAddressBitcoinWallet_1 = require("../../../bitcoin/wallet/SingleAddressBitcoinWallet");
|
|
13
|
+
const buffer_1 = require("buffer");
|
|
14
|
+
const FeeType_1 = require("../../../enums/FeeType");
|
|
15
|
+
const PercentagePPM_1 = require("../../../types/fees/PercentagePPM");
|
|
16
|
+
const TokenAmount_1 = require("../../../types/TokenAmount");
|
|
17
|
+
const Token_1 = require("../../../types/Token");
|
|
18
|
+
const Logger_1 = require("../../../utils/Logger");
|
|
19
|
+
const TimeoutUtils_1 = require("../../../utils/TimeoutUtils");
|
|
20
|
+
const BitcoinWalletUtils_1 = require("../../../utils/BitcoinWalletUtils");
|
|
21
|
+
/**
|
|
22
|
+
* State enum for trusted on-chain gas swaps
|
|
23
|
+
*
|
|
24
|
+
* @category Swaps/Trusted Gas Swaps
|
|
25
|
+
*/
|
|
26
|
+
var OnchainForGasSwapState;
|
|
27
|
+
(function (OnchainForGasSwapState) {
|
|
28
|
+
/**
|
|
29
|
+
* The swap quote expired without user sending in the BTC
|
|
30
|
+
*/
|
|
31
|
+
OnchainForGasSwapState[OnchainForGasSwapState["EXPIRED"] = -3] = "EXPIRED";
|
|
32
|
+
/**
|
|
33
|
+
* The swap has failed after the intermediary already received the BTC on the source chain
|
|
34
|
+
*/
|
|
35
|
+
OnchainForGasSwapState[OnchainForGasSwapState["FAILED"] = -2] = "FAILED";
|
|
36
|
+
/**
|
|
37
|
+
* Swap was refunded and BTC returned to the user's refund address
|
|
38
|
+
*/
|
|
39
|
+
OnchainForGasSwapState[OnchainForGasSwapState["REFUNDED"] = -1] = "REFUNDED";
|
|
40
|
+
/**
|
|
41
|
+
* Swap was created, send the BTC to the swap address
|
|
42
|
+
*/
|
|
43
|
+
OnchainForGasSwapState[OnchainForGasSwapState["PR_CREATED"] = 0] = "PR_CREATED";
|
|
44
|
+
/**
|
|
45
|
+
* The swap is finished after the intermediary sent funds on the destination chain
|
|
46
|
+
*/
|
|
47
|
+
OnchainForGasSwapState[OnchainForGasSwapState["FINISHED"] = 1] = "FINISHED";
|
|
48
|
+
/**
|
|
49
|
+
* Swap is refundable because the intermediary cannot honor the swap request on the destination chain
|
|
50
|
+
*/
|
|
51
|
+
OnchainForGasSwapState[OnchainForGasSwapState["REFUNDABLE"] = 2] = "REFUNDABLE";
|
|
52
|
+
})(OnchainForGasSwapState || (exports.OnchainForGasSwapState = OnchainForGasSwapState = {}));
|
|
53
|
+
const OnchainForGasSwapStateDescription = {
|
|
54
|
+
[OnchainForGasSwapState.EXPIRED]: "The swap quote expired without user sending in the BTC",
|
|
55
|
+
[OnchainForGasSwapState.FAILED]: "The swap has failed after the intermediary already received the BTC on the source chain",
|
|
56
|
+
[OnchainForGasSwapState.REFUNDED]: "Swap was refunded and BTC returned to the user's refund address",
|
|
57
|
+
[OnchainForGasSwapState.PR_CREATED]: "Swap was created, send the BTC to the swap address",
|
|
58
|
+
[OnchainForGasSwapState.FINISHED]: "The swap is finished after the intermediary sent funds on the destination chain",
|
|
59
|
+
[OnchainForGasSwapState.REFUNDABLE]: "Swap is refundable because the intermediary cannot honor the swap request on the destination chain",
|
|
60
|
+
};
|
|
61
|
+
function isOnchainForGasSwapInit(obj) {
|
|
62
|
+
return typeof (obj.paymentHash) === "string" &&
|
|
63
|
+
typeof (obj.sequence) === "bigint" &&
|
|
64
|
+
typeof (obj.address) === "string" &&
|
|
65
|
+
typeof (obj.inputAmount) === "bigint" &&
|
|
66
|
+
typeof (obj.outputAmount) === "bigint" &&
|
|
67
|
+
typeof (obj.recipient) === "string" &&
|
|
68
|
+
typeof (obj.token) === "string" &&
|
|
69
|
+
(obj.refundAddress == null || typeof (obj.refundAddress) === "string") &&
|
|
70
|
+
(0, ISwap_1.isISwapInit)(obj);
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Trusted swap for Bitcoin -> Smart chains, to be used for minor amounts to get gas tokens on the
|
|
74
|
+
* destination chain, which is only needed for Solana, which still uses legacy swaps
|
|
75
|
+
*
|
|
76
|
+
* @category Swaps/Trusted Gas Swaps
|
|
77
|
+
*/
|
|
78
|
+
class OnchainForGasSwap extends ISwap_1.ISwap {
|
|
79
|
+
constructor(wrapper, initOrObj) {
|
|
80
|
+
if (isOnchainForGasSwapInit(initOrObj) && initOrObj.url != null)
|
|
81
|
+
initOrObj.url += "/frombtc_trusted";
|
|
82
|
+
super(wrapper, initOrObj);
|
|
83
|
+
this.TYPE = SwapType_1.SwapType.TRUSTED_FROM_BTC;
|
|
84
|
+
/**
|
|
85
|
+
* @internal
|
|
86
|
+
*/
|
|
87
|
+
this.swapStateDescription = OnchainForGasSwapStateDescription;
|
|
88
|
+
/**
|
|
89
|
+
* @internal
|
|
90
|
+
*/
|
|
91
|
+
this.swapStateName = (state) => OnchainForGasSwapState[state];
|
|
92
|
+
this.wrapper = wrapper;
|
|
93
|
+
if (isOnchainForGasSwapInit(initOrObj)) {
|
|
94
|
+
this.paymentHash = initOrObj.paymentHash;
|
|
95
|
+
this.sequence = initOrObj.sequence;
|
|
96
|
+
this.address = initOrObj.address;
|
|
97
|
+
this.inputAmount = initOrObj.inputAmount;
|
|
98
|
+
this.outputAmount = initOrObj.outputAmount;
|
|
99
|
+
this.recipient = initOrObj.recipient;
|
|
100
|
+
this.token = initOrObj.token;
|
|
101
|
+
this.refundAddress = initOrObj.refundAddress;
|
|
102
|
+
this._state = OnchainForGasSwapState.PR_CREATED;
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
this.paymentHash = initOrObj.paymentHash;
|
|
106
|
+
this.sequence = (0, Utils_1.toBigInt)(initOrObj.sequence);
|
|
107
|
+
this.address = initOrObj.address;
|
|
108
|
+
this.inputAmount = (0, Utils_1.toBigInt)(initOrObj.inputAmount);
|
|
109
|
+
this.outputAmount = (0, Utils_1.toBigInt)(initOrObj.outputAmount);
|
|
110
|
+
this.recipient = initOrObj.recipient;
|
|
111
|
+
this.token = initOrObj.token;
|
|
112
|
+
this.refundAddress = initOrObj.refundAddress;
|
|
113
|
+
this.scTxId = initOrObj.scTxId;
|
|
114
|
+
this.txId = initOrObj.txId;
|
|
115
|
+
this.refundTxId = initOrObj.refundTxId;
|
|
116
|
+
}
|
|
117
|
+
this.logger = (0, Logger_1.getLogger)("OnchainForGas(" + this.getId() + "): ");
|
|
118
|
+
this.tryRecomputeSwapPrice();
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* @inheritDoc
|
|
122
|
+
* @internal
|
|
123
|
+
*/
|
|
124
|
+
upgradeVersion() {
|
|
125
|
+
if (this.version == null) {
|
|
126
|
+
//Noop
|
|
127
|
+
this.version = 1;
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* @inheritDoc
|
|
132
|
+
* @internal
|
|
133
|
+
*/
|
|
134
|
+
tryRecomputeSwapPrice() {
|
|
135
|
+
if (this.swapFeeBtc == null && this.swapFee != null) {
|
|
136
|
+
this.swapFeeBtc = this.swapFee * this.getInput().rawAmount / this.getOutAmountWithoutFee();
|
|
137
|
+
}
|
|
138
|
+
super.tryRecomputeSwapPrice();
|
|
139
|
+
}
|
|
140
|
+
//////////////////////////////
|
|
141
|
+
//// Getters & utils
|
|
142
|
+
/**
|
|
143
|
+
* @inheritDoc
|
|
144
|
+
* @internal
|
|
145
|
+
*/
|
|
146
|
+
_getEscrowHash() {
|
|
147
|
+
return this.paymentHash;
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* @inheritDoc
|
|
151
|
+
*/
|
|
152
|
+
getOutputAddress() {
|
|
153
|
+
return this.recipient;
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* @inheritDoc
|
|
157
|
+
*/
|
|
158
|
+
getInputAddress() {
|
|
159
|
+
//TODO: Fuck this, it's not used anyway
|
|
160
|
+
return null;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* @inheritDoc
|
|
164
|
+
*/
|
|
165
|
+
getInputTxId() {
|
|
166
|
+
return this.txId ?? null;
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* @inheritDoc
|
|
170
|
+
*/
|
|
171
|
+
getOutputTxId() {
|
|
172
|
+
return this.scTxId ?? null;
|
|
173
|
+
}
|
|
174
|
+
/**
|
|
175
|
+
* @inheritDoc
|
|
176
|
+
*/
|
|
177
|
+
getId() {
|
|
178
|
+
return this.paymentHash;
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* @inheritDoc
|
|
182
|
+
*/
|
|
183
|
+
getAddress() {
|
|
184
|
+
return this.address;
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* @inheritDoc
|
|
188
|
+
*/
|
|
189
|
+
getHyperlink() {
|
|
190
|
+
return "bitcoin:" + this.address + "?amount=" + encodeURIComponent((Number(this.inputAmount) / 100000000).toString(10));
|
|
191
|
+
}
|
|
192
|
+
/**
|
|
193
|
+
* @inheritDoc
|
|
194
|
+
*/
|
|
195
|
+
requiresAction() {
|
|
196
|
+
return this._state === OnchainForGasSwapState.REFUNDABLE;
|
|
197
|
+
}
|
|
198
|
+
/**
|
|
199
|
+
* @inheritDoc
|
|
200
|
+
*/
|
|
201
|
+
isFinished() {
|
|
202
|
+
return this._state === OnchainForGasSwapState.FINISHED || this._state === OnchainForGasSwapState.FAILED || this._state === OnchainForGasSwapState.EXPIRED || this._state === OnchainForGasSwapState.REFUNDED;
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* @inheritDoc
|
|
206
|
+
*/
|
|
207
|
+
isQuoteExpired() {
|
|
208
|
+
return this._state === OnchainForGasSwapState.EXPIRED;
|
|
209
|
+
}
|
|
210
|
+
/**
|
|
211
|
+
* @inheritDoc
|
|
212
|
+
*/
|
|
213
|
+
isQuoteSoftExpired() {
|
|
214
|
+
return this.expiry < Date.now();
|
|
215
|
+
}
|
|
216
|
+
/**
|
|
217
|
+
* @inheritDoc
|
|
218
|
+
*/
|
|
219
|
+
isFailed() {
|
|
220
|
+
return this._state === OnchainForGasSwapState.FAILED;
|
|
221
|
+
}
|
|
222
|
+
/**
|
|
223
|
+
* @inheritDoc
|
|
224
|
+
*/
|
|
225
|
+
isSuccessful() {
|
|
226
|
+
return this._state === OnchainForGasSwapState.FINISHED;
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* @inheritDoc
|
|
230
|
+
*/
|
|
231
|
+
isInProgress() {
|
|
232
|
+
return (this._state === OnchainForGasSwapState.PR_CREATED && this.txId != null) ||
|
|
233
|
+
(this._state === OnchainForGasSwapState.REFUNDABLE && this.refundAddress != null);
|
|
234
|
+
}
|
|
235
|
+
/**
|
|
236
|
+
* @inheritDoc
|
|
237
|
+
* @internal
|
|
238
|
+
*/
|
|
239
|
+
_verifyQuoteDefinitelyExpired() {
|
|
240
|
+
return Promise.resolve(this.expiry < Date.now());
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* @inheritDoc
|
|
244
|
+
* @internal
|
|
245
|
+
*/
|
|
246
|
+
_verifyQuoteValid() {
|
|
247
|
+
return Promise.resolve(this.expiry > Date.now());
|
|
248
|
+
}
|
|
249
|
+
//////////////////////////////
|
|
250
|
+
//// Amounts & fees
|
|
251
|
+
/**
|
|
252
|
+
* Returns an output amount in base units without a swap fee included, hence this value
|
|
253
|
+
* is larger than the actual output amount
|
|
254
|
+
*
|
|
255
|
+
* @internal
|
|
256
|
+
*/
|
|
257
|
+
getOutAmountWithoutFee() {
|
|
258
|
+
return this.outputAmount + (this.swapFee ?? 0n);
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* @inheritDoc
|
|
262
|
+
*/
|
|
263
|
+
getOutputToken() {
|
|
264
|
+
return this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()];
|
|
265
|
+
}
|
|
266
|
+
/**
|
|
267
|
+
* @inheritDoc
|
|
268
|
+
*/
|
|
269
|
+
getOutput() {
|
|
270
|
+
return (0, TokenAmount_1.toTokenAmount)(this.outputAmount, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo);
|
|
271
|
+
}
|
|
272
|
+
/**
|
|
273
|
+
* @inheritDoc
|
|
274
|
+
*/
|
|
275
|
+
getInputToken() {
|
|
276
|
+
return Token_1.BitcoinTokens.BTC;
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* @inheritDoc
|
|
280
|
+
*/
|
|
281
|
+
getInput() {
|
|
282
|
+
return (0, TokenAmount_1.toTokenAmount)(this.inputAmount, Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
283
|
+
}
|
|
284
|
+
/**
|
|
285
|
+
* @inheritDoc
|
|
286
|
+
*/
|
|
287
|
+
getInputWithoutFee() {
|
|
288
|
+
return (0, TokenAmount_1.toTokenAmount)(this.inputAmount - (this.swapFeeBtc ?? 0n), Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
289
|
+
}
|
|
290
|
+
/**
|
|
291
|
+
* Returns the swap fee charged by the intermediary (LP) on this swap
|
|
292
|
+
*
|
|
293
|
+
* @internal
|
|
294
|
+
*/
|
|
295
|
+
getSwapFee() {
|
|
296
|
+
if (this.pricingInfo == null)
|
|
297
|
+
throw new Error("No pricing info known!");
|
|
298
|
+
const feeWithoutBaseFee = this.swapFeeBtc == null ? 0n : this.swapFeeBtc - this.pricingInfo.satsBaseFee;
|
|
299
|
+
const swapFeePPM = feeWithoutBaseFee * 1000000n / this.getInputWithoutFee().rawAmount;
|
|
300
|
+
const amountInSrcToken = (0, TokenAmount_1.toTokenAmount)(this.swapFeeBtc ?? 0n, Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
301
|
+
return {
|
|
302
|
+
amountInSrcToken,
|
|
303
|
+
amountInDstToken: (0, TokenAmount_1.toTokenAmount)(this.swapFee ?? 0n, this.wrapper._tokens[this.wrapper._chain.getNativeCurrencyAddress()], this.wrapper._prices, this.pricingInfo),
|
|
304
|
+
currentUsdValue: amountInSrcToken.currentUsdValue,
|
|
305
|
+
usdValue: amountInSrcToken.usdValue,
|
|
306
|
+
pastUsdValue: amountInSrcToken.pastUsdValue,
|
|
307
|
+
composition: {
|
|
308
|
+
base: (0, TokenAmount_1.toTokenAmount)(this.pricingInfo.satsBaseFee, Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo),
|
|
309
|
+
percentage: (0, PercentagePPM_1.ppmToPercentage)(swapFeePPM)
|
|
310
|
+
}
|
|
311
|
+
};
|
|
312
|
+
}
|
|
313
|
+
/**
|
|
314
|
+
* @inheritDoc
|
|
315
|
+
*/
|
|
316
|
+
getFee() {
|
|
317
|
+
return this.getSwapFee();
|
|
318
|
+
}
|
|
319
|
+
/**
|
|
320
|
+
* @inheritDoc
|
|
321
|
+
*/
|
|
322
|
+
getFeeBreakdown() {
|
|
323
|
+
return [{
|
|
324
|
+
type: FeeType_1.FeeType.SWAP,
|
|
325
|
+
fee: this.getSwapFee()
|
|
326
|
+
}];
|
|
327
|
+
}
|
|
328
|
+
/**
|
|
329
|
+
* @inheritDoc
|
|
330
|
+
*/
|
|
331
|
+
getRequiredConfirmationsCount() {
|
|
332
|
+
return 1;
|
|
333
|
+
}
|
|
334
|
+
/**
|
|
335
|
+
* @inheritDoc
|
|
336
|
+
*/
|
|
337
|
+
async getFundedPsbt(_bitcoinWallet, feeRate, additionalOutputs) {
|
|
338
|
+
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
339
|
+
throw new Error("Swap already paid for!");
|
|
340
|
+
let bitcoinWallet;
|
|
341
|
+
if ((0, IBitcoinWallet_1.isIBitcoinWallet)(_bitcoinWallet)) {
|
|
342
|
+
bitcoinWallet = _bitcoinWallet;
|
|
343
|
+
}
|
|
344
|
+
else {
|
|
345
|
+
bitcoinWallet = new SingleAddressBitcoinWallet_1.SingleAddressBitcoinWallet(this.wrapper._btcRpc, this.wrapper._options.bitcoinNetwork, _bitcoinWallet);
|
|
346
|
+
}
|
|
347
|
+
//TODO: Maybe re-introduce fee rate check here if passed from the user
|
|
348
|
+
if (feeRate == null) {
|
|
349
|
+
feeRate = await bitcoinWallet.getFeeRate();
|
|
350
|
+
}
|
|
351
|
+
const basePsbt = new btc_signer_1.Transaction({
|
|
352
|
+
allowUnknownOutputs: true,
|
|
353
|
+
allowLegacyWitnessUtxo: true
|
|
354
|
+
});
|
|
355
|
+
basePsbt.addOutput({
|
|
356
|
+
amount: this.outputAmount,
|
|
357
|
+
script: (0, BitcoinUtils_1.toOutputScript)(this.wrapper._options.bitcoinNetwork, this.address)
|
|
358
|
+
});
|
|
359
|
+
if (additionalOutputs != null)
|
|
360
|
+
additionalOutputs.forEach(output => {
|
|
361
|
+
basePsbt.addOutput({
|
|
362
|
+
amount: output.amount,
|
|
363
|
+
script: output.outputScript ?? (0, BitcoinUtils_1.toOutputScript)(this.wrapper._options.bitcoinNetwork, output.address)
|
|
364
|
+
});
|
|
365
|
+
});
|
|
366
|
+
const psbt = await bitcoinWallet.fundPsbt(basePsbt, feeRate);
|
|
367
|
+
//Sign every input
|
|
368
|
+
const signInputs = [];
|
|
369
|
+
for (let i = 0; i < psbt.inputsLength; i++) {
|
|
370
|
+
signInputs.push(i);
|
|
371
|
+
}
|
|
372
|
+
const serializedPsbt = buffer_1.Buffer.from(psbt.toPSBT());
|
|
373
|
+
return {
|
|
374
|
+
psbt,
|
|
375
|
+
psbtHex: serializedPsbt.toString("hex"),
|
|
376
|
+
psbtBase64: serializedPsbt.toString("base64"),
|
|
377
|
+
signInputs
|
|
378
|
+
};
|
|
379
|
+
}
|
|
380
|
+
/**
|
|
381
|
+
* @inheritDoc
|
|
382
|
+
*/
|
|
383
|
+
async submitPsbt(_psbt) {
|
|
384
|
+
const psbt = (0, BitcoinUtils_1.parsePsbtTransaction)(_psbt);
|
|
385
|
+
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
386
|
+
throw new Error("Swap already paid for!");
|
|
387
|
+
//Ensure not expired
|
|
388
|
+
if (this.expiry < Date.now()) {
|
|
389
|
+
throw new Error("Swap expired!");
|
|
390
|
+
}
|
|
391
|
+
const output0 = psbt.getOutput(0);
|
|
392
|
+
if (output0.amount !== this.outputAmount)
|
|
393
|
+
throw new Error("PSBT output amount invalid, expected: " + this.outputAmount + " got: " + output0.amount);
|
|
394
|
+
const expectedOutputScript = (0, BitcoinUtils_1.toOutputScript)(this.wrapper._options.bitcoinNetwork, this.address);
|
|
395
|
+
if (output0.script == null || !expectedOutputScript.equals(output0.script))
|
|
396
|
+
throw new Error("PSBT output script invalid!");
|
|
397
|
+
if (!psbt.isFinal)
|
|
398
|
+
psbt.finalize();
|
|
399
|
+
return await this.wrapper._btcRpc.sendRawTransaction(buffer_1.Buffer.from(psbt.toBytes(true, true)).toString("hex"));
|
|
400
|
+
}
|
|
401
|
+
/**
|
|
402
|
+
* @inheritDoc
|
|
403
|
+
*/
|
|
404
|
+
async estimateBitcoinFee(_bitcoinWallet, feeRate) {
|
|
405
|
+
const bitcoinWallet = (0, BitcoinWalletUtils_1.toBitcoinWallet)(_bitcoinWallet, this.wrapper._btcRpc, this.wrapper._options.bitcoinNetwork);
|
|
406
|
+
const txFee = await bitcoinWallet.getTransactionFee(this.address, this.inputAmount, feeRate);
|
|
407
|
+
if (txFee == null)
|
|
408
|
+
return null;
|
|
409
|
+
return (0, TokenAmount_1.toTokenAmount)(BigInt(txFee), Token_1.BitcoinTokens.BTC, this.wrapper._prices, this.pricingInfo);
|
|
410
|
+
}
|
|
411
|
+
/**
|
|
412
|
+
* @inheritDoc
|
|
413
|
+
*/
|
|
414
|
+
async sendBitcoinTransaction(wallet, feeRate) {
|
|
415
|
+
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
416
|
+
throw new Error("Swap already paid for!");
|
|
417
|
+
//Ensure not expired
|
|
418
|
+
if (this.expiry < Date.now()) {
|
|
419
|
+
throw new Error("Swap expired!");
|
|
420
|
+
}
|
|
421
|
+
if ((0, IBitcoinWallet_1.isIBitcoinWallet)(wallet)) {
|
|
422
|
+
return await wallet.sendTransaction(this.address, this.inputAmount, feeRate);
|
|
423
|
+
}
|
|
424
|
+
else {
|
|
425
|
+
const { psbt, psbtHex, psbtBase64, signInputs } = await this.getFundedPsbt(wallet, feeRate);
|
|
426
|
+
const signedPsbt = await wallet.signPsbt({
|
|
427
|
+
psbt, psbtHex, psbtBase64
|
|
428
|
+
}, signInputs);
|
|
429
|
+
return await this.submitPsbt(signedPsbt);
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
/**
|
|
433
|
+
* @inheritDoc
|
|
434
|
+
*
|
|
435
|
+
* @param options.bitcoinWallet Optional bitcoin wallet address specification to return a funded PSBT,
|
|
436
|
+
* if not provided an address is returned instead.
|
|
437
|
+
*/
|
|
438
|
+
async txsExecute(options) {
|
|
439
|
+
if (this._state === OnchainForGasSwapState.PR_CREATED) {
|
|
440
|
+
if (!await this._verifyQuoteValid())
|
|
441
|
+
throw new Error("Quote already expired or close to expiry!");
|
|
442
|
+
return [
|
|
443
|
+
{
|
|
444
|
+
name: "Payment",
|
|
445
|
+
description: "Send funds to the bitcoin swap address",
|
|
446
|
+
chain: "BITCOIN",
|
|
447
|
+
txs: [
|
|
448
|
+
options?.bitcoinWallet == null ? {
|
|
449
|
+
address: this.address,
|
|
450
|
+
amount: Number(this.inputAmount),
|
|
451
|
+
hyperlink: this.getHyperlink(),
|
|
452
|
+
type: "ADDRESS"
|
|
453
|
+
} : {
|
|
454
|
+
...await this.getFundedPsbt(options.bitcoinWallet),
|
|
455
|
+
type: "FUNDED_PSBT"
|
|
456
|
+
}
|
|
457
|
+
]
|
|
458
|
+
}
|
|
459
|
+
];
|
|
460
|
+
}
|
|
461
|
+
throw new Error("Invalid swap state to obtain execution txns, required PR_CREATED or CLAIM_COMMITED");
|
|
462
|
+
}
|
|
463
|
+
/**
|
|
464
|
+
* @remark Not supported
|
|
465
|
+
*/
|
|
466
|
+
async execute() {
|
|
467
|
+
throw new Error("Not supported");
|
|
468
|
+
}
|
|
469
|
+
/**
|
|
470
|
+
* @inheritDoc
|
|
471
|
+
*
|
|
472
|
+
* @param options.bitcoinWallet Optional bitcoin wallet address specification to return a funded PSBT,
|
|
473
|
+
* if not provided an address is returned instead.
|
|
474
|
+
*/
|
|
475
|
+
async getCurrentActions(options) {
|
|
476
|
+
try {
|
|
477
|
+
return await this.txsExecute(options);
|
|
478
|
+
}
|
|
479
|
+
catch (e) {
|
|
480
|
+
return [];
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
//////////////////////////////
|
|
484
|
+
//// Payment
|
|
485
|
+
/**
|
|
486
|
+
* Queries the intermediary (LP) node for the state of the swap
|
|
487
|
+
*
|
|
488
|
+
* @param save Whether the save the result or not
|
|
489
|
+
*
|
|
490
|
+
* @returns Whether the swap was successful as `boolean` or `null` if the swap is still pending
|
|
491
|
+
* @internal
|
|
492
|
+
*/
|
|
493
|
+
async checkAddress(save = true) {
|
|
494
|
+
if (this._state === OnchainForGasSwapState.FAILED ||
|
|
495
|
+
this._state === OnchainForGasSwapState.EXPIRED ||
|
|
496
|
+
this._state === OnchainForGasSwapState.REFUNDED)
|
|
497
|
+
return false;
|
|
498
|
+
if (this._state === OnchainForGasSwapState.FINISHED)
|
|
499
|
+
return false;
|
|
500
|
+
if (this.url == null)
|
|
501
|
+
return false;
|
|
502
|
+
const response = await TrustedIntermediaryAPI_1.TrustedIntermediaryAPI.getAddressStatus(this.url, this.paymentHash, this.sequence, this.wrapper._options.getRequestTimeout);
|
|
503
|
+
switch (response.code) {
|
|
504
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.AWAIT_PAYMENT:
|
|
505
|
+
if (this.txId != null) {
|
|
506
|
+
this.txId = undefined;
|
|
507
|
+
if (save)
|
|
508
|
+
await this._save();
|
|
509
|
+
return true;
|
|
510
|
+
}
|
|
511
|
+
return false;
|
|
512
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.AWAIT_CONFIRMATION:
|
|
513
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.PENDING:
|
|
514
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.TX_SENT:
|
|
515
|
+
const inputAmount = BigInt(response.data.adjustedAmount);
|
|
516
|
+
const outputAmount = BigInt(response.data.adjustedTotal);
|
|
517
|
+
const adjustedFee = response.data.adjustedFee == null ? null : BigInt(response.data.adjustedFee);
|
|
518
|
+
const adjustedFeeSats = response.data.adjustedFeeSats == null ? null : BigInt(response.data.adjustedFeeSats);
|
|
519
|
+
const txId = response.data.txId;
|
|
520
|
+
if (this.txId != txId ||
|
|
521
|
+
this.inputAmount !== inputAmount ||
|
|
522
|
+
this.outputAmount !== outputAmount) {
|
|
523
|
+
this.txId = txId;
|
|
524
|
+
this.inputAmount = inputAmount;
|
|
525
|
+
this.outputAmount = outputAmount;
|
|
526
|
+
if (adjustedFee != null)
|
|
527
|
+
this.swapFee = adjustedFee;
|
|
528
|
+
if (adjustedFeeSats != null)
|
|
529
|
+
this.swapFeeBtc = adjustedFeeSats;
|
|
530
|
+
if (save)
|
|
531
|
+
await this._save();
|
|
532
|
+
return true;
|
|
533
|
+
}
|
|
534
|
+
return false;
|
|
535
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.PAID:
|
|
536
|
+
const txStatus = await this.wrapper._chain.getTxIdStatus(response.data.txId);
|
|
537
|
+
if (txStatus === "success") {
|
|
538
|
+
this._state = OnchainForGasSwapState.FINISHED;
|
|
539
|
+
this.scTxId = response.data.txId;
|
|
540
|
+
if (save)
|
|
541
|
+
await this._saveAndEmit();
|
|
542
|
+
return true;
|
|
543
|
+
}
|
|
544
|
+
return false;
|
|
545
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.EXPIRED:
|
|
546
|
+
this._state = OnchainForGasSwapState.EXPIRED;
|
|
547
|
+
if (save)
|
|
548
|
+
await this._saveAndEmit();
|
|
549
|
+
return true;
|
|
550
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.REFUNDABLE:
|
|
551
|
+
if (this._state === OnchainForGasSwapState.REFUNDABLE)
|
|
552
|
+
return null;
|
|
553
|
+
this._state = OnchainForGasSwapState.REFUNDABLE;
|
|
554
|
+
if (save)
|
|
555
|
+
await this._saveAndEmit();
|
|
556
|
+
return true;
|
|
557
|
+
case TrustedIntermediaryAPI_1.AddressStatusResponseCodes.REFUNDED:
|
|
558
|
+
this._state = OnchainForGasSwapState.REFUNDED;
|
|
559
|
+
this.refundTxId = response.data.txId;
|
|
560
|
+
if (save)
|
|
561
|
+
await this._saveAndEmit();
|
|
562
|
+
return true;
|
|
563
|
+
default:
|
|
564
|
+
this._state = OnchainForGasSwapState.FAILED;
|
|
565
|
+
if (save)
|
|
566
|
+
await this._saveAndEmit();
|
|
567
|
+
return true;
|
|
568
|
+
}
|
|
569
|
+
}
|
|
570
|
+
/**
|
|
571
|
+
* Sets the bitcoin address used for possible refunds in case something goes wrong with the swap
|
|
572
|
+
*
|
|
573
|
+
* @param refundAddress Bitcoin address to receive the refund to
|
|
574
|
+
* @internal
|
|
575
|
+
*/
|
|
576
|
+
async setRefundAddress(refundAddress) {
|
|
577
|
+
if (this.refundAddress != null) {
|
|
578
|
+
if (this.refundAddress !== refundAddress)
|
|
579
|
+
throw new Error("Different refund address already set!");
|
|
580
|
+
return;
|
|
581
|
+
}
|
|
582
|
+
if (this.url == null)
|
|
583
|
+
throw new Error("LP URL not known, cannot set refund address!");
|
|
584
|
+
await TrustedIntermediaryAPI_1.TrustedIntermediaryAPI.setRefundAddress(this.url, this.paymentHash, this.sequence, refundAddress, this.wrapper._options.getRequestTimeout);
|
|
585
|
+
this.refundAddress = refundAddress;
|
|
586
|
+
}
|
|
587
|
+
/**
|
|
588
|
+
* @inheritDoc
|
|
589
|
+
*/
|
|
590
|
+
async waitForBitcoinTransaction(updateCallback, checkIntervalSeconds = 5, abortSignal) {
|
|
591
|
+
if (this._state !== OnchainForGasSwapState.PR_CREATED)
|
|
592
|
+
throw new Error("Must be in PR_CREATED state!");
|
|
593
|
+
if (!this.initiated) {
|
|
594
|
+
this.initiated = true;
|
|
595
|
+
await this._saveAndEmit();
|
|
596
|
+
}
|
|
597
|
+
while (!abortSignal?.aborted &&
|
|
598
|
+
this._state === OnchainForGasSwapState.PR_CREATED) {
|
|
599
|
+
await this.checkAddress(true);
|
|
600
|
+
if (this.txId != null && updateCallback != null) {
|
|
601
|
+
const res = await this.wrapper._btcRpc.getTransaction(this.txId);
|
|
602
|
+
if (res == null) {
|
|
603
|
+
updateCallback();
|
|
604
|
+
}
|
|
605
|
+
else if (res.confirmations != null && res.confirmations > 0) {
|
|
606
|
+
updateCallback(res.txid, res.confirmations, 1, 0);
|
|
607
|
+
}
|
|
608
|
+
else {
|
|
609
|
+
const delay = await this.wrapper._btcRpc.getConfirmationDelay(res, 1);
|
|
610
|
+
updateCallback(res.txid, 0, 1, delay ?? undefined);
|
|
611
|
+
}
|
|
612
|
+
}
|
|
613
|
+
if (this._state === OnchainForGasSwapState.PR_CREATED)
|
|
614
|
+
await (0, TimeoutUtils_1.timeoutPromise)(checkIntervalSeconds * 1000, abortSignal);
|
|
615
|
+
}
|
|
616
|
+
if (this._state === OnchainForGasSwapState.REFUNDABLE ||
|
|
617
|
+
this._state === OnchainForGasSwapState.REFUNDED)
|
|
618
|
+
return this.txId;
|
|
619
|
+
if (this.isQuoteExpired())
|
|
620
|
+
throw new Error("Swap expired");
|
|
621
|
+
if (this.isFailed())
|
|
622
|
+
throw new Error("Swap failed");
|
|
623
|
+
return this.txId;
|
|
624
|
+
}
|
|
625
|
+
/**
|
|
626
|
+
* Waits till the LP processes a refund for a failed swap. The swap must be in
|
|
627
|
+
* {@link OnchainForGasSwapState.REFUNDABLE} state
|
|
628
|
+
*
|
|
629
|
+
* @param checkIntervalSeconds How often to check (default 5 seconds)
|
|
630
|
+
* @param abortSignal Abort signal
|
|
631
|
+
*/
|
|
632
|
+
async waitTillRefunded(checkIntervalSeconds, abortSignal) {
|
|
633
|
+
checkIntervalSeconds ??= 5;
|
|
634
|
+
if (this._state === OnchainForGasSwapState.REFUNDED)
|
|
635
|
+
return;
|
|
636
|
+
if (this._state !== OnchainForGasSwapState.REFUNDABLE)
|
|
637
|
+
throw new Error("Must be in REFUNDABLE state!");
|
|
638
|
+
while (!abortSignal?.aborted &&
|
|
639
|
+
this._state === OnchainForGasSwapState.REFUNDABLE) {
|
|
640
|
+
await this.checkAddress(true);
|
|
641
|
+
if (this._state === OnchainForGasSwapState.REFUNDABLE)
|
|
642
|
+
await (0, TimeoutUtils_1.timeoutPromise)(checkIntervalSeconds * 1000, abortSignal);
|
|
643
|
+
}
|
|
644
|
+
if (this.isQuoteExpired())
|
|
645
|
+
throw new Error("Swap expired");
|
|
646
|
+
if (this.isFailed())
|
|
647
|
+
throw new Error("Swap failed");
|
|
648
|
+
}
|
|
649
|
+
/**
|
|
650
|
+
* Requests a refund after the swap failed, this also waits till the refund is actually sent by the
|
|
651
|
+
* intermediary (LP). The swap must be in {@link OnchainForGasSwapState.REFUNDABLE} state
|
|
652
|
+
*
|
|
653
|
+
* @param refundAddress Bitcoin address to receive the refund to
|
|
654
|
+
* @param abortSignal Abort signal
|
|
655
|
+
*/
|
|
656
|
+
async requestRefund(refundAddress, abortSignal) {
|
|
657
|
+
if (refundAddress != null)
|
|
658
|
+
await this.setRefundAddress(refundAddress);
|
|
659
|
+
await this.waitTillRefunded(undefined, abortSignal);
|
|
660
|
+
}
|
|
661
|
+
//////////////////////////////
|
|
662
|
+
//// Storage
|
|
663
|
+
/**
|
|
664
|
+
* @inheritDoc
|
|
665
|
+
*/
|
|
666
|
+
serialize() {
|
|
667
|
+
return {
|
|
668
|
+
...super.serialize(),
|
|
669
|
+
paymentHash: this.paymentHash,
|
|
670
|
+
sequence: this.sequence == null ? null : this.sequence.toString(10),
|
|
671
|
+
address: this.address,
|
|
672
|
+
inputAmount: this.inputAmount == null ? null : this.inputAmount.toString(10),
|
|
673
|
+
outputAmount: this.outputAmount == null ? null : this.outputAmount.toString(10),
|
|
674
|
+
recipient: this.recipient,
|
|
675
|
+
token: this.token,
|
|
676
|
+
refundAddress: this.refundAddress,
|
|
677
|
+
scTxId: this.scTxId,
|
|
678
|
+
txId: this.txId,
|
|
679
|
+
refundTxId: this.refundTxId,
|
|
680
|
+
};
|
|
681
|
+
}
|
|
682
|
+
/**
|
|
683
|
+
* @inheritDoc
|
|
684
|
+
* @internal
|
|
685
|
+
*/
|
|
686
|
+
_getInitiator() {
|
|
687
|
+
return this.recipient;
|
|
688
|
+
}
|
|
689
|
+
//////////////////////////////
|
|
690
|
+
//// Swap ticks & sync
|
|
691
|
+
/**
|
|
692
|
+
* @inheritDoc
|
|
693
|
+
* @internal
|
|
694
|
+
*/
|
|
695
|
+
async _sync(save) {
|
|
696
|
+
if (this._state === OnchainForGasSwapState.PR_CREATED) {
|
|
697
|
+
//Check if it's maybe already paid
|
|
698
|
+
const result = await this.checkAddress(false);
|
|
699
|
+
if (result) {
|
|
700
|
+
if (save)
|
|
701
|
+
await this._saveAndEmit();
|
|
702
|
+
return true;
|
|
703
|
+
}
|
|
704
|
+
}
|
|
705
|
+
return false;
|
|
706
|
+
}
|
|
707
|
+
/**
|
|
708
|
+
* @inheritDoc
|
|
709
|
+
* @internal
|
|
710
|
+
*/
|
|
711
|
+
_tick(save) {
|
|
712
|
+
return Promise.resolve(false);
|
|
713
|
+
}
|
|
714
|
+
}
|
|
715
|
+
exports.OnchainForGasSwap = OnchainForGasSwap;
|