@atomiqlabs/lp-lib 14.0.0-dev.21 → 14.0.0-dev.23
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +201 -201
- package/dist/fees/IBtcFeeEstimator.d.ts +3 -3
- package/dist/fees/IBtcFeeEstimator.js +2 -2
- package/dist/index.d.ts +42 -42
- package/dist/index.js +58 -58
- package/dist/info/InfoHandler.d.ts +17 -17
- package/dist/info/InfoHandler.js +61 -61
- package/dist/plugins/IPlugin.d.ts +143 -143
- package/dist/plugins/IPlugin.js +34 -34
- package/dist/plugins/PluginManager.d.ts +112 -112
- package/dist/plugins/PluginManager.js +259 -259
- package/dist/prices/BinanceSwapPrice.d.ts +26 -26
- package/dist/prices/BinanceSwapPrice.js +92 -92
- package/dist/prices/CoinGeckoSwapPrice.d.ts +30 -30
- package/dist/prices/CoinGeckoSwapPrice.js +64 -64
- package/dist/prices/ISwapPrice.d.ts +43 -43
- package/dist/prices/ISwapPrice.js +55 -55
- package/dist/prices/OKXSwapPrice.d.ts +26 -26
- package/dist/prices/OKXSwapPrice.js +92 -92
- package/dist/storage/IIntermediaryStorage.d.ts +18 -18
- package/dist/storage/IIntermediaryStorage.js +2 -2
- package/dist/storagemanager/IntermediaryStorageManager.d.ts +19 -19
- package/dist/storagemanager/IntermediaryStorageManager.js +111 -111
- package/dist/storagemanager/StorageManager.d.ts +13 -13
- package/dist/storagemanager/StorageManager.js +64 -64
- package/dist/swaps/SwapHandler.d.ts +153 -153
- package/dist/swaps/SwapHandler.js +160 -160
- package/dist/swaps/SwapHandlerSwap.d.ts +79 -79
- package/dist/swaps/SwapHandlerSwap.js +78 -78
- package/dist/swaps/assertions/AmountAssertions.d.ts +28 -28
- package/dist/swaps/assertions/AmountAssertions.js +72 -72
- package/dist/swaps/assertions/FromBtcAmountAssertions.d.ts +76 -76
- package/dist/swaps/assertions/FromBtcAmountAssertions.js +180 -180
- package/dist/swaps/assertions/LightningAssertions.d.ts +44 -44
- package/dist/swaps/assertions/LightningAssertions.js +86 -86
- package/dist/swaps/assertions/ToBtcAmountAssertions.d.ts +53 -53
- package/dist/swaps/assertions/ToBtcAmountAssertions.js +150 -150
- package/dist/swaps/escrow/EscrowHandler.d.ts +51 -51
- package/dist/swaps/escrow/EscrowHandler.js +158 -158
- package/dist/swaps/escrow/EscrowHandlerSwap.d.ts +35 -35
- package/dist/swaps/escrow/EscrowHandlerSwap.js +69 -69
- package/dist/swaps/escrow/FromBtcBaseSwap.d.ts +14 -14
- package/dist/swaps/escrow/FromBtcBaseSwap.js +32 -32
- package/dist/swaps/escrow/FromBtcBaseSwapHandler.d.ts +102 -102
- package/dist/swaps/escrow/FromBtcBaseSwapHandler.js +210 -210
- package/dist/swaps/escrow/ToBtcBaseSwap.d.ts +36 -36
- package/dist/swaps/escrow/ToBtcBaseSwap.js +67 -67
- package/dist/swaps/escrow/ToBtcBaseSwapHandler.d.ts +53 -53
- package/dist/swaps/escrow/ToBtcBaseSwapHandler.js +81 -81
- package/dist/swaps/escrow/frombtc_abstract/FromBtcAbs.d.ts +83 -83
- package/dist/swaps/escrow/frombtc_abstract/FromBtcAbs.js +318 -318
- package/dist/swaps/escrow/frombtc_abstract/FromBtcSwapAbs.d.ts +21 -21
- package/dist/swaps/escrow/frombtc_abstract/FromBtcSwapAbs.js +50 -50
- package/dist/swaps/escrow/frombtcln_abstract/FromBtcLnAbs.d.ts +107 -107
- package/dist/swaps/escrow/frombtcln_abstract/FromBtcLnAbs.js +675 -675
- package/dist/swaps/escrow/frombtcln_abstract/FromBtcLnSwapAbs.d.ts +33 -33
- package/dist/swaps/escrow/frombtcln_abstract/FromBtcLnSwapAbs.js +91 -91
- package/dist/swaps/escrow/frombtcln_autoinit/FromBtcLnAuto.d.ts +111 -111
- package/dist/swaps/escrow/frombtcln_autoinit/FromBtcLnAuto.js +684 -684
- package/dist/swaps/escrow/frombtcln_autoinit/FromBtcLnAutoSwap.d.ts +55 -55
- package/dist/swaps/escrow/frombtcln_autoinit/FromBtcLnAutoSwap.js +120 -120
- package/dist/swaps/escrow/tobtc_abstract/ToBtcAbs.d.ts +171 -171
- package/dist/swaps/escrow/tobtc_abstract/ToBtcAbs.js +706 -706
- package/dist/swaps/escrow/tobtc_abstract/ToBtcSwapAbs.d.ts +26 -26
- package/dist/swaps/escrow/tobtc_abstract/ToBtcSwapAbs.js +62 -62
- package/dist/swaps/escrow/tobtcln_abstract/ToBtcLnAbs.d.ts +177 -177
- package/dist/swaps/escrow/tobtcln_abstract/ToBtcLnAbs.js +863 -863
- package/dist/swaps/escrow/tobtcln_abstract/ToBtcLnSwapAbs.d.ts +24 -24
- package/dist/swaps/escrow/tobtcln_abstract/ToBtcLnSwapAbs.js +58 -58
- package/dist/swaps/spv_vault_swap/SpvVault.d.ts +40 -41
- package/dist/swaps/spv_vault_swap/SpvVault.js +111 -111
- package/dist/swaps/spv_vault_swap/SpvVaultSwap.d.ts +67 -67
- package/dist/swaps/spv_vault_swap/SpvVaultSwap.js +158 -158
- package/dist/swaps/spv_vault_swap/SpvVaultSwapHandler.d.ts +68 -68
- package/dist/swaps/spv_vault_swap/SpvVaultSwapHandler.js +492 -492
- package/dist/swaps/spv_vault_swap/SpvVaults.d.ts +52 -52
- package/dist/swaps/spv_vault_swap/SpvVaults.js +386 -364
- package/dist/swaps/trusted/frombtc_trusted/FromBtcTrusted.d.ts +51 -51
- package/dist/swaps/trusted/frombtc_trusted/FromBtcTrusted.js +650 -650
- package/dist/swaps/trusted/frombtc_trusted/FromBtcTrustedSwap.d.ts +52 -52
- package/dist/swaps/trusted/frombtc_trusted/FromBtcTrustedSwap.js +118 -118
- package/dist/swaps/trusted/frombtcln_trusted/FromBtcLnTrusted.d.ts +76 -76
- package/dist/swaps/trusted/frombtcln_trusted/FromBtcLnTrusted.js +494 -494
- package/dist/swaps/trusted/frombtcln_trusted/FromBtcLnTrustedSwap.d.ts +34 -34
- package/dist/swaps/trusted/frombtcln_trusted/FromBtcLnTrustedSwap.js +81 -81
- package/dist/utils/BitcoinUtils.d.ts +2 -2
- package/dist/utils/BitcoinUtils.js +45 -45
- package/dist/utils/Utils.d.ts +29 -29
- package/dist/utils/Utils.js +89 -89
- package/dist/utils/paramcoders/IParamReader.d.ts +5 -5
- package/dist/utils/paramcoders/IParamReader.js +2 -2
- package/dist/utils/paramcoders/IParamWriter.d.ts +4 -4
- package/dist/utils/paramcoders/IParamWriter.js +2 -2
- package/dist/utils/paramcoders/LegacyParamEncoder.d.ts +10 -10
- package/dist/utils/paramcoders/LegacyParamEncoder.js +22 -22
- package/dist/utils/paramcoders/ParamDecoder.d.ts +25 -25
- package/dist/utils/paramcoders/ParamDecoder.js +222 -222
- package/dist/utils/paramcoders/ParamEncoder.d.ts +9 -9
- package/dist/utils/paramcoders/ParamEncoder.js +22 -22
- package/dist/utils/paramcoders/SchemaVerifier.d.ts +21 -21
- package/dist/utils/paramcoders/SchemaVerifier.js +84 -84
- package/dist/utils/paramcoders/server/ServerParamDecoder.d.ts +8 -8
- package/dist/utils/paramcoders/server/ServerParamDecoder.js +107 -107
- package/dist/utils/paramcoders/server/ServerParamEncoder.d.ts +11 -11
- package/dist/utils/paramcoders/server/ServerParamEncoder.js +65 -65
- package/dist/wallets/IBitcoinWallet.d.ts +67 -67
- package/dist/wallets/IBitcoinWallet.js +2 -2
- package/dist/wallets/ILightningWallet.d.ts +117 -117
- package/dist/wallets/ILightningWallet.js +37 -37
- package/dist/wallets/ISpvVaultSigner.d.ts +7 -7
- package/dist/wallets/ISpvVaultSigner.js +2 -2
- package/package.json +36 -36
- package/src/fees/IBtcFeeEstimator.ts +6 -6
- package/src/index.ts +53 -53
- package/src/info/InfoHandler.ts +106 -106
- package/src/plugins/IPlugin.ts +168 -168
- package/src/plugins/PluginManager.ts +336 -336
- package/src/prices/BinanceSwapPrice.ts +113 -113
- package/src/prices/CoinGeckoSwapPrice.ts +87 -87
- package/src/prices/ISwapPrice.ts +88 -88
- package/src/prices/OKXSwapPrice.ts +113 -113
- package/src/storage/IIntermediaryStorage.ts +19 -19
- package/src/storagemanager/IntermediaryStorageManager.ts +118 -118
- package/src/storagemanager/StorageManager.ts +78 -78
- package/src/swaps/SwapHandler.ts +277 -277
- package/src/swaps/SwapHandlerSwap.ts +141 -141
- package/src/swaps/assertions/AmountAssertions.ts +76 -76
- package/src/swaps/assertions/FromBtcAmountAssertions.ts +246 -246
- package/src/swaps/assertions/LightningAssertions.ts +103 -103
- package/src/swaps/assertions/ToBtcAmountAssertions.ts +203 -203
- package/src/swaps/escrow/EscrowHandler.ts +179 -179
- package/src/swaps/escrow/EscrowHandlerSwap.ts +86 -86
- package/src/swaps/escrow/FromBtcBaseSwap.ts +38 -38
- package/src/swaps/escrow/FromBtcBaseSwapHandler.ts +286 -286
- package/src/swaps/escrow/ToBtcBaseSwap.ts +85 -85
- package/src/swaps/escrow/ToBtcBaseSwapHandler.ts +129 -129
- package/src/swaps/escrow/frombtc_abstract/FromBtcAbs.ts +452 -452
- package/src/swaps/escrow/frombtc_abstract/FromBtcSwapAbs.ts +61 -61
- package/src/swaps/escrow/frombtcln_abstract/FromBtcLnAbs.ts +856 -856
- package/src/swaps/escrow/frombtcln_abstract/FromBtcLnSwapAbs.ts +141 -141
- package/src/swaps/escrow/frombtcln_autoinit/FromBtcLnAuto.ts +850 -850
- package/src/swaps/escrow/frombtcln_autoinit/FromBtcLnAutoSwap.ts +196 -196
- package/src/swaps/escrow/tobtc_abstract/ToBtcAbs.ts +879 -879
- package/src/swaps/escrow/tobtc_abstract/ToBtcSwapAbs.ts +102 -102
- package/src/swaps/escrow/tobtcln_abstract/ToBtcLnAbs.ts +1112 -1112
- package/src/swaps/escrow/tobtcln_abstract/ToBtcLnSwapAbs.ts +80 -80
- package/src/swaps/spv_vault_swap/SpvVault.ts +143 -143
- package/src/swaps/spv_vault_swap/SpvVaultSwap.ts +225 -225
- package/src/swaps/spv_vault_swap/SpvVaultSwapHandler.ts +628 -628
- package/src/swaps/spv_vault_swap/SpvVaults.ts +458 -435
- package/src/swaps/trusted/frombtc_trusted/FromBtcTrusted.ts +747 -747
- package/src/swaps/trusted/frombtc_trusted/FromBtcTrustedSwap.ts +185 -185
- package/src/swaps/trusted/frombtcln_trusted/FromBtcLnTrusted.ts +590 -590
- package/src/swaps/trusted/frombtcln_trusted/FromBtcLnTrustedSwap.ts +121 -121
- package/src/utils/BitcoinUtils.ts +42 -42
- package/src/utils/Utils.ts +104 -104
- package/src/utils/paramcoders/IParamReader.ts +7 -7
- package/src/utils/paramcoders/IParamWriter.ts +8 -8
- package/src/utils/paramcoders/LegacyParamEncoder.ts +27 -27
- package/src/utils/paramcoders/ParamDecoder.ts +218 -218
- package/src/utils/paramcoders/ParamEncoder.ts +29 -29
- package/src/utils/paramcoders/SchemaVerifier.ts +96 -96
- package/src/utils/paramcoders/server/ServerParamDecoder.ts +118 -118
- package/src/utils/paramcoders/server/ServerParamEncoder.ts +75 -75
- package/src/wallets/IBitcoinWallet.ts +68 -68
- package/src/wallets/ILightningWallet.ts +178 -178
- package/src/wallets/ISpvVaultSigner.ts +10 -10
package/src/swaps/SwapHandler.ts
CHANGED
|
@@ -1,277 +1,277 @@
|
|
|
1
|
-
import {Express, Request} from "express";
|
|
2
|
-
import {ISwapPrice} from "../prices/ISwapPrice";
|
|
3
|
-
import {
|
|
4
|
-
ChainSwapType,
|
|
5
|
-
ChainType
|
|
6
|
-
} from "@atomiqlabs/base";
|
|
7
|
-
import {SwapHandlerSwap} from "./SwapHandlerSwap";
|
|
8
|
-
import {PluginManager} from "../plugins/PluginManager";
|
|
9
|
-
import {IIntermediaryStorage} from "../storage/IIntermediaryStorage";
|
|
10
|
-
import {IParamReader} from "../utils/paramcoders/IParamReader";
|
|
11
|
-
import {getLogger, LoggerType} from "../utils/Utils";
|
|
12
|
-
|
|
13
|
-
export enum SwapHandlerType {
|
|
14
|
-
TO_BTC = "TO_BTC",
|
|
15
|
-
FROM_BTC = "FROM_BTC",
|
|
16
|
-
TO_BTCLN = "TO_BTCLN",
|
|
17
|
-
FROM_BTCLN = "FROM_BTCLN",
|
|
18
|
-
FROM_BTCLN_TRUSTED = "FROM_BTCLN_TRUSTED",
|
|
19
|
-
FROM_BTC_TRUSTED = "FROM_BTC_TRUSTED",
|
|
20
|
-
FROM_BTC_SPV = "FROM_BTC_SPV",
|
|
21
|
-
FROM_BTCLN_AUTO = "FROM_BTCLN_AUTO"
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
export type SwapHandlerInfoType = {
|
|
25
|
-
swapFeePPM: number,
|
|
26
|
-
swapBaseFee: number,
|
|
27
|
-
min: number,
|
|
28
|
-
max: number,
|
|
29
|
-
tokens: string[],
|
|
30
|
-
chainTokens: {[chainId: string]: string[]};
|
|
31
|
-
data?: any,
|
|
32
|
-
};
|
|
33
|
-
|
|
34
|
-
export type SwapBaseConfig = {
|
|
35
|
-
initAuthorizationTimeout: number,
|
|
36
|
-
initAuthorizationTimeouts?: {
|
|
37
|
-
[chainId: string]: number
|
|
38
|
-
},
|
|
39
|
-
bitcoinBlocktime: bigint,
|
|
40
|
-
baseFee: bigint,
|
|
41
|
-
feePPM: bigint,
|
|
42
|
-
max: bigint,
|
|
43
|
-
min: bigint,
|
|
44
|
-
safetyFactor: bigint,
|
|
45
|
-
swapCheckInterval: number
|
|
46
|
-
};
|
|
47
|
-
|
|
48
|
-
export type MultichainData = {
|
|
49
|
-
chains: {
|
|
50
|
-
[identifier: string]: ChainData
|
|
51
|
-
},
|
|
52
|
-
default: string
|
|
53
|
-
};
|
|
54
|
-
|
|
55
|
-
export type ChainData<T extends ChainType = ChainType> = {
|
|
56
|
-
signer: T["Signer"],
|
|
57
|
-
swapContract: T["Contract"],
|
|
58
|
-
spvVaultContract: T["SpvVaultContract"],
|
|
59
|
-
chainInterface: T["ChainInterface"],
|
|
60
|
-
chainEvents: T["Events"],
|
|
61
|
-
allowedTokens: string[],
|
|
62
|
-
tokenMultipliers?: {[tokenAddress: string]: bigint},
|
|
63
|
-
allowedDepositTokens?: string[],
|
|
64
|
-
btcRelay?: T["BtcRelay"]
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
export type RequestData<T> = {
|
|
68
|
-
chainIdentifier: string,
|
|
69
|
-
raw: Request & {paramReader: IParamReader},
|
|
70
|
-
parsed: T,
|
|
71
|
-
metadata: any
|
|
72
|
-
};
|
|
73
|
-
|
|
74
|
-
/**
|
|
75
|
-
* An abstract class defining a singular swap service
|
|
76
|
-
*/
|
|
77
|
-
export abstract class SwapHandler<V extends SwapHandlerSwap<S> = SwapHandlerSwap, S = any> {
|
|
78
|
-
|
|
79
|
-
abstract readonly type: SwapHandlerType;
|
|
80
|
-
|
|
81
|
-
readonly storageManager: IIntermediaryStorage<V>;
|
|
82
|
-
readonly path: string;
|
|
83
|
-
|
|
84
|
-
readonly chains: MultichainData;
|
|
85
|
-
readonly allowedTokens: {[chainId: string]: Set<string>};
|
|
86
|
-
readonly swapPricing: ISwapPrice;
|
|
87
|
-
|
|
88
|
-
abstract config: SwapBaseConfig;
|
|
89
|
-
|
|
90
|
-
logger: LoggerType = getLogger(() => "SwapHandler("+this.type+"): ");
|
|
91
|
-
|
|
92
|
-
protected swapLogger = {
|
|
93
|
-
debug: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.debug(swap.getIdentifier()+": "+msg, ...args),
|
|
94
|
-
info: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.info(swap.getIdentifier()+": "+msg, ...args),
|
|
95
|
-
warn: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.warn(swap.getIdentifier()+": "+msg, ...args),
|
|
96
|
-
error: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.error(swap.getIdentifier()+": "+msg, ...args)
|
|
97
|
-
};
|
|
98
|
-
|
|
99
|
-
protected constructor(
|
|
100
|
-
storageDirectory: IIntermediaryStorage<V>,
|
|
101
|
-
path: string,
|
|
102
|
-
chainsData: MultichainData,
|
|
103
|
-
swapPricing: ISwapPrice
|
|
104
|
-
) {
|
|
105
|
-
this.storageManager = storageDirectory;
|
|
106
|
-
this.chains = chainsData;
|
|
107
|
-
if(this.chains.chains[this.chains.default]==null) throw new Error("Invalid default chain specified");
|
|
108
|
-
this.path = path;
|
|
109
|
-
this.swapPricing = swapPricing;
|
|
110
|
-
this.allowedTokens = {};
|
|
111
|
-
for(let chainId in chainsData.chains) {
|
|
112
|
-
this.allowedTokens[chainId] = new Set<string>(chainsData.chains[chainId].allowedTokens);
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
protected getDefaultChain(): ChainData {
|
|
117
|
-
return this.chains.chains[this.chains.default];
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
protected getChain(identifier: string): ChainData {
|
|
121
|
-
if(this.chains.chains[identifier]==null)
|
|
122
|
-
throw {
|
|
123
|
-
code: 20200,
|
|
124
|
-
msg: "Invalid chain specified!"
|
|
125
|
-
};
|
|
126
|
-
return this.chains.chains[identifier];
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
protected abstract processPastSwaps(): Promise<void>;
|
|
130
|
-
|
|
131
|
-
/**
|
|
132
|
-
* Starts the watchdog checking past swaps for expiry or claim eligibility.
|
|
133
|
-
*/
|
|
134
|
-
async startWatchdog() {
|
|
135
|
-
let rerun: () => Promise<void>;
|
|
136
|
-
rerun = async () => {
|
|
137
|
-
await this.processPastSwaps().catch( e => this.logger.error("startWatchdog(): Error when processing past swaps: ", e));
|
|
138
|
-
setTimeout(rerun, this.config.swapCheckInterval);
|
|
139
|
-
};
|
|
140
|
-
await rerun();
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
/**
|
|
144
|
-
* Initializes swap handler, loads data and subscribes to chain events
|
|
145
|
-
*/
|
|
146
|
-
abstract init(): Promise<void>;
|
|
147
|
-
|
|
148
|
-
protected async loadData(ctor: new (data: any) => V) {
|
|
149
|
-
await this.storageManager.loadData(ctor);
|
|
150
|
-
//Check if all swaps contain a valid amount
|
|
151
|
-
for(let {obj: swap, hash, sequence} of await this.storageManager.query([])) {
|
|
152
|
-
if(hash!==swap.getIdentifierHash() || sequence !== (swap.getSequence() ?? 0n)) {
|
|
153
|
-
this.swapLogger.info(swap, "loadData(): Swap storage key or sequence mismatch, fixing,"+
|
|
154
|
-
" old hash: "+hash+" new hash: "+swap.getIdentifierHash()+
|
|
155
|
-
" old seq: "+sequence.toString(10)+" new seq: "+(swap.getSequence() ?? 0n).toString(10));
|
|
156
|
-
|
|
157
|
-
await this.storageManager.removeData(hash, sequence);
|
|
158
|
-
await this.storageManager.saveData(swap.getIdentifierHash(), swap.getSequence(), swap);
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
/**
|
|
164
|
-
* Sets up required listeners for the REST server
|
|
165
|
-
*
|
|
166
|
-
* @param restServer
|
|
167
|
-
*/
|
|
168
|
-
abstract startRestServer(restServer: Express): void;
|
|
169
|
-
|
|
170
|
-
/**
|
|
171
|
-
* Returns data to be returned in swap handler info
|
|
172
|
-
*/
|
|
173
|
-
abstract getInfoData(): any;
|
|
174
|
-
|
|
175
|
-
/**
|
|
176
|
-
* Remove swap data
|
|
177
|
-
*
|
|
178
|
-
* @param hash
|
|
179
|
-
* @param sequence
|
|
180
|
-
*/
|
|
181
|
-
protected removeSwapData(hash: string, sequence: bigint): Promise<void>;
|
|
182
|
-
|
|
183
|
-
/**
|
|
184
|
-
* Remove swap data
|
|
185
|
-
*
|
|
186
|
-
* @param swap
|
|
187
|
-
* @param ultimateState set the ultimate state of the swap before removing
|
|
188
|
-
*/
|
|
189
|
-
protected removeSwapData(swap: V, ultimateState?: S): Promise<void>;
|
|
190
|
-
|
|
191
|
-
protected async removeSwapData(hashOrSwap: string | V, sequenceOrUltimateState?: bigint | S) {
|
|
192
|
-
let swap: V;
|
|
193
|
-
if(typeof(hashOrSwap)==="string") {
|
|
194
|
-
if(typeof(sequenceOrUltimateState)!=="bigint") throw new Error("Sequence must be a BN instance!");
|
|
195
|
-
swap = await this.storageManager.getData(hashOrSwap, sequenceOrUltimateState);
|
|
196
|
-
} else {
|
|
197
|
-
swap = hashOrSwap;
|
|
198
|
-
if(sequenceOrUltimateState!=null && typeof(sequenceOrUltimateState)!=="bigint") await swap.setState(sequenceOrUltimateState);
|
|
199
|
-
}
|
|
200
|
-
if(swap!=null) await PluginManager.swapRemove(swap);
|
|
201
|
-
this.swapLogger.debug(swap, "removeSwapData(): removing swap final state: "+swap.state);
|
|
202
|
-
await this.storageManager.removeData(swap.getIdentifierHash(), swap.getSequence());
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
protected async saveSwapData(swap: V) {
|
|
206
|
-
await this.storageManager.saveData(swap.getIdentifierHash(), swap.getSequence(), swap);
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
/**
|
|
210
|
-
* Checks if we have enough balance of the token in the swap vault
|
|
211
|
-
*
|
|
212
|
-
* @param totalInToken
|
|
213
|
-
* @param balancePrefetch
|
|
214
|
-
* @param signal
|
|
215
|
-
* @throws {DefinedRuntimeError} will throw an error if there are not enough funds in the vault
|
|
216
|
-
*/
|
|
217
|
-
protected async checkBalance(totalInToken: bigint, balancePrefetch: Promise<bigint>, signal: AbortSignal | null): Promise<void> {
|
|
218
|
-
const balance = await balancePrefetch;
|
|
219
|
-
if(signal!=null) signal.throwIfAborted();
|
|
220
|
-
|
|
221
|
-
if(balance==null || balance < totalInToken) {
|
|
222
|
-
throw {
|
|
223
|
-
code: 20002,
|
|
224
|
-
msg: "Not enough liquidity"
|
|
225
|
-
};
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
/**
|
|
230
|
-
* Checks if the sequence number is between 0-2^64
|
|
231
|
-
*
|
|
232
|
-
* @param sequence
|
|
233
|
-
* @throws {DefinedRuntimeError} will throw an error if sequence number is out of bounds
|
|
234
|
-
*/
|
|
235
|
-
protected checkSequence(sequence: bigint) {
|
|
236
|
-
if(sequence < 0n || sequence >= 2n ** 64n) {
|
|
237
|
-
throw {
|
|
238
|
-
code: 20060,
|
|
239
|
-
msg: "Invalid sequence"
|
|
240
|
-
};
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
/**
|
|
245
|
-
* Checks whether a given token is supported on a specified chain
|
|
246
|
-
*
|
|
247
|
-
* @param chainId
|
|
248
|
-
* @param token
|
|
249
|
-
* @protected
|
|
250
|
-
*/
|
|
251
|
-
protected isTokenSupported(chainId: string, token: string): boolean {
|
|
252
|
-
const chainTokens = this.allowedTokens[chainId];
|
|
253
|
-
if(chainTokens==null) return false;
|
|
254
|
-
return chainTokens.has(token);
|
|
255
|
-
}
|
|
256
|
-
|
|
257
|
-
getInfo(): SwapHandlerInfoType {
|
|
258
|
-
const chainTokens: {[chainId: string]: string[]} = {};
|
|
259
|
-
for(let chainId in this.allowedTokens) {
|
|
260
|
-
chainTokens[chainId] = Array.from<string>(this.allowedTokens[chainId]);
|
|
261
|
-
}
|
|
262
|
-
return {
|
|
263
|
-
swapFeePPM: Number(this.config.feePPM),
|
|
264
|
-
swapBaseFee: Number(this.config.baseFee),
|
|
265
|
-
min: Number(this.config.min),
|
|
266
|
-
max: Number(this.config.max),
|
|
267
|
-
data: this.getInfoData(),
|
|
268
|
-
tokens: Array.from<string>(this.allowedTokens[this.chains.default]),
|
|
269
|
-
chainTokens
|
|
270
|
-
};
|
|
271
|
-
}
|
|
272
|
-
|
|
273
|
-
protected getInitAuthorizationTimeout(chainIdentifier: string) {
|
|
274
|
-
return this.config.initAuthorizationTimeouts?.[chainIdentifier] ?? this.config.initAuthorizationTimeout;
|
|
275
|
-
}
|
|
276
|
-
|
|
277
|
-
}
|
|
1
|
+
import {Express, Request} from "express";
|
|
2
|
+
import {ISwapPrice} from "../prices/ISwapPrice";
|
|
3
|
+
import {
|
|
4
|
+
ChainSwapType,
|
|
5
|
+
ChainType
|
|
6
|
+
} from "@atomiqlabs/base";
|
|
7
|
+
import {SwapHandlerSwap} from "./SwapHandlerSwap";
|
|
8
|
+
import {PluginManager} from "../plugins/PluginManager";
|
|
9
|
+
import {IIntermediaryStorage} from "../storage/IIntermediaryStorage";
|
|
10
|
+
import {IParamReader} from "../utils/paramcoders/IParamReader";
|
|
11
|
+
import {getLogger, LoggerType} from "../utils/Utils";
|
|
12
|
+
|
|
13
|
+
export enum SwapHandlerType {
|
|
14
|
+
TO_BTC = "TO_BTC",
|
|
15
|
+
FROM_BTC = "FROM_BTC",
|
|
16
|
+
TO_BTCLN = "TO_BTCLN",
|
|
17
|
+
FROM_BTCLN = "FROM_BTCLN",
|
|
18
|
+
FROM_BTCLN_TRUSTED = "FROM_BTCLN_TRUSTED",
|
|
19
|
+
FROM_BTC_TRUSTED = "FROM_BTC_TRUSTED",
|
|
20
|
+
FROM_BTC_SPV = "FROM_BTC_SPV",
|
|
21
|
+
FROM_BTCLN_AUTO = "FROM_BTCLN_AUTO"
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export type SwapHandlerInfoType = {
|
|
25
|
+
swapFeePPM: number,
|
|
26
|
+
swapBaseFee: number,
|
|
27
|
+
min: number,
|
|
28
|
+
max: number,
|
|
29
|
+
tokens: string[],
|
|
30
|
+
chainTokens: {[chainId: string]: string[]};
|
|
31
|
+
data?: any,
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
export type SwapBaseConfig = {
|
|
35
|
+
initAuthorizationTimeout: number,
|
|
36
|
+
initAuthorizationTimeouts?: {
|
|
37
|
+
[chainId: string]: number
|
|
38
|
+
},
|
|
39
|
+
bitcoinBlocktime: bigint,
|
|
40
|
+
baseFee: bigint,
|
|
41
|
+
feePPM: bigint,
|
|
42
|
+
max: bigint,
|
|
43
|
+
min: bigint,
|
|
44
|
+
safetyFactor: bigint,
|
|
45
|
+
swapCheckInterval: number
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
export type MultichainData = {
|
|
49
|
+
chains: {
|
|
50
|
+
[identifier: string]: ChainData
|
|
51
|
+
},
|
|
52
|
+
default: string
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
export type ChainData<T extends ChainType = ChainType> = {
|
|
56
|
+
signer: T["Signer"],
|
|
57
|
+
swapContract: T["Contract"],
|
|
58
|
+
spvVaultContract: T["SpvVaultContract"],
|
|
59
|
+
chainInterface: T["ChainInterface"],
|
|
60
|
+
chainEvents: T["Events"],
|
|
61
|
+
allowedTokens: string[],
|
|
62
|
+
tokenMultipliers?: {[tokenAddress: string]: bigint},
|
|
63
|
+
allowedDepositTokens?: string[],
|
|
64
|
+
btcRelay?: T["BtcRelay"]
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export type RequestData<T> = {
|
|
68
|
+
chainIdentifier: string,
|
|
69
|
+
raw: Request & {paramReader: IParamReader},
|
|
70
|
+
parsed: T,
|
|
71
|
+
metadata: any
|
|
72
|
+
};
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* An abstract class defining a singular swap service
|
|
76
|
+
*/
|
|
77
|
+
export abstract class SwapHandler<V extends SwapHandlerSwap<S> = SwapHandlerSwap, S = any> {
|
|
78
|
+
|
|
79
|
+
abstract readonly type: SwapHandlerType;
|
|
80
|
+
|
|
81
|
+
readonly storageManager: IIntermediaryStorage<V>;
|
|
82
|
+
readonly path: string;
|
|
83
|
+
|
|
84
|
+
readonly chains: MultichainData;
|
|
85
|
+
readonly allowedTokens: {[chainId: string]: Set<string>};
|
|
86
|
+
readonly swapPricing: ISwapPrice;
|
|
87
|
+
|
|
88
|
+
abstract config: SwapBaseConfig;
|
|
89
|
+
|
|
90
|
+
logger: LoggerType = getLogger(() => "SwapHandler("+this.type+"): ");
|
|
91
|
+
|
|
92
|
+
protected swapLogger = {
|
|
93
|
+
debug: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.debug(swap.getIdentifier()+": "+msg, ...args),
|
|
94
|
+
info: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.info(swap.getIdentifier()+": "+msg, ...args),
|
|
95
|
+
warn: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.warn(swap.getIdentifier()+": "+msg, ...args),
|
|
96
|
+
error: (swap: SwapHandlerSwap, msg: string, ...args: any) => this.logger.error(swap.getIdentifier()+": "+msg, ...args)
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
protected constructor(
|
|
100
|
+
storageDirectory: IIntermediaryStorage<V>,
|
|
101
|
+
path: string,
|
|
102
|
+
chainsData: MultichainData,
|
|
103
|
+
swapPricing: ISwapPrice
|
|
104
|
+
) {
|
|
105
|
+
this.storageManager = storageDirectory;
|
|
106
|
+
this.chains = chainsData;
|
|
107
|
+
if(this.chains.chains[this.chains.default]==null) throw new Error("Invalid default chain specified");
|
|
108
|
+
this.path = path;
|
|
109
|
+
this.swapPricing = swapPricing;
|
|
110
|
+
this.allowedTokens = {};
|
|
111
|
+
for(let chainId in chainsData.chains) {
|
|
112
|
+
this.allowedTokens[chainId] = new Set<string>(chainsData.chains[chainId].allowedTokens);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
protected getDefaultChain(): ChainData {
|
|
117
|
+
return this.chains.chains[this.chains.default];
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
protected getChain(identifier: string): ChainData {
|
|
121
|
+
if(this.chains.chains[identifier]==null)
|
|
122
|
+
throw {
|
|
123
|
+
code: 20200,
|
|
124
|
+
msg: "Invalid chain specified!"
|
|
125
|
+
};
|
|
126
|
+
return this.chains.chains[identifier];
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
protected abstract processPastSwaps(): Promise<void>;
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Starts the watchdog checking past swaps for expiry or claim eligibility.
|
|
133
|
+
*/
|
|
134
|
+
async startWatchdog() {
|
|
135
|
+
let rerun: () => Promise<void>;
|
|
136
|
+
rerun = async () => {
|
|
137
|
+
await this.processPastSwaps().catch( e => this.logger.error("startWatchdog(): Error when processing past swaps: ", e));
|
|
138
|
+
setTimeout(rerun, this.config.swapCheckInterval);
|
|
139
|
+
};
|
|
140
|
+
await rerun();
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
/**
|
|
144
|
+
* Initializes swap handler, loads data and subscribes to chain events
|
|
145
|
+
*/
|
|
146
|
+
abstract init(): Promise<void>;
|
|
147
|
+
|
|
148
|
+
protected async loadData(ctor: new (data: any) => V) {
|
|
149
|
+
await this.storageManager.loadData(ctor);
|
|
150
|
+
//Check if all swaps contain a valid amount
|
|
151
|
+
for(let {obj: swap, hash, sequence} of await this.storageManager.query([])) {
|
|
152
|
+
if(hash!==swap.getIdentifierHash() || sequence !== (swap.getSequence() ?? 0n)) {
|
|
153
|
+
this.swapLogger.info(swap, "loadData(): Swap storage key or sequence mismatch, fixing,"+
|
|
154
|
+
" old hash: "+hash+" new hash: "+swap.getIdentifierHash()+
|
|
155
|
+
" old seq: "+sequence.toString(10)+" new seq: "+(swap.getSequence() ?? 0n).toString(10));
|
|
156
|
+
|
|
157
|
+
await this.storageManager.removeData(hash, sequence);
|
|
158
|
+
await this.storageManager.saveData(swap.getIdentifierHash(), swap.getSequence(), swap);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Sets up required listeners for the REST server
|
|
165
|
+
*
|
|
166
|
+
* @param restServer
|
|
167
|
+
*/
|
|
168
|
+
abstract startRestServer(restServer: Express): void;
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* Returns data to be returned in swap handler info
|
|
172
|
+
*/
|
|
173
|
+
abstract getInfoData(): any;
|
|
174
|
+
|
|
175
|
+
/**
|
|
176
|
+
* Remove swap data
|
|
177
|
+
*
|
|
178
|
+
* @param hash
|
|
179
|
+
* @param sequence
|
|
180
|
+
*/
|
|
181
|
+
protected removeSwapData(hash: string, sequence: bigint): Promise<void>;
|
|
182
|
+
|
|
183
|
+
/**
|
|
184
|
+
* Remove swap data
|
|
185
|
+
*
|
|
186
|
+
* @param swap
|
|
187
|
+
* @param ultimateState set the ultimate state of the swap before removing
|
|
188
|
+
*/
|
|
189
|
+
protected removeSwapData(swap: V, ultimateState?: S): Promise<void>;
|
|
190
|
+
|
|
191
|
+
protected async removeSwapData(hashOrSwap: string | V, sequenceOrUltimateState?: bigint | S) {
|
|
192
|
+
let swap: V;
|
|
193
|
+
if(typeof(hashOrSwap)==="string") {
|
|
194
|
+
if(typeof(sequenceOrUltimateState)!=="bigint") throw new Error("Sequence must be a BN instance!");
|
|
195
|
+
swap = await this.storageManager.getData(hashOrSwap, sequenceOrUltimateState);
|
|
196
|
+
} else {
|
|
197
|
+
swap = hashOrSwap;
|
|
198
|
+
if(sequenceOrUltimateState!=null && typeof(sequenceOrUltimateState)!=="bigint") await swap.setState(sequenceOrUltimateState);
|
|
199
|
+
}
|
|
200
|
+
if(swap!=null) await PluginManager.swapRemove(swap);
|
|
201
|
+
this.swapLogger.debug(swap, "removeSwapData(): removing swap final state: "+swap.state);
|
|
202
|
+
await this.storageManager.removeData(swap.getIdentifierHash(), swap.getSequence());
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
protected async saveSwapData(swap: V) {
|
|
206
|
+
await this.storageManager.saveData(swap.getIdentifierHash(), swap.getSequence(), swap);
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
/**
|
|
210
|
+
* Checks if we have enough balance of the token in the swap vault
|
|
211
|
+
*
|
|
212
|
+
* @param totalInToken
|
|
213
|
+
* @param balancePrefetch
|
|
214
|
+
* @param signal
|
|
215
|
+
* @throws {DefinedRuntimeError} will throw an error if there are not enough funds in the vault
|
|
216
|
+
*/
|
|
217
|
+
protected async checkBalance(totalInToken: bigint, balancePrefetch: Promise<bigint>, signal: AbortSignal | null): Promise<void> {
|
|
218
|
+
const balance = await balancePrefetch;
|
|
219
|
+
if(signal!=null) signal.throwIfAborted();
|
|
220
|
+
|
|
221
|
+
if(balance==null || balance < totalInToken) {
|
|
222
|
+
throw {
|
|
223
|
+
code: 20002,
|
|
224
|
+
msg: "Not enough liquidity"
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* Checks if the sequence number is between 0-2^64
|
|
231
|
+
*
|
|
232
|
+
* @param sequence
|
|
233
|
+
* @throws {DefinedRuntimeError} will throw an error if sequence number is out of bounds
|
|
234
|
+
*/
|
|
235
|
+
protected checkSequence(sequence: bigint) {
|
|
236
|
+
if(sequence < 0n || sequence >= 2n ** 64n) {
|
|
237
|
+
throw {
|
|
238
|
+
code: 20060,
|
|
239
|
+
msg: "Invalid sequence"
|
|
240
|
+
};
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
/**
|
|
245
|
+
* Checks whether a given token is supported on a specified chain
|
|
246
|
+
*
|
|
247
|
+
* @param chainId
|
|
248
|
+
* @param token
|
|
249
|
+
* @protected
|
|
250
|
+
*/
|
|
251
|
+
protected isTokenSupported(chainId: string, token: string): boolean {
|
|
252
|
+
const chainTokens = this.allowedTokens[chainId];
|
|
253
|
+
if(chainTokens==null) return false;
|
|
254
|
+
return chainTokens.has(token);
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
getInfo(): SwapHandlerInfoType {
|
|
258
|
+
const chainTokens: {[chainId: string]: string[]} = {};
|
|
259
|
+
for(let chainId in this.allowedTokens) {
|
|
260
|
+
chainTokens[chainId] = Array.from<string>(this.allowedTokens[chainId]);
|
|
261
|
+
}
|
|
262
|
+
return {
|
|
263
|
+
swapFeePPM: Number(this.config.feePPM),
|
|
264
|
+
swapBaseFee: Number(this.config.baseFee),
|
|
265
|
+
min: Number(this.config.min),
|
|
266
|
+
max: Number(this.config.max),
|
|
267
|
+
data: this.getInfoData(),
|
|
268
|
+
tokens: Array.from<string>(this.allowedTokens[this.chains.default]),
|
|
269
|
+
chainTokens
|
|
270
|
+
};
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
protected getInitAuthorizationTimeout(chainIdentifier: string) {
|
|
274
|
+
return this.config.initAuthorizationTimeouts?.[chainIdentifier] ?? this.config.initAuthorizationTimeout;
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
}
|