@atomiqlabs/sdk 8.6.6 → 8.6.7
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 +52 -51
- package/dist/bitcoin/coinselect2/blackjack.d.ts +6 -6
- package/dist/bitcoin/coinselect2/blackjack.js +38 -37
- package/dist/bitcoin/coinselect2/index.d.ts +19 -17
- package/dist/bitcoin/coinselect2/index.js +69 -69
- package/dist/bitcoin/coinselect2/utils.d.ts +77 -75
- package/dist/bitcoin/coinselect2/utils.js +123 -123
- package/dist/bitcoin/wallet/BitcoinWallet.d.ts +130 -128
- package/dist/bitcoin/wallet/BitcoinWallet.js +322 -322
- package/dist/bitcoin/wallet/IBitcoinWallet.d.ts +78 -78
- package/dist/bitcoin/wallet/IBitcoinWallet.js +21 -20
- package/dist/bitcoin/wallet/SingleAddressBitcoinWallet.d.ts +101 -99
- 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 +91 -90
- 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 -42
- package/dist/http/paramcoders/ParamDecoder.js +137 -137
- package/dist/http/paramcoders/ParamEncoder.d.ts +20 -18
- 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 -11
- 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 +174 -173
- 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 +199 -198
- package/dist/intermediaries/IntermediaryDiscovery.js +406 -406
- package/dist/intermediaries/apis/IntermediaryAPI.d.ts +439 -437
- package/dist/intermediaries/apis/IntermediaryAPI.js +603 -603
- 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 +687 -686
- 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 +14 -13
- package/dist/swaps/IBTCWalletSwap.d.ts +73 -73
- package/dist/swaps/IBTCWalletSwap.js +18 -17
- package/dist/swaps/IClaimableSwap.d.ts +49 -49
- package/dist/swaps/IClaimableSwap.js +15 -14
- 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 -13
- package/dist/swaps/ISwap.d.ts +387 -386
- package/dist/swaps/ISwap.js +346 -346
- package/dist/swaps/ISwapWithGasDrop.d.ts +21 -21
- package/dist/swaps/ISwapWithGasDrop.js +12 -11
- package/dist/swaps/ISwapWrapper.d.ts +285 -283
- 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 +135 -133
- package/dist/swaps/escrow_swaps/IEscrowSwap.js +169 -169
- package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.d.ts +115 -114
- package/dist/swaps/escrow_swaps/IEscrowSwapWrapper.js +134 -134
- package/dist/swaps/escrow_swaps/frombtc/IFromBTCLNWrapper.d.ts +101 -98
- 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 +531 -529
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNSwap.js +1285 -1285
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.d.ts +184 -181
- package/dist/swaps/escrow_swaps/frombtc/ln/FromBTCLNWrapper.js +418 -418
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.d.ts +583 -581
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoSwap.js +1371 -1371
- package/dist/swaps/escrow_swaps/frombtc/ln_auto/FromBTCLNAutoWrapper.d.ts +228 -225
- 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 +191 -190
- 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 +127 -125
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNSwap.js +256 -256
- package/dist/swaps/escrow_swaps/tobtc/ln/ToBTCLNWrapper.d.ts +242 -241
- 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 +128 -127
- 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 +214 -213
- package/dist/swaps/spv_swaps/SpvFromBTCWrapper.js +756 -756
- 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 +69 -68
- 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 +57 -56
- 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 +43 -42
- 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 +60 -59
- 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 -17
- package/dist/types/lnurl/LNURLPay.d.ts +61 -61
- package/dist/types/lnurl/LNURLPay.js +31 -30
- package/dist/types/lnurl/LNURLWithdraw.d.ts +48 -48
- package/dist/types/lnurl/LNURLWithdraw.js +27 -26
- package/dist/types/wallets/LightningInvoiceCreateService.d.ts +24 -24
- package/dist/types/wallets/LightningInvoiceCreateService.js +15 -14
- 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 -69
- package/dist/utils/BitcoinUtils.d.ts +14 -12
- package/dist/utils/BitcoinUtils.js +102 -101
- package/dist/utils/BitcoinWalletUtils.d.ts +7 -7
- package/dist/utils/BitcoinWalletUtils.js +14 -13
- package/dist/utils/Logger.d.ts +7 -7
- package/dist/utils/Logger.js +12 -11
- package/dist/utils/RetryUtils.d.ts +22 -22
- package/dist/utils/RetryUtils.js +67 -66
- 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 -54
- package/dist/utils/TokenUtils.d.ts +19 -19
- package/dist/utils/TokenUtils.js +37 -36
- package/dist/utils/TypeUtils.d.ts +7 -7
- package/dist/utils/TypeUtils.js +2 -2
- package/dist/utils/Utils.d.ts +58 -56
- package/dist/utils/Utils.js +194 -193
- package/package.json +1 -1
package/dist/lnurl/LNURL.js
CHANGED
|
@@ -1,321 +1,321 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.LNURL = void 0;
|
|
4
|
-
|
|
5
|
-
const
|
|
6
|
-
const
|
|
7
|
-
const
|
|
8
|
-
const
|
|
9
|
-
const
|
|
10
|
-
const
|
|
11
|
-
const
|
|
12
|
-
const
|
|
13
|
-
const
|
|
14
|
-
const
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
(obj.
|
|
23
|
-
(obj.
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
obj.url
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
obj.
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
const BASE64_REGEX = /^(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{2}==|[A-Za-z0-9+/]{3}=)?$/;
|
|
46
|
-
const MAIL_REGEX = /(?:[A-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[A-z0-9!#$%&'*+/=?^_`{|}~-]+)*|"(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21\x23-\x5b\x5d-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])*")@(?:(?:[A-z0-9](?:[A-z0-9-]*[A-z0-9])?\.)+[A-z0-9](?:[A-z0-9-]*[A-z0-9])?|\[(?:(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9]))\.){3}(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9])|[A-z0-9-]*[A-z0-9]:(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21-\x5a\x53-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])+)\])/;
|
|
47
|
-
/**
|
|
48
|
-
* LNURL encoding/decoding and callback handling
|
|
49
|
-
* @category Bitcoin
|
|
50
|
-
*/
|
|
51
|
-
class LNURL {
|
|
52
|
-
static findBech32LNURL(str) {
|
|
53
|
-
const arr = /,*?((lnurl)([0-9]{1,}[a-z0-9]+){1})/.exec(str.toLowerCase());
|
|
54
|
-
if (arr == null)
|
|
55
|
-
return null;
|
|
56
|
-
return arr[1];
|
|
57
|
-
}
|
|
58
|
-
static isBech32LNURL(str) {
|
|
59
|
-
return this.findBech32LNURL(str) != null;
|
|
60
|
-
}
|
|
61
|
-
/**
|
|
62
|
-
* Checks whether a provided string is bare (non bech32 encoded) lnurl
|
|
63
|
-
* @param str
|
|
64
|
-
* @private
|
|
65
|
-
*/
|
|
66
|
-
static isBareLNURL(str) {
|
|
67
|
-
try {
|
|
68
|
-
return str.startsWith("lnurlw://") || str.startsWith("lnurlp://");
|
|
69
|
-
}
|
|
70
|
-
catch (e) { }
|
|
71
|
-
return false;
|
|
72
|
-
}
|
|
73
|
-
/**
|
|
74
|
-
* Checks if the provided string is a lightning network address (e.g. satoshi@nakamoto.com)
|
|
75
|
-
* @param str
|
|
76
|
-
* @private
|
|
77
|
-
*/
|
|
78
|
-
static isLightningAddress(str) {
|
|
79
|
-
return MAIL_REGEX.test(str);
|
|
80
|
-
}
|
|
81
|
-
/**
|
|
82
|
-
* Checks whether a given string is a LNURL or lightning address
|
|
83
|
-
* @param str
|
|
84
|
-
*/
|
|
85
|
-
static isLNURL(str) {
|
|
86
|
-
return LNURL.isBech32LNURL(str) || LNURL.isLightningAddress(str) || LNURL.isBareLNURL(str);
|
|
87
|
-
}
|
|
88
|
-
/**
|
|
89
|
-
* Extracts the URL that needs to be request from LNURL or lightning address
|
|
90
|
-
* @param str
|
|
91
|
-
* @private
|
|
92
|
-
* @returns An URL to send the request to, or null if it cannot be parsed
|
|
93
|
-
*/
|
|
94
|
-
static extractCallUrl(str) {
|
|
95
|
-
if (MAIL_REGEX.test(str)) {
|
|
96
|
-
//lightning e-mail like address
|
|
97
|
-
const arr = str.split("@");
|
|
98
|
-
const username = arr[0];
|
|
99
|
-
const domain = arr[1];
|
|
100
|
-
let scheme = "https";
|
|
101
|
-
if (domain.endsWith(".onion")) {
|
|
102
|
-
scheme = "http";
|
|
103
|
-
}
|
|
104
|
-
return scheme + "://" + domain + "/.well-known/lnurlp/" + username;
|
|
105
|
-
}
|
|
106
|
-
else if (LNURL.isBareLNURL(str)) {
|
|
107
|
-
//non-bech32m encoded lnurl
|
|
108
|
-
const data = str.substring("lnurlw://".length);
|
|
109
|
-
const httpUrl = new URL("http://" + data);
|
|
110
|
-
let scheme = "https";
|
|
111
|
-
if (httpUrl.hostname.endsWith(".onion")) {
|
|
112
|
-
scheme = "http";
|
|
113
|
-
}
|
|
114
|
-
return scheme + "://" + data;
|
|
115
|
-
}
|
|
116
|
-
else {
|
|
117
|
-
const lnurl = LNURL.findBech32LNURL(str);
|
|
118
|
-
if (lnurl != null) {
|
|
119
|
-
let { prefix: hrp, words: dataPart } = base_1.bech32.decode(lnurl, 2000);
|
|
120
|
-
let requestByteArray = base_1.bech32.fromWords(dataPart);
|
|
121
|
-
return buffer_1.Buffer.from(requestByteArray).toString();
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
return null;
|
|
125
|
-
}
|
|
126
|
-
/**
|
|
127
|
-
* Sends a request to obtain data about a specific LNURL or lightning address
|
|
128
|
-
*
|
|
129
|
-
* @param str A lnurl or lightning address
|
|
130
|
-
* @param shouldRetry Whether we should retry in case of network failure
|
|
131
|
-
* @param timeout Request timeout in milliseconds
|
|
132
|
-
* @param abortSignal
|
|
133
|
-
*/
|
|
134
|
-
static async getLNURL(str, shouldRetry = true, timeout, abortSignal) {
|
|
135
|
-
if (shouldRetry == null)
|
|
136
|
-
shouldRetry = true;
|
|
137
|
-
const url = LNURL.extractCallUrl(str);
|
|
138
|
-
if (url == null)
|
|
139
|
-
return null;
|
|
140
|
-
const sendRequest = () => (0, HttpUtils_1.httpGet)(url, timeout, abortSignal, true);
|
|
141
|
-
let response = shouldRetry ?
|
|
142
|
-
await (0, RetryUtils_1.tryWithRetries)(sendRequest, undefined, RequestError_1.RequestError, abortSignal) :
|
|
143
|
-
await sendRequest();
|
|
144
|
-
if (isLNURLError(response))
|
|
145
|
-
return null;
|
|
146
|
-
if (response.tag === "payRequest")
|
|
147
|
-
try {
|
|
148
|
-
response.decodedMetadata = JSON.parse(response.metadata);
|
|
149
|
-
}
|
|
150
|
-
catch (err) {
|
|
151
|
-
response.decodedMetadata = [];
|
|
152
|
-
}
|
|
153
|
-
if (!(0, LNURLPay_1.isLNURLPayParams)(response) && !(0, LNURLWithdraw_1.isLNURLWithdrawParams)(response))
|
|
154
|
-
return null;
|
|
155
|
-
return {
|
|
156
|
-
...response,
|
|
157
|
-
url: str
|
|
158
|
-
};
|
|
159
|
-
}
|
|
160
|
-
/**
|
|
161
|
-
* Sends a request to obtain data about a specific LNURL or lightning address
|
|
162
|
-
*
|
|
163
|
-
* @param str A lnurl or lightning address
|
|
164
|
-
* @param shouldRetry Whether we should retry in case of network failure
|
|
165
|
-
* @param timeout Request timeout in milliseconds
|
|
166
|
-
* @param abortSignal
|
|
167
|
-
*/
|
|
168
|
-
static async getLNURLType(str, shouldRetry, timeout, abortSignal) {
|
|
169
|
-
let res = await LNURL.getLNURL(str, shouldRetry, timeout, abortSignal);
|
|
170
|
-
if (res.tag === "payRequest") {
|
|
171
|
-
const payRequest = res;
|
|
172
|
-
let shortDescription = undefined;
|
|
173
|
-
let longDescription = undefined;
|
|
174
|
-
let icon = undefined;
|
|
175
|
-
payRequest.decodedMetadata.forEach(data => {
|
|
176
|
-
switch (data[0]) {
|
|
177
|
-
case "text/plain":
|
|
178
|
-
shortDescription = data[1];
|
|
179
|
-
break;
|
|
180
|
-
case "text/long-desc":
|
|
181
|
-
longDescription = data[1];
|
|
182
|
-
break;
|
|
183
|
-
case "image/png;base64":
|
|
184
|
-
icon = "data:" + data[0] + "," + data[1];
|
|
185
|
-
break;
|
|
186
|
-
case "image/jpeg;base64":
|
|
187
|
-
icon = "data:" + data[0] + "," + data[1];
|
|
188
|
-
break;
|
|
189
|
-
}
|
|
190
|
-
});
|
|
191
|
-
return {
|
|
192
|
-
type: "pay",
|
|
193
|
-
min: BigInt(payRequest.minSendable) / 1000n,
|
|
194
|
-
max: BigInt(payRequest.maxSendable) / 1000n,
|
|
195
|
-
commentMaxLength: payRequest.commentAllowed || 0,
|
|
196
|
-
shortDescription,
|
|
197
|
-
longDescription,
|
|
198
|
-
icon,
|
|
199
|
-
params: payRequest
|
|
200
|
-
};
|
|
201
|
-
}
|
|
202
|
-
if (res.tag === "withdrawRequest") {
|
|
203
|
-
const payRequest = res;
|
|
204
|
-
return {
|
|
205
|
-
type: "withdraw",
|
|
206
|
-
min: BigInt(payRequest.minWithdrawable) / 1000n,
|
|
207
|
-
max: BigInt(payRequest.maxWithdrawable) / 1000n,
|
|
208
|
-
params: payRequest
|
|
209
|
-
};
|
|
210
|
-
}
|
|
211
|
-
return null;
|
|
212
|
-
}
|
|
213
|
-
/**
|
|
214
|
-
* Uses a LNURL-pay request by obtaining a lightning network invoice from it
|
|
215
|
-
*
|
|
216
|
-
* @param payRequest LNURL params as returned from the getLNURL call
|
|
217
|
-
* @param amount Amount of sats (BTC) to pay
|
|
218
|
-
* @param comment Optional comment for the payment request
|
|
219
|
-
* @param timeout Request timeout in milliseconds
|
|
220
|
-
* @param abortSignal
|
|
221
|
-
* @throws {RequestError} If the response is non-200, status: ERROR, or invalid format
|
|
222
|
-
*/
|
|
223
|
-
static async useLNURLPay(payRequest, amount, comment, timeout, abortSignal) {
|
|
224
|
-
const params = ["amount=" + (amount * 1000n).toString(10)];
|
|
225
|
-
if (comment != null) {
|
|
226
|
-
params.push("comment=" + encodeURIComponent(comment));
|
|
227
|
-
}
|
|
228
|
-
const queryParams = (payRequest.callback.includes("?") ? "&" : "?") + params.join("&");
|
|
229
|
-
const response = await (0, RetryUtils_1.tryWithRetries)(() => (0, HttpUtils_1.httpGet)(payRequest.callback + queryParams, timeout, abortSignal, true), undefined, RequestError_1.RequestError, abortSignal);
|
|
230
|
-
if (isLNURLError(response))
|
|
231
|
-
throw new RequestError_1.RequestError("LNURL callback error: " + response.reason, 200);
|
|
232
|
-
if (!isLNURLPayResult(response))
|
|
233
|
-
throw new RequestError_1.RequestError("Invalid LNURL response!", 200);
|
|
234
|
-
const parsedPR = (0, bolt11_1.decode)(response.pr);
|
|
235
|
-
const descHash = buffer_1.Buffer.from((0, sha2_1.sha256)(payRequest.metadata)).toString("hex");
|
|
236
|
-
if (parsedPR.tagsObject.purpose_commit_hash !== descHash)
|
|
237
|
-
throw new RequestError_1.RequestError("Invalid invoice received (description hash)!", 200);
|
|
238
|
-
const msats = parsedPR.millisatoshis;
|
|
239
|
-
if (msats == null)
|
|
240
|
-
throw new RequestError_1.RequestError("Invalid invoice received (amount msats not defined)", 200);
|
|
241
|
-
const invoiceMSats = BigInt(msats);
|
|
242
|
-
if (invoiceMSats !== (amount * 1000n))
|
|
243
|
-
throw new RequestError_1.RequestError("Invalid invoice received (amount)!", 200);
|
|
244
|
-
return {
|
|
245
|
-
invoice: response.pr,
|
|
246
|
-
parsedInvoice: parsedPR,
|
|
247
|
-
successAction: response.successAction ?? undefined
|
|
248
|
-
};
|
|
249
|
-
}
|
|
250
|
-
/**
|
|
251
|
-
* Submits the bolt11 lightning invoice to the lnurl withdraw url
|
|
252
|
-
*
|
|
253
|
-
* @param withdrawRequest Withdraw request to use
|
|
254
|
-
* @param withdrawRequest.k1 K1 parameter
|
|
255
|
-
* @param withdrawRequest.callback A URL to call
|
|
256
|
-
* @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
|
|
257
|
-
* @throws {RequestError} If the response is non-200 or status: ERROR
|
|
258
|
-
*/
|
|
259
|
-
static async postInvoiceToLNURLWithdraw(withdrawRequest, lnpr) {
|
|
260
|
-
const params = [
|
|
261
|
-
"pr=" + lnpr,
|
|
262
|
-
"k1=" + withdrawRequest.k1
|
|
263
|
-
];
|
|
264
|
-
const queryParams = (withdrawRequest.callback.includes("?") ? "&" : "?") + params.join("&");
|
|
265
|
-
const response = await (0, RetryUtils_1.tryWithRetries)(() => (0, HttpUtils_1.httpGet)(withdrawRequest.callback + queryParams, undefined, undefined, true), undefined, RequestError_1.RequestError);
|
|
266
|
-
if (isLNURLError(response))
|
|
267
|
-
throw new RequestError_1.RequestError("LNURL callback error: " + response.reason, 200);
|
|
268
|
-
}
|
|
269
|
-
/**
|
|
270
|
-
* Uses a LNURL-withdraw request by submitting a lightning network invoice to it
|
|
271
|
-
*
|
|
272
|
-
* @param withdrawRequest Withdrawal request as returned from getLNURL call
|
|
273
|
-
* @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
|
|
274
|
-
* @throws {UserError} In case the provided bolt11 lightning invoice has an amount that is out of bounds for
|
|
275
|
-
* the specified LNURL-withdraw request
|
|
276
|
-
*/
|
|
277
|
-
static async useLNURLWithdraw(withdrawRequest, lnpr) {
|
|
278
|
-
const min = BigInt(withdrawRequest.minWithdrawable) / 1000n;
|
|
279
|
-
const max = BigInt(withdrawRequest.maxWithdrawable) / 1000n;
|
|
280
|
-
const parsedPR = (0, bolt11_1.decode)(lnpr);
|
|
281
|
-
const msats = parsedPR.millisatoshis;
|
|
282
|
-
if (msats == null)
|
|
283
|
-
throw new UserError_1.UserError("Invoice without msats value field!");
|
|
284
|
-
const amount = (BigInt(msats) + 999n) / 1000n;
|
|
285
|
-
if (amount < min)
|
|
286
|
-
throw new UserError_1.UserError("Invoice amount less than minimum LNURL-withdraw limit");
|
|
287
|
-
if (amount > max)
|
|
288
|
-
throw new UserError_1.UserError("Invoice amount more than maximum LNURL-withdraw limit");
|
|
289
|
-
return await LNURL.postInvoiceToLNURLWithdraw(withdrawRequest, lnpr);
|
|
290
|
-
}
|
|
291
|
-
static decodeSuccessAction(successAction, secret) {
|
|
292
|
-
if (secret == null)
|
|
293
|
-
return null;
|
|
294
|
-
if (successAction == null)
|
|
295
|
-
return null;
|
|
296
|
-
if (successAction.tag === "message" && successAction.message != null) {
|
|
297
|
-
return {
|
|
298
|
-
description: successAction.message
|
|
299
|
-
};
|
|
300
|
-
}
|
|
301
|
-
if (successAction.tag === "url" && successAction.description != null && successAction.url != null) {
|
|
302
|
-
return {
|
|
303
|
-
description: successAction.description,
|
|
304
|
-
url: successAction.url
|
|
305
|
-
};
|
|
306
|
-
}
|
|
307
|
-
if (successAction.tag === "aes" && successAction.iv != null && successAction.ciphertext != null && successAction.description != null) {
|
|
308
|
-
const CBC = (0, aes_1.cbc)(buffer_1.Buffer.from(secret, "hex"), buffer_1.Buffer.from(successAction.iv, "hex"));
|
|
309
|
-
let plaintext = CBC.decrypt(buffer_1.Buffer.from(successAction.ciphertext, "base64"));
|
|
310
|
-
// remove padding
|
|
311
|
-
const size = plaintext.length;
|
|
312
|
-
const pad = plaintext[size - 1];
|
|
313
|
-
return {
|
|
314
|
-
description: successAction.description,
|
|
315
|
-
text: buffer_1.Buffer.from(plaintext).toString("utf8", 0, size - pad)
|
|
316
|
-
};
|
|
317
|
-
}
|
|
318
|
-
return null;
|
|
319
|
-
}
|
|
320
|
-
}
|
|
321
|
-
exports.LNURL = LNURL;
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.LNURL = exports.isLNURLPaySuccessAction = void 0;
|
|
4
|
+
const RequestError_1 = require("../errors/RequestError");
|
|
5
|
+
const bolt11_1 = require("@atomiqlabs/bolt11");
|
|
6
|
+
const UserError_1 = require("../errors/UserError");
|
|
7
|
+
const base_1 = require("@scure/base");
|
|
8
|
+
const aes_1 = require("@noble/ciphers/aes");
|
|
9
|
+
const buffer_1 = require("buffer");
|
|
10
|
+
const sha2_1 = require("@noble/hashes/sha2");
|
|
11
|
+
const HttpUtils_1 = require("../http/HttpUtils");
|
|
12
|
+
const LNURLWithdraw_1 = require("../types/lnurl/LNURLWithdraw");
|
|
13
|
+
const LNURLPay_1 = require("../types/lnurl/LNURLPay");
|
|
14
|
+
const RetryUtils_1 = require("../utils/RetryUtils");
|
|
15
|
+
function isLNURLError(obj) {
|
|
16
|
+
return obj.status === "ERROR" &&
|
|
17
|
+
(obj.reason == null || typeof obj.reason === "string");
|
|
18
|
+
}
|
|
19
|
+
function isLNURLPayResult(obj, domain) {
|
|
20
|
+
return typeof obj.pr === "string" &&
|
|
21
|
+
(obj.routes == null || Array.isArray(obj.routes)) &&
|
|
22
|
+
(obj.disposable === null || obj.disposable === undefined || typeof obj.disposable === "boolean") &&
|
|
23
|
+
(obj.successAction == null || isLNURLPaySuccessAction(obj.successAction, domain));
|
|
24
|
+
}
|
|
25
|
+
function isLNURLPaySuccessAction(obj, domain) {
|
|
26
|
+
if (obj == null || typeof obj !== 'object' || typeof obj.tag !== 'string')
|
|
27
|
+
return false;
|
|
28
|
+
switch (obj.tag) {
|
|
29
|
+
case "message":
|
|
30
|
+
return obj.message != null && obj.message.length <= 144;
|
|
31
|
+
case "url":
|
|
32
|
+
return obj.description != null && obj.description.length <= 144 &&
|
|
33
|
+
obj.url != null &&
|
|
34
|
+
(domain == null || new URL(obj.url).hostname === domain);
|
|
35
|
+
case "aes":
|
|
36
|
+
return obj.description != null && obj.description.length <= 144 &&
|
|
37
|
+
obj.ciphertext != null && obj.ciphertext.length <= 4096 && BASE64_REGEX.test(obj.ciphertext) &&
|
|
38
|
+
obj.iv != null && obj.iv.length <= 24 && BASE64_REGEX.test(obj.iv);
|
|
39
|
+
default:
|
|
40
|
+
//Unsupported action
|
|
41
|
+
return false;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
exports.isLNURLPaySuccessAction = isLNURLPaySuccessAction;
|
|
45
|
+
const BASE64_REGEX = /^(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{2}==|[A-Za-z0-9+/]{3}=)?$/;
|
|
46
|
+
const MAIL_REGEX = /(?:[A-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[A-z0-9!#$%&'*+/=?^_`{|}~-]+)*|"(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21\x23-\x5b\x5d-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])*")@(?:(?:[A-z0-9](?:[A-z0-9-]*[A-z0-9])?\.)+[A-z0-9](?:[A-z0-9-]*[A-z0-9])?|\[(?:(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9]))\.){3}(?:(2(5[0-5]|[0-4][0-9])|1[0-9][0-9]|[1-9]?[0-9])|[A-z0-9-]*[A-z0-9]:(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21-\x5a\x53-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])+)\])/;
|
|
47
|
+
/**
|
|
48
|
+
* LNURL encoding/decoding and callback handling
|
|
49
|
+
* @category Bitcoin
|
|
50
|
+
*/
|
|
51
|
+
class LNURL {
|
|
52
|
+
static findBech32LNURL(str) {
|
|
53
|
+
const arr = /,*?((lnurl)([0-9]{1,}[a-z0-9]+){1})/.exec(str.toLowerCase());
|
|
54
|
+
if (arr == null)
|
|
55
|
+
return null;
|
|
56
|
+
return arr[1];
|
|
57
|
+
}
|
|
58
|
+
static isBech32LNURL(str) {
|
|
59
|
+
return this.findBech32LNURL(str) != null;
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Checks whether a provided string is bare (non bech32 encoded) lnurl
|
|
63
|
+
* @param str
|
|
64
|
+
* @private
|
|
65
|
+
*/
|
|
66
|
+
static isBareLNURL(str) {
|
|
67
|
+
try {
|
|
68
|
+
return str.startsWith("lnurlw://") || str.startsWith("lnurlp://");
|
|
69
|
+
}
|
|
70
|
+
catch (e) { }
|
|
71
|
+
return false;
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Checks if the provided string is a lightning network address (e.g. satoshi@nakamoto.com)
|
|
75
|
+
* @param str
|
|
76
|
+
* @private
|
|
77
|
+
*/
|
|
78
|
+
static isLightningAddress(str) {
|
|
79
|
+
return MAIL_REGEX.test(str);
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Checks whether a given string is a LNURL or lightning address
|
|
83
|
+
* @param str
|
|
84
|
+
*/
|
|
85
|
+
static isLNURL(str) {
|
|
86
|
+
return LNURL.isBech32LNURL(str) || LNURL.isLightningAddress(str) || LNURL.isBareLNURL(str);
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Extracts the URL that needs to be request from LNURL or lightning address
|
|
90
|
+
* @param str
|
|
91
|
+
* @private
|
|
92
|
+
* @returns An URL to send the request to, or null if it cannot be parsed
|
|
93
|
+
*/
|
|
94
|
+
static extractCallUrl(str) {
|
|
95
|
+
if (MAIL_REGEX.test(str)) {
|
|
96
|
+
//lightning e-mail like address
|
|
97
|
+
const arr = str.split("@");
|
|
98
|
+
const username = arr[0];
|
|
99
|
+
const domain = arr[1];
|
|
100
|
+
let scheme = "https";
|
|
101
|
+
if (domain.endsWith(".onion")) {
|
|
102
|
+
scheme = "http";
|
|
103
|
+
}
|
|
104
|
+
return scheme + "://" + domain + "/.well-known/lnurlp/" + username;
|
|
105
|
+
}
|
|
106
|
+
else if (LNURL.isBareLNURL(str)) {
|
|
107
|
+
//non-bech32m encoded lnurl
|
|
108
|
+
const data = str.substring("lnurlw://".length);
|
|
109
|
+
const httpUrl = new URL("http://" + data);
|
|
110
|
+
let scheme = "https";
|
|
111
|
+
if (httpUrl.hostname.endsWith(".onion")) {
|
|
112
|
+
scheme = "http";
|
|
113
|
+
}
|
|
114
|
+
return scheme + "://" + data;
|
|
115
|
+
}
|
|
116
|
+
else {
|
|
117
|
+
const lnurl = LNURL.findBech32LNURL(str);
|
|
118
|
+
if (lnurl != null) {
|
|
119
|
+
let { prefix: hrp, words: dataPart } = base_1.bech32.decode(lnurl, 2000);
|
|
120
|
+
let requestByteArray = base_1.bech32.fromWords(dataPart);
|
|
121
|
+
return buffer_1.Buffer.from(requestByteArray).toString();
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
return null;
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Sends a request to obtain data about a specific LNURL or lightning address
|
|
128
|
+
*
|
|
129
|
+
* @param str A lnurl or lightning address
|
|
130
|
+
* @param shouldRetry Whether we should retry in case of network failure
|
|
131
|
+
* @param timeout Request timeout in milliseconds
|
|
132
|
+
* @param abortSignal
|
|
133
|
+
*/
|
|
134
|
+
static async getLNURL(str, shouldRetry = true, timeout, abortSignal) {
|
|
135
|
+
if (shouldRetry == null)
|
|
136
|
+
shouldRetry = true;
|
|
137
|
+
const url = LNURL.extractCallUrl(str);
|
|
138
|
+
if (url == null)
|
|
139
|
+
return null;
|
|
140
|
+
const sendRequest = () => (0, HttpUtils_1.httpGet)(url, timeout, abortSignal, true);
|
|
141
|
+
let response = shouldRetry ?
|
|
142
|
+
await (0, RetryUtils_1.tryWithRetries)(sendRequest, undefined, RequestError_1.RequestError, abortSignal) :
|
|
143
|
+
await sendRequest();
|
|
144
|
+
if (isLNURLError(response))
|
|
145
|
+
return null;
|
|
146
|
+
if (response.tag === "payRequest")
|
|
147
|
+
try {
|
|
148
|
+
response.decodedMetadata = JSON.parse(response.metadata);
|
|
149
|
+
}
|
|
150
|
+
catch (err) {
|
|
151
|
+
response.decodedMetadata = [];
|
|
152
|
+
}
|
|
153
|
+
if (!(0, LNURLPay_1.isLNURLPayParams)(response) && !(0, LNURLWithdraw_1.isLNURLWithdrawParams)(response))
|
|
154
|
+
return null;
|
|
155
|
+
return {
|
|
156
|
+
...response,
|
|
157
|
+
url: str
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Sends a request to obtain data about a specific LNURL or lightning address
|
|
162
|
+
*
|
|
163
|
+
* @param str A lnurl or lightning address
|
|
164
|
+
* @param shouldRetry Whether we should retry in case of network failure
|
|
165
|
+
* @param timeout Request timeout in milliseconds
|
|
166
|
+
* @param abortSignal
|
|
167
|
+
*/
|
|
168
|
+
static async getLNURLType(str, shouldRetry, timeout, abortSignal) {
|
|
169
|
+
let res = await LNURL.getLNURL(str, shouldRetry, timeout, abortSignal);
|
|
170
|
+
if (res.tag === "payRequest") {
|
|
171
|
+
const payRequest = res;
|
|
172
|
+
let shortDescription = undefined;
|
|
173
|
+
let longDescription = undefined;
|
|
174
|
+
let icon = undefined;
|
|
175
|
+
payRequest.decodedMetadata.forEach(data => {
|
|
176
|
+
switch (data[0]) {
|
|
177
|
+
case "text/plain":
|
|
178
|
+
shortDescription = data[1];
|
|
179
|
+
break;
|
|
180
|
+
case "text/long-desc":
|
|
181
|
+
longDescription = data[1];
|
|
182
|
+
break;
|
|
183
|
+
case "image/png;base64":
|
|
184
|
+
icon = "data:" + data[0] + "," + data[1];
|
|
185
|
+
break;
|
|
186
|
+
case "image/jpeg;base64":
|
|
187
|
+
icon = "data:" + data[0] + "," + data[1];
|
|
188
|
+
break;
|
|
189
|
+
}
|
|
190
|
+
});
|
|
191
|
+
return {
|
|
192
|
+
type: "pay",
|
|
193
|
+
min: BigInt(payRequest.minSendable) / 1000n,
|
|
194
|
+
max: BigInt(payRequest.maxSendable) / 1000n,
|
|
195
|
+
commentMaxLength: payRequest.commentAllowed || 0,
|
|
196
|
+
shortDescription,
|
|
197
|
+
longDescription,
|
|
198
|
+
icon,
|
|
199
|
+
params: payRequest
|
|
200
|
+
};
|
|
201
|
+
}
|
|
202
|
+
if (res.tag === "withdrawRequest") {
|
|
203
|
+
const payRequest = res;
|
|
204
|
+
return {
|
|
205
|
+
type: "withdraw",
|
|
206
|
+
min: BigInt(payRequest.minWithdrawable) / 1000n,
|
|
207
|
+
max: BigInt(payRequest.maxWithdrawable) / 1000n,
|
|
208
|
+
params: payRequest
|
|
209
|
+
};
|
|
210
|
+
}
|
|
211
|
+
return null;
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* Uses a LNURL-pay request by obtaining a lightning network invoice from it
|
|
215
|
+
*
|
|
216
|
+
* @param payRequest LNURL params as returned from the getLNURL call
|
|
217
|
+
* @param amount Amount of sats (BTC) to pay
|
|
218
|
+
* @param comment Optional comment for the payment request
|
|
219
|
+
* @param timeout Request timeout in milliseconds
|
|
220
|
+
* @param abortSignal
|
|
221
|
+
* @throws {RequestError} If the response is non-200, status: ERROR, or invalid format
|
|
222
|
+
*/
|
|
223
|
+
static async useLNURLPay(payRequest, amount, comment, timeout, abortSignal) {
|
|
224
|
+
const params = ["amount=" + (amount * 1000n).toString(10)];
|
|
225
|
+
if (comment != null) {
|
|
226
|
+
params.push("comment=" + encodeURIComponent(comment));
|
|
227
|
+
}
|
|
228
|
+
const queryParams = (payRequest.callback.includes("?") ? "&" : "?") + params.join("&");
|
|
229
|
+
const response = await (0, RetryUtils_1.tryWithRetries)(() => (0, HttpUtils_1.httpGet)(payRequest.callback + queryParams, timeout, abortSignal, true), undefined, RequestError_1.RequestError, abortSignal);
|
|
230
|
+
if (isLNURLError(response))
|
|
231
|
+
throw new RequestError_1.RequestError("LNURL callback error: " + response.reason, 200);
|
|
232
|
+
if (!isLNURLPayResult(response))
|
|
233
|
+
throw new RequestError_1.RequestError("Invalid LNURL response!", 200);
|
|
234
|
+
const parsedPR = (0, bolt11_1.decode)(response.pr);
|
|
235
|
+
const descHash = buffer_1.Buffer.from((0, sha2_1.sha256)(payRequest.metadata)).toString("hex");
|
|
236
|
+
if (parsedPR.tagsObject.purpose_commit_hash !== descHash)
|
|
237
|
+
throw new RequestError_1.RequestError("Invalid invoice received (description hash)!", 200);
|
|
238
|
+
const msats = parsedPR.millisatoshis;
|
|
239
|
+
if (msats == null)
|
|
240
|
+
throw new RequestError_1.RequestError("Invalid invoice received (amount msats not defined)", 200);
|
|
241
|
+
const invoiceMSats = BigInt(msats);
|
|
242
|
+
if (invoiceMSats !== (amount * 1000n))
|
|
243
|
+
throw new RequestError_1.RequestError("Invalid invoice received (amount)!", 200);
|
|
244
|
+
return {
|
|
245
|
+
invoice: response.pr,
|
|
246
|
+
parsedInvoice: parsedPR,
|
|
247
|
+
successAction: response.successAction ?? undefined
|
|
248
|
+
};
|
|
249
|
+
}
|
|
250
|
+
/**
|
|
251
|
+
* Submits the bolt11 lightning invoice to the lnurl withdraw url
|
|
252
|
+
*
|
|
253
|
+
* @param withdrawRequest Withdraw request to use
|
|
254
|
+
* @param withdrawRequest.k1 K1 parameter
|
|
255
|
+
* @param withdrawRequest.callback A URL to call
|
|
256
|
+
* @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
|
|
257
|
+
* @throws {RequestError} If the response is non-200 or status: ERROR
|
|
258
|
+
*/
|
|
259
|
+
static async postInvoiceToLNURLWithdraw(withdrawRequest, lnpr) {
|
|
260
|
+
const params = [
|
|
261
|
+
"pr=" + lnpr,
|
|
262
|
+
"k1=" + withdrawRequest.k1
|
|
263
|
+
];
|
|
264
|
+
const queryParams = (withdrawRequest.callback.includes("?") ? "&" : "?") + params.join("&");
|
|
265
|
+
const response = await (0, RetryUtils_1.tryWithRetries)(() => (0, HttpUtils_1.httpGet)(withdrawRequest.callback + queryParams, undefined, undefined, true), undefined, RequestError_1.RequestError);
|
|
266
|
+
if (isLNURLError(response))
|
|
267
|
+
throw new RequestError_1.RequestError("LNURL callback error: " + response.reason, 200);
|
|
268
|
+
}
|
|
269
|
+
/**
|
|
270
|
+
* Uses a LNURL-withdraw request by submitting a lightning network invoice to it
|
|
271
|
+
*
|
|
272
|
+
* @param withdrawRequest Withdrawal request as returned from getLNURL call
|
|
273
|
+
* @param lnpr bolt11 lightning network invoice to submit to the withdrawal endpoint
|
|
274
|
+
* @throws {UserError} In case the provided bolt11 lightning invoice has an amount that is out of bounds for
|
|
275
|
+
* the specified LNURL-withdraw request
|
|
276
|
+
*/
|
|
277
|
+
static async useLNURLWithdraw(withdrawRequest, lnpr) {
|
|
278
|
+
const min = BigInt(withdrawRequest.minWithdrawable) / 1000n;
|
|
279
|
+
const max = BigInt(withdrawRequest.maxWithdrawable) / 1000n;
|
|
280
|
+
const parsedPR = (0, bolt11_1.decode)(lnpr);
|
|
281
|
+
const msats = parsedPR.millisatoshis;
|
|
282
|
+
if (msats == null)
|
|
283
|
+
throw new UserError_1.UserError("Invoice without msats value field!");
|
|
284
|
+
const amount = (BigInt(msats) + 999n) / 1000n;
|
|
285
|
+
if (amount < min)
|
|
286
|
+
throw new UserError_1.UserError("Invoice amount less than minimum LNURL-withdraw limit");
|
|
287
|
+
if (amount > max)
|
|
288
|
+
throw new UserError_1.UserError("Invoice amount more than maximum LNURL-withdraw limit");
|
|
289
|
+
return await LNURL.postInvoiceToLNURLWithdraw(withdrawRequest, lnpr);
|
|
290
|
+
}
|
|
291
|
+
static decodeSuccessAction(successAction, secret) {
|
|
292
|
+
if (secret == null)
|
|
293
|
+
return null;
|
|
294
|
+
if (successAction == null)
|
|
295
|
+
return null;
|
|
296
|
+
if (successAction.tag === "message" && successAction.message != null) {
|
|
297
|
+
return {
|
|
298
|
+
description: successAction.message
|
|
299
|
+
};
|
|
300
|
+
}
|
|
301
|
+
if (successAction.tag === "url" && successAction.description != null && successAction.url != null) {
|
|
302
|
+
return {
|
|
303
|
+
description: successAction.description,
|
|
304
|
+
url: successAction.url
|
|
305
|
+
};
|
|
306
|
+
}
|
|
307
|
+
if (successAction.tag === "aes" && successAction.iv != null && successAction.ciphertext != null && successAction.description != null) {
|
|
308
|
+
const CBC = (0, aes_1.cbc)(buffer_1.Buffer.from(secret, "hex"), buffer_1.Buffer.from(successAction.iv, "hex"));
|
|
309
|
+
let plaintext = CBC.decrypt(buffer_1.Buffer.from(successAction.ciphertext, "base64"));
|
|
310
|
+
// remove padding
|
|
311
|
+
const size = plaintext.length;
|
|
312
|
+
const pad = plaintext[size - 1];
|
|
313
|
+
return {
|
|
314
|
+
description: successAction.description,
|
|
315
|
+
text: buffer_1.Buffer.from(plaintext).toString("utf8", 0, size - pad)
|
|
316
|
+
};
|
|
317
|
+
}
|
|
318
|
+
return null;
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
exports.LNURL = LNURL;
|