@atomiqlabs/chain-evm 1.0.0-dev.100
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 -0
- package/dist/chains/botanix/BotanixChainType.d.ts +14 -0
- package/dist/chains/botanix/BotanixChainType.js +2 -0
- package/dist/chains/botanix/BotanixInitializer.d.ts +30 -0
- package/dist/chains/botanix/BotanixInitializer.js +125 -0
- package/dist/chains/citrea/CitreaBtcRelay.d.ts +21 -0
- package/dist/chains/citrea/CitreaBtcRelay.js +43 -0
- package/dist/chains/citrea/CitreaChainType.d.ts +14 -0
- package/dist/chains/citrea/CitreaChainType.js +2 -0
- package/dist/chains/citrea/CitreaFees.d.ts +29 -0
- package/dist/chains/citrea/CitreaFees.js +67 -0
- package/dist/chains/citrea/CitreaInitializer.d.ts +30 -0
- package/dist/chains/citrea/CitreaInitializer.js +132 -0
- package/dist/chains/citrea/CitreaSpvVaultContract.d.ts +15 -0
- package/dist/chains/citrea/CitreaSpvVaultContract.js +74 -0
- package/dist/chains/citrea/CitreaSwapContract.d.ts +22 -0
- package/dist/chains/citrea/CitreaSwapContract.js +96 -0
- package/dist/chains/citrea/CitreaTokens.d.ts +9 -0
- package/dist/chains/citrea/CitreaTokens.js +20 -0
- package/dist/evm/btcrelay/BtcRelayAbi.d.ts +198 -0
- package/dist/evm/btcrelay/BtcRelayAbi.js +261 -0
- package/dist/evm/btcrelay/BtcRelayTypechain.d.ts +172 -0
- package/dist/evm/btcrelay/BtcRelayTypechain.js +2 -0
- package/dist/evm/btcrelay/EVMBtcRelay.d.ts +197 -0
- package/dist/evm/btcrelay/EVMBtcRelay.js +435 -0
- package/dist/evm/btcrelay/headers/EVMBtcHeader.d.ts +33 -0
- package/dist/evm/btcrelay/headers/EVMBtcHeader.js +84 -0
- package/dist/evm/btcrelay/headers/EVMBtcStoredHeader.d.ts +56 -0
- package/dist/evm/btcrelay/headers/EVMBtcStoredHeader.js +123 -0
- package/dist/evm/chain/EVMChainInterface.d.ts +60 -0
- package/dist/evm/chain/EVMChainInterface.js +107 -0
- package/dist/evm/chain/EVMModule.d.ts +9 -0
- package/dist/evm/chain/EVMModule.js +13 -0
- package/dist/evm/chain/modules/ERC20Abi.d.ts +168 -0
- package/dist/evm/chain/modules/ERC20Abi.js +225 -0
- package/dist/evm/chain/modules/EVMAddresses.d.ts +10 -0
- package/dist/evm/chain/modules/EVMAddresses.js +30 -0
- package/dist/evm/chain/modules/EVMBlocks.d.ts +27 -0
- package/dist/evm/chain/modules/EVMBlocks.js +73 -0
- package/dist/evm/chain/modules/EVMEvents.d.ts +46 -0
- package/dist/evm/chain/modules/EVMEvents.js +151 -0
- package/dist/evm/chain/modules/EVMFees.d.ts +36 -0
- package/dist/evm/chain/modules/EVMFees.js +74 -0
- package/dist/evm/chain/modules/EVMSignatures.d.ts +29 -0
- package/dist/evm/chain/modules/EVMSignatures.js +68 -0
- package/dist/evm/chain/modules/EVMTokens.d.ts +70 -0
- package/dist/evm/chain/modules/EVMTokens.js +142 -0
- package/dist/evm/chain/modules/EVMTransactions.d.ts +94 -0
- package/dist/evm/chain/modules/EVMTransactions.js +307 -0
- package/dist/evm/contract/EVMContractBase.d.ts +22 -0
- package/dist/evm/contract/EVMContractBase.js +34 -0
- package/dist/evm/contract/EVMContractModule.d.ts +8 -0
- package/dist/evm/contract/EVMContractModule.js +11 -0
- package/dist/evm/contract/modules/EVMContractEvents.d.ts +43 -0
- package/dist/evm/contract/modules/EVMContractEvents.js +76 -0
- package/dist/evm/events/EVMChainEvents.d.ts +22 -0
- package/dist/evm/events/EVMChainEvents.js +69 -0
- package/dist/evm/events/EVMChainEventsBrowser.d.ts +102 -0
- package/dist/evm/events/EVMChainEventsBrowser.js +412 -0
- package/dist/evm/providers/JsonRpcProviderWithRetries.d.ts +16 -0
- package/dist/evm/providers/JsonRpcProviderWithRetries.js +27 -0
- package/dist/evm/providers/ReconnectingWebSocketProvider.d.ts +22 -0
- package/dist/evm/providers/ReconnectingWebSocketProvider.js +91 -0
- package/dist/evm/providers/SocketProvider.d.ts +111 -0
- package/dist/evm/providers/SocketProvider.js +336 -0
- package/dist/evm/providers/WebSocketProviderWithRetries.d.ts +17 -0
- package/dist/evm/providers/WebSocketProviderWithRetries.js +23 -0
- package/dist/evm/spv_swap/EVMSpvVaultContract.d.ts +110 -0
- package/dist/evm/spv_swap/EVMSpvVaultContract.js +629 -0
- package/dist/evm/spv_swap/EVMSpvVaultData.d.ts +40 -0
- package/dist/evm/spv_swap/EVMSpvVaultData.js +184 -0
- package/dist/evm/spv_swap/EVMSpvWithdrawalData.d.ts +19 -0
- package/dist/evm/spv_swap/EVMSpvWithdrawalData.js +55 -0
- package/dist/evm/spv_swap/SpvVaultContractAbi.d.ts +91 -0
- package/dist/evm/spv_swap/SpvVaultContractAbi.js +849 -0
- package/dist/evm/spv_swap/SpvVaultContractTypechain.d.ts +450 -0
- package/dist/evm/spv_swap/SpvVaultContractTypechain.js +2 -0
- package/dist/evm/swaps/EVMSwapContract.d.ts +199 -0
- package/dist/evm/swaps/EVMSwapContract.js +394 -0
- package/dist/evm/swaps/EVMSwapData.d.ts +66 -0
- package/dist/evm/swaps/EVMSwapData.js +260 -0
- package/dist/evm/swaps/EVMSwapModule.d.ts +9 -0
- package/dist/evm/swaps/EVMSwapModule.js +11 -0
- package/dist/evm/swaps/EscrowManagerAbi.d.ts +120 -0
- package/dist/evm/swaps/EscrowManagerAbi.js +985 -0
- package/dist/evm/swaps/EscrowManagerTypechain.d.ts +475 -0
- package/dist/evm/swaps/EscrowManagerTypechain.js +2 -0
- package/dist/evm/swaps/handlers/IHandler.d.ts +13 -0
- package/dist/evm/swaps/handlers/IHandler.js +2 -0
- package/dist/evm/swaps/handlers/claim/ClaimHandlers.d.ts +10 -0
- package/dist/evm/swaps/handlers/claim/ClaimHandlers.js +13 -0
- package/dist/evm/swaps/handlers/claim/HashlockClaimHandler.d.ts +20 -0
- package/dist/evm/swaps/handlers/claim/HashlockClaimHandler.js +39 -0
- package/dist/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.d.ts +24 -0
- package/dist/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.js +59 -0
- package/dist/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.d.ts +25 -0
- package/dist/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.js +51 -0
- package/dist/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.d.ts +21 -0
- package/dist/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.js +28 -0
- package/dist/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.d.ts +48 -0
- package/dist/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.js +63 -0
- package/dist/evm/swaps/handlers/refund/TimelockRefundHandler.d.ts +17 -0
- package/dist/evm/swaps/handlers/refund/TimelockRefundHandler.js +28 -0
- package/dist/evm/swaps/modules/EVMLpVault.d.ts +69 -0
- package/dist/evm/swaps/modules/EVMLpVault.js +134 -0
- package/dist/evm/swaps/modules/EVMSwapClaim.d.ts +54 -0
- package/dist/evm/swaps/modules/EVMSwapClaim.js +137 -0
- package/dist/evm/swaps/modules/EVMSwapInit.d.ts +88 -0
- package/dist/evm/swaps/modules/EVMSwapInit.js +274 -0
- package/dist/evm/swaps/modules/EVMSwapRefund.d.ts +62 -0
- package/dist/evm/swaps/modules/EVMSwapRefund.js +167 -0
- package/dist/evm/typechain/common.d.ts +50 -0
- package/dist/evm/typechain/common.js +2 -0
- package/dist/evm/wallet/EVMBrowserSigner.d.ts +5 -0
- package/dist/evm/wallet/EVMBrowserSigner.js +11 -0
- package/dist/evm/wallet/EVMPersistentSigner.d.ts +29 -0
- package/dist/evm/wallet/EVMPersistentSigner.js +230 -0
- package/dist/evm/wallet/EVMSigner.d.ts +12 -0
- package/dist/evm/wallet/EVMSigner.js +28 -0
- package/dist/index.d.ts +44 -0
- package/dist/index.js +60 -0
- package/dist/utils/Utils.d.ts +19 -0
- package/dist/utils/Utils.js +98 -0
- package/package.json +39 -0
- package/src/chains/botanix/BotanixChainType.ts +30 -0
- package/src/chains/botanix/BotanixInitializer.ts +175 -0
- package/src/chains/citrea/CitreaBtcRelay.ts +58 -0
- package/src/chains/citrea/CitreaChainType.ts +30 -0
- package/src/chains/citrea/CitreaFees.ts +77 -0
- package/src/chains/citrea/CitreaInitializer.ts +182 -0
- package/src/chains/citrea/CitreaSpvVaultContract.ts +75 -0
- package/src/chains/citrea/CitreaSwapContract.ts +103 -0
- package/src/chains/citrea/CitreaTokens.ts +22 -0
- package/src/evm/btcrelay/BtcRelayAbi.ts +258 -0
- package/src/evm/btcrelay/BtcRelayTypechain.ts +371 -0
- package/src/evm/btcrelay/EVMBtcRelay.ts +537 -0
- package/src/evm/btcrelay/headers/EVMBtcHeader.ts +110 -0
- package/src/evm/btcrelay/headers/EVMBtcStoredHeader.ts +153 -0
- package/src/evm/chain/EVMChainInterface.ts +189 -0
- package/src/evm/chain/EVMModule.ts +21 -0
- package/src/evm/chain/modules/ERC20Abi.ts +222 -0
- package/src/evm/chain/modules/EVMAddresses.ts +29 -0
- package/src/evm/chain/modules/EVMBlocks.ts +86 -0
- package/src/evm/chain/modules/EVMEvents.ts +182 -0
- package/src/evm/chain/modules/EVMFees.ts +104 -0
- package/src/evm/chain/modules/EVMSignatures.ts +76 -0
- package/src/evm/chain/modules/EVMTokens.ts +155 -0
- package/src/evm/chain/modules/EVMTransactions.ts +346 -0
- package/src/evm/contract/EVMContractBase.ts +63 -0
- package/src/evm/contract/EVMContractModule.ts +16 -0
- package/src/evm/contract/modules/EVMContractEvents.ts +104 -0
- package/src/evm/events/EVMChainEvents.ts +82 -0
- package/src/evm/events/EVMChainEventsBrowser.ts +533 -0
- package/src/evm/providers/JsonRpcProviderWithRetries.ts +34 -0
- package/src/evm/providers/ReconnectingWebSocketProvider.ts +107 -0
- package/src/evm/providers/SocketProvider.ts +371 -0
- package/src/evm/providers/WebSocketProviderWithRetries.ts +35 -0
- package/src/evm/spv_swap/EVMSpvVaultContract.ts +779 -0
- package/src/evm/spv_swap/EVMSpvVaultData.ts +228 -0
- package/src/evm/spv_swap/EVMSpvWithdrawalData.ts +70 -0
- package/src/evm/spv_swap/SpvVaultContractAbi.ts +846 -0
- package/src/evm/spv_swap/SpvVaultContractTypechain.ts +685 -0
- package/src/evm/swaps/EVMSwapContract.ts +621 -0
- package/src/evm/swaps/EVMSwapData.ts +378 -0
- package/src/evm/swaps/EVMSwapModule.ts +16 -0
- package/src/evm/swaps/EscrowManagerAbi.ts +982 -0
- package/src/evm/swaps/EscrowManagerTypechain.ts +723 -0
- package/src/evm/swaps/handlers/IHandler.ts +17 -0
- package/src/evm/swaps/handlers/claim/ClaimHandlers.ts +20 -0
- package/src/evm/swaps/handlers/claim/HashlockClaimHandler.ts +47 -0
- package/src/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.ts +82 -0
- package/src/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.ts +76 -0
- package/src/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.ts +46 -0
- package/src/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.ts +115 -0
- package/src/evm/swaps/handlers/refund/TimelockRefundHandler.ts +38 -0
- package/src/evm/swaps/modules/EVMLpVault.ts +155 -0
- package/src/evm/swaps/modules/EVMSwapClaim.ts +173 -0
- package/src/evm/swaps/modules/EVMSwapInit.ts +329 -0
- package/src/evm/swaps/modules/EVMSwapRefund.ts +229 -0
- package/src/evm/typechain/common.ts +131 -0
- package/src/evm/wallet/EVMBrowserSigner.ts +12 -0
- package/src/evm/wallet/EVMPersistentSigner.ts +307 -0
- package/src/evm/wallet/EVMSigner.ts +35 -0
- package/src/index.ts +53 -0
- package/src/utils/Utils.ts +111 -0
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import {BigIntBufferUtils, BtcStoredHeader, StatePredictorUtils} from "@atomiqlabs/base";
|
|
2
|
+
import {EVMBtcHeader, EVMBtcHeaderType} from "./EVMBtcHeader";
|
|
3
|
+
import {Buffer} from "buffer";
|
|
4
|
+
import {keccak256} from "ethers";
|
|
5
|
+
|
|
6
|
+
export type StarknetBtcStoredHeaderType = {
|
|
7
|
+
blockheader: EVMBtcHeader | EVMBtcHeaderType,
|
|
8
|
+
blockHash: Buffer,
|
|
9
|
+
chainWork: bigint,
|
|
10
|
+
blockHeight: number,
|
|
11
|
+
lastDiffAdjustment: number,
|
|
12
|
+
prevBlockTimestamps: number[]
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export class EVMBtcStoredHeader implements BtcStoredHeader<EVMBtcHeader> {
|
|
16
|
+
|
|
17
|
+
blockheader: EVMBtcHeader;
|
|
18
|
+
blockHash: Buffer;
|
|
19
|
+
chainWork: bigint;
|
|
20
|
+
blockHeight: number;
|
|
21
|
+
lastDiffAdjustment: number;
|
|
22
|
+
prevBlockTimestamps: number[];
|
|
23
|
+
|
|
24
|
+
constructor(obj: StarknetBtcStoredHeaderType) {
|
|
25
|
+
this.blockheader = obj.blockheader instanceof EVMBtcHeader ? obj.blockheader : new EVMBtcHeader(obj.blockheader);
|
|
26
|
+
this.blockHash = obj.blockHash;
|
|
27
|
+
this.chainWork = obj.chainWork;
|
|
28
|
+
this.blockHeight = obj.blockHeight;
|
|
29
|
+
this.lastDiffAdjustment = obj.lastDiffAdjustment;
|
|
30
|
+
this.prevBlockTimestamps = obj.prevBlockTimestamps;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
getBlockheight(): number {
|
|
34
|
+
return this.blockHeight;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
getChainWork(): Buffer {
|
|
38
|
+
return Buffer.from(this.chainWork.toString(16).padStart(64, "0"), "hex");
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
getHeader(): EVMBtcHeader {
|
|
42
|
+
return this.blockheader;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
getLastDiffAdjustment(): number {
|
|
46
|
+
return this.lastDiffAdjustment;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
getPrevBlockTimestamps(): number[] {
|
|
50
|
+
return this.prevBlockTimestamps;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
getBlockHash(): Buffer {
|
|
54
|
+
return Buffer.from([...this.blockHash]).reverse();
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Computes prevBlockTimestamps for a next block, shifting the old block timestamps to the left & appending
|
|
59
|
+
* this block's timestamp to the end
|
|
60
|
+
*
|
|
61
|
+
* @private
|
|
62
|
+
*/
|
|
63
|
+
private computeNextBlockTimestamps(): number[] {
|
|
64
|
+
const prevBlockTimestamps = [...this.prevBlockTimestamps];
|
|
65
|
+
for(let i=1;i<10;i++) {
|
|
66
|
+
prevBlockTimestamps[i-1] = prevBlockTimestamps[i];
|
|
67
|
+
}
|
|
68
|
+
prevBlockTimestamps[9] = this.blockheader.getTimestamp();
|
|
69
|
+
return prevBlockTimestamps;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Computes total chain work after a new header with "nbits" is added to the chain
|
|
74
|
+
*
|
|
75
|
+
* @param nbits
|
|
76
|
+
* @private
|
|
77
|
+
*/
|
|
78
|
+
private computeNextChainWork(nbits: number): bigint {
|
|
79
|
+
return this.chainWork + BigIntBufferUtils.fromBuffer(StatePredictorUtils.getChainwork(nbits));
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Computes lastDiffAdjustment, this changes only once every DIFF_ADJUSTMENT_PERIOD blocks
|
|
84
|
+
*
|
|
85
|
+
* @param headerTimestamp
|
|
86
|
+
* @private
|
|
87
|
+
*/
|
|
88
|
+
private computeNextLastDiffAdjustment(headerTimestamp: number) {
|
|
89
|
+
const blockheight = this.blockHeight+1;
|
|
90
|
+
|
|
91
|
+
let lastDiffAdjustment = this.lastDiffAdjustment;
|
|
92
|
+
if(blockheight % StatePredictorUtils.DIFF_ADJUSTMENT_PERIOD === 0) {
|
|
93
|
+
lastDiffAdjustment = headerTimestamp;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
return lastDiffAdjustment;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
computeNext(header: EVMBtcHeader): EVMBtcStoredHeader {
|
|
100
|
+
header.previousBlockhash = this.blockHash;
|
|
101
|
+
return new EVMBtcStoredHeader({
|
|
102
|
+
chainWork: this.computeNextChainWork(header.getNbits()),
|
|
103
|
+
prevBlockTimestamps: this.computeNextBlockTimestamps(),
|
|
104
|
+
blockHeight: this.blockHeight+1,
|
|
105
|
+
lastDiffAdjustment: this.computeNextLastDiffAdjustment(header.getTimestamp()),
|
|
106
|
+
blockHash: header.getHash(),
|
|
107
|
+
blockheader: header
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
getCommitHash(): string {
|
|
112
|
+
return keccak256(this.serialize());
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
serialize(): Buffer {
|
|
116
|
+
const buffer = Buffer.alloc(160);
|
|
117
|
+
this.blockheader.serialize().copy(buffer, 0, 0, 80);
|
|
118
|
+
BigIntBufferUtils.toBuffer(this.chainWork, "be", 32).copy(buffer, 80, 0, 32);
|
|
119
|
+
buffer.writeUint32BE(this.blockHeight, 112);
|
|
120
|
+
buffer.writeUint32BE(this.lastDiffAdjustment, 116);
|
|
121
|
+
for(let i=0;i<10;i++) {
|
|
122
|
+
buffer.writeUint32BE(this.prevBlockTimestamps[i], 120 + (i*4));
|
|
123
|
+
}
|
|
124
|
+
return buffer;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
serializeToStruct(): {data: [string, string, string, string, string]} {
|
|
128
|
+
const buffer = this.serialize();
|
|
129
|
+
const result: string[] = [];
|
|
130
|
+
for(let i=0;i<5;i++) {
|
|
131
|
+
result[i] = "0x"+buffer.subarray(i*32, (i+1)*32).toString("hex");
|
|
132
|
+
}
|
|
133
|
+
return {data: result as any};
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
static deserialize(data: Buffer): EVMBtcStoredHeader {
|
|
137
|
+
if(data.length!==160) throw new Error(`Invalid size Expected 160, got: ${data.length}!`);
|
|
138
|
+
const blockheader = EVMBtcHeader.deserialize(data.subarray(0, 80));
|
|
139
|
+
const prevBlockTimestamps: number[] = [];
|
|
140
|
+
for(let i=0;i<10;i++) {
|
|
141
|
+
prevBlockTimestamps[i] = data.readUint32BE(120 + (i*4));
|
|
142
|
+
}
|
|
143
|
+
return new EVMBtcStoredHeader({
|
|
144
|
+
blockheader,
|
|
145
|
+
blockHash: blockheader.getHash(),
|
|
146
|
+
chainWork: BigIntBufferUtils.fromBuffer(data.subarray(80, 112)),
|
|
147
|
+
blockHeight: data.readUint32BE(112),
|
|
148
|
+
lastDiffAdjustment: data.readUint32BE(116),
|
|
149
|
+
prevBlockTimestamps
|
|
150
|
+
});
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
}
|
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
import {ChainInterface, TransactionConfirmationOptions} from "@atomiqlabs/base";
|
|
2
|
+
import {getLogger, LoggerType} from "../../utils/Utils";
|
|
3
|
+
import {
|
|
4
|
+
BrowserProvider,
|
|
5
|
+
JsonRpcApiProvider,
|
|
6
|
+
JsonRpcSigner,
|
|
7
|
+
Signer,
|
|
8
|
+
Transaction,
|
|
9
|
+
TransactionRequest,
|
|
10
|
+
Wallet,
|
|
11
|
+
getAddress
|
|
12
|
+
} from "ethers";
|
|
13
|
+
import {EVMBlocks, EVMBlockTag} from "./modules/EVMBlocks";
|
|
14
|
+
import {EVMEvents} from "./modules/EVMEvents";
|
|
15
|
+
import {EVMFees} from "./modules/EVMFees";
|
|
16
|
+
import {EVMTokens} from "./modules/EVMTokens";
|
|
17
|
+
import {EVMTransactions, EVMTx} from "./modules/EVMTransactions";
|
|
18
|
+
import { EVMSignatures } from "./modules/EVMSignatures";
|
|
19
|
+
import {EVMAddresses} from "./modules/EVMAddresses";
|
|
20
|
+
import {EVMSigner} from "../wallet/EVMSigner";
|
|
21
|
+
import {EVMBrowserSigner} from "../wallet/EVMBrowserSigner";
|
|
22
|
+
|
|
23
|
+
export type EVMRetryPolicy = {
|
|
24
|
+
maxRetries?: number,
|
|
25
|
+
delay?: number,
|
|
26
|
+
exponential?: boolean
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export type EVMConfiguration = {
|
|
30
|
+
safeBlockTag: EVMBlockTag,
|
|
31
|
+
maxLogsBlockRange: number,
|
|
32
|
+
maxParallelLogRequests: number,
|
|
33
|
+
maxParallelCalls: number,
|
|
34
|
+
maxLogTopics: number
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
export class EVMChainInterface<ChainId extends string = string> implements ChainInterface<EVMTx, EVMSigner, ChainId, Signer> {
|
|
38
|
+
|
|
39
|
+
readonly chainId: ChainId;
|
|
40
|
+
|
|
41
|
+
readonly provider: JsonRpcApiProvider;
|
|
42
|
+
readonly retryPolicy: EVMRetryPolicy;
|
|
43
|
+
|
|
44
|
+
public readonly evmChainId: number;
|
|
45
|
+
|
|
46
|
+
public readonly config: EVMConfiguration;
|
|
47
|
+
|
|
48
|
+
public Fees: EVMFees;
|
|
49
|
+
public Tokens: EVMTokens;
|
|
50
|
+
public Transactions: EVMTransactions;
|
|
51
|
+
public Signatures: EVMSignatures;
|
|
52
|
+
public Events: EVMEvents;
|
|
53
|
+
public Blocks: EVMBlocks;
|
|
54
|
+
|
|
55
|
+
protected logger: LoggerType;
|
|
56
|
+
|
|
57
|
+
constructor(
|
|
58
|
+
chainId: ChainId,
|
|
59
|
+
evmChainId: number,
|
|
60
|
+
provider: JsonRpcApiProvider,
|
|
61
|
+
config: EVMConfiguration,
|
|
62
|
+
retryPolicy?: EVMRetryPolicy,
|
|
63
|
+
evmFeeEstimator: EVMFees = new EVMFees(provider)
|
|
64
|
+
) {
|
|
65
|
+
this.chainId = chainId;
|
|
66
|
+
this.evmChainId = evmChainId;
|
|
67
|
+
this.provider = provider;
|
|
68
|
+
this.retryPolicy = retryPolicy;
|
|
69
|
+
this.config = config;
|
|
70
|
+
this.config.safeBlockTag ??= "safe";
|
|
71
|
+
|
|
72
|
+
this.logger = getLogger("EVMChainInterface("+this.evmChainId+"): ");
|
|
73
|
+
|
|
74
|
+
this.Fees = evmFeeEstimator;
|
|
75
|
+
this.Tokens = new EVMTokens(this);
|
|
76
|
+
this.Transactions = new EVMTransactions(this);
|
|
77
|
+
this.Signatures = new EVMSignatures(this);
|
|
78
|
+
this.Events = new EVMEvents(this);
|
|
79
|
+
this.Blocks = new EVMBlocks(this);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
|
|
83
|
+
async getBalance(signer: string, tokenAddress: string): Promise<bigint> {
|
|
84
|
+
//TODO: For native token we should discount the cost of transactions
|
|
85
|
+
return await this.Tokens.getTokenBalance(signer, tokenAddress);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
getNativeCurrencyAddress(): string {
|
|
89
|
+
return this.Tokens.getNativeCurrencyAddress();
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
isValidToken(tokenIdentifier: string): boolean {
|
|
93
|
+
return this.Tokens.isValidToken(tokenIdentifier);
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
isValidAddress(address: string): boolean {
|
|
97
|
+
return EVMAddresses.isValidAddress(address);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
normalizeAddress(address: string): string {
|
|
101
|
+
return getAddress(address);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
///////////////////////////////////
|
|
105
|
+
//// Callbacks & handlers
|
|
106
|
+
offBeforeTxReplace(callback: (oldTx: string, oldTxId: string, newTx: string, newTxId: string) => Promise<void>): boolean {
|
|
107
|
+
return true;
|
|
108
|
+
}
|
|
109
|
+
onBeforeTxReplace(callback: (oldTx: string, oldTxId: string, newTx: string, newTxId: string) => Promise<void>): void {}
|
|
110
|
+
|
|
111
|
+
onBeforeTxSigned(callback: (tx: TransactionRequest) => Promise<void>): void {
|
|
112
|
+
this.Transactions.onBeforeTxSigned(callback);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
offBeforeTxSigned(callback: (tx: TransactionRequest) => Promise<void>): boolean {
|
|
116
|
+
return this.Transactions.offBeforeTxSigned(callback);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
randomAddress(): string {
|
|
120
|
+
return EVMAddresses.randomAddress();
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
randomSigner(): EVMSigner {
|
|
124
|
+
const wallet = Wallet.createRandom();
|
|
125
|
+
return new EVMSigner(wallet, wallet.address);
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
////////////////////////////////////////////
|
|
129
|
+
//// Transactions
|
|
130
|
+
sendAndConfirm(
|
|
131
|
+
signer: EVMSigner,
|
|
132
|
+
txs: TransactionRequest[],
|
|
133
|
+
waitForConfirmation?: boolean,
|
|
134
|
+
abortSignal?: AbortSignal,
|
|
135
|
+
parallel?: boolean,
|
|
136
|
+
onBeforePublish?: (txId: string, rawTx: string) => Promise<void>
|
|
137
|
+
): Promise<string[]> {
|
|
138
|
+
return this.Transactions.sendAndConfirm(signer, txs, waitForConfirmation, abortSignal, parallel, onBeforePublish);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
serializeTx(tx: Transaction): Promise<string> {
|
|
142
|
+
return this.Transactions.serializeTx(tx);
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
deserializeTx(txData: string): Promise<Transaction> {
|
|
146
|
+
return this.Transactions.deserializeTx(txData);
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
getTxIdStatus(txId: string): Promise<"not_found" | "pending" | "success" | "reverted"> {
|
|
150
|
+
return this.Transactions.getTxIdStatus(txId);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
getTxStatus(tx: string): Promise<"not_found" | "pending" | "success" | "reverted"> {
|
|
154
|
+
return this.Transactions.getTxStatus(tx);
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
async getFinalizedBlock(): Promise<{ height: number; blockHash: string }> {
|
|
158
|
+
const block = await this.Blocks.getBlock("finalized");
|
|
159
|
+
return {
|
|
160
|
+
height: block.number,
|
|
161
|
+
blockHash: block.hash
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
async txsTransfer(signer: string, token: string, amount: bigint, dstAddress: string, feeRate?: string): Promise<TransactionRequest[]> {
|
|
166
|
+
return [await this.Tokens.Transfer(signer, token, amount, dstAddress, feeRate)];
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
async transfer(
|
|
170
|
+
signer: EVMSigner,
|
|
171
|
+
token: string,
|
|
172
|
+
amount: bigint,
|
|
173
|
+
dstAddress: string,
|
|
174
|
+
txOptions?: TransactionConfirmationOptions
|
|
175
|
+
): Promise<string> {
|
|
176
|
+
const tx = await this.Tokens.Transfer(signer.getAddress(), token, amount, dstAddress, txOptions?.feeRate);
|
|
177
|
+
const [txId] = await this.Transactions.sendAndConfirm(signer, [tx], txOptions?.waitForConfirmation, txOptions?.abortSignal, false);
|
|
178
|
+
return txId;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
async wrapSigner(signer: Signer): Promise<EVMSigner> {
|
|
182
|
+
const address = await signer.getAddress();
|
|
183
|
+
if(signer instanceof JsonRpcSigner || signer.provider instanceof BrowserProvider) {
|
|
184
|
+
return new EVMBrowserSigner(signer, address);
|
|
185
|
+
}
|
|
186
|
+
return new EVMSigner(signer, address);
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import {getLogger} from "../../utils/Utils";
|
|
2
|
+
import {JsonRpcApiProvider} from "ethers";
|
|
3
|
+
import {EVMChainInterface, EVMRetryPolicy} from "./EVMChainInterface";
|
|
4
|
+
|
|
5
|
+
export class EVMModule<ChainId extends string = string> {
|
|
6
|
+
|
|
7
|
+
protected readonly provider: JsonRpcApiProvider;
|
|
8
|
+
protected readonly retryPolicy: EVMRetryPolicy;
|
|
9
|
+
protected readonly root: EVMChainInterface<ChainId>;
|
|
10
|
+
|
|
11
|
+
protected readonly logger = getLogger(this.constructor.name+": ");
|
|
12
|
+
|
|
13
|
+
constructor(
|
|
14
|
+
root: EVMChainInterface<ChainId>
|
|
15
|
+
) {
|
|
16
|
+
this.provider = root.provider;
|
|
17
|
+
this.retryPolicy = root.retryPolicy;
|
|
18
|
+
this.root = root;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
}
|
|
@@ -0,0 +1,222 @@
|
|
|
1
|
+
export const ERC20Abi = [
|
|
2
|
+
{
|
|
3
|
+
"constant": true,
|
|
4
|
+
"inputs": [],
|
|
5
|
+
"name": "name",
|
|
6
|
+
"outputs": [
|
|
7
|
+
{
|
|
8
|
+
"name": "",
|
|
9
|
+
"type": "string"
|
|
10
|
+
}
|
|
11
|
+
],
|
|
12
|
+
"payable": false,
|
|
13
|
+
"stateMutability": "view",
|
|
14
|
+
"type": "function"
|
|
15
|
+
},
|
|
16
|
+
{
|
|
17
|
+
"constant": false,
|
|
18
|
+
"inputs": [
|
|
19
|
+
{
|
|
20
|
+
"name": "_spender",
|
|
21
|
+
"type": "address"
|
|
22
|
+
},
|
|
23
|
+
{
|
|
24
|
+
"name": "_value",
|
|
25
|
+
"type": "uint256"
|
|
26
|
+
}
|
|
27
|
+
],
|
|
28
|
+
"name": "approve",
|
|
29
|
+
"outputs": [
|
|
30
|
+
{
|
|
31
|
+
"name": "",
|
|
32
|
+
"type": "bool"
|
|
33
|
+
}
|
|
34
|
+
],
|
|
35
|
+
"payable": false,
|
|
36
|
+
"stateMutability": "nonpayable",
|
|
37
|
+
"type": "function"
|
|
38
|
+
},
|
|
39
|
+
{
|
|
40
|
+
"constant": true,
|
|
41
|
+
"inputs": [],
|
|
42
|
+
"name": "totalSupply",
|
|
43
|
+
"outputs": [
|
|
44
|
+
{
|
|
45
|
+
"name": "",
|
|
46
|
+
"type": "uint256"
|
|
47
|
+
}
|
|
48
|
+
],
|
|
49
|
+
"payable": false,
|
|
50
|
+
"stateMutability": "view",
|
|
51
|
+
"type": "function"
|
|
52
|
+
},
|
|
53
|
+
{
|
|
54
|
+
"constant": false,
|
|
55
|
+
"inputs": [
|
|
56
|
+
{
|
|
57
|
+
"name": "_from",
|
|
58
|
+
"type": "address"
|
|
59
|
+
},
|
|
60
|
+
{
|
|
61
|
+
"name": "_to",
|
|
62
|
+
"type": "address"
|
|
63
|
+
},
|
|
64
|
+
{
|
|
65
|
+
"name": "_value",
|
|
66
|
+
"type": "uint256"
|
|
67
|
+
}
|
|
68
|
+
],
|
|
69
|
+
"name": "transferFrom",
|
|
70
|
+
"outputs": [
|
|
71
|
+
{
|
|
72
|
+
"name": "",
|
|
73
|
+
"type": "bool"
|
|
74
|
+
}
|
|
75
|
+
],
|
|
76
|
+
"payable": false,
|
|
77
|
+
"stateMutability": "nonpayable",
|
|
78
|
+
"type": "function"
|
|
79
|
+
},
|
|
80
|
+
{
|
|
81
|
+
"constant": true,
|
|
82
|
+
"inputs": [],
|
|
83
|
+
"name": "decimals",
|
|
84
|
+
"outputs": [
|
|
85
|
+
{
|
|
86
|
+
"name": "",
|
|
87
|
+
"type": "uint8"
|
|
88
|
+
}
|
|
89
|
+
],
|
|
90
|
+
"payable": false,
|
|
91
|
+
"stateMutability": "view",
|
|
92
|
+
"type": "function"
|
|
93
|
+
},
|
|
94
|
+
{
|
|
95
|
+
"constant": true,
|
|
96
|
+
"inputs": [
|
|
97
|
+
{
|
|
98
|
+
"name": "_owner",
|
|
99
|
+
"type": "address"
|
|
100
|
+
}
|
|
101
|
+
],
|
|
102
|
+
"name": "balanceOf",
|
|
103
|
+
"outputs": [
|
|
104
|
+
{
|
|
105
|
+
"name": "balance",
|
|
106
|
+
"type": "uint256"
|
|
107
|
+
}
|
|
108
|
+
],
|
|
109
|
+
"payable": false,
|
|
110
|
+
"stateMutability": "view",
|
|
111
|
+
"type": "function"
|
|
112
|
+
},
|
|
113
|
+
{
|
|
114
|
+
"constant": true,
|
|
115
|
+
"inputs": [],
|
|
116
|
+
"name": "symbol",
|
|
117
|
+
"outputs": [
|
|
118
|
+
{
|
|
119
|
+
"name": "",
|
|
120
|
+
"type": "string"
|
|
121
|
+
}
|
|
122
|
+
],
|
|
123
|
+
"payable": false,
|
|
124
|
+
"stateMutability": "view",
|
|
125
|
+
"type": "function"
|
|
126
|
+
},
|
|
127
|
+
{
|
|
128
|
+
"constant": false,
|
|
129
|
+
"inputs": [
|
|
130
|
+
{
|
|
131
|
+
"name": "_to",
|
|
132
|
+
"type": "address"
|
|
133
|
+
},
|
|
134
|
+
{
|
|
135
|
+
"name": "_value",
|
|
136
|
+
"type": "uint256"
|
|
137
|
+
}
|
|
138
|
+
],
|
|
139
|
+
"name": "transfer",
|
|
140
|
+
"outputs": [
|
|
141
|
+
{
|
|
142
|
+
"name": "",
|
|
143
|
+
"type": "bool"
|
|
144
|
+
}
|
|
145
|
+
],
|
|
146
|
+
"payable": false,
|
|
147
|
+
"stateMutability": "nonpayable",
|
|
148
|
+
"type": "function"
|
|
149
|
+
},
|
|
150
|
+
{
|
|
151
|
+
"constant": true,
|
|
152
|
+
"inputs": [
|
|
153
|
+
{
|
|
154
|
+
"name": "_owner",
|
|
155
|
+
"type": "address"
|
|
156
|
+
},
|
|
157
|
+
{
|
|
158
|
+
"name": "_spender",
|
|
159
|
+
"type": "address"
|
|
160
|
+
}
|
|
161
|
+
],
|
|
162
|
+
"name": "allowance",
|
|
163
|
+
"outputs": [
|
|
164
|
+
{
|
|
165
|
+
"name": "",
|
|
166
|
+
"type": "uint256"
|
|
167
|
+
}
|
|
168
|
+
],
|
|
169
|
+
"payable": false,
|
|
170
|
+
"stateMutability": "view",
|
|
171
|
+
"type": "function"
|
|
172
|
+
},
|
|
173
|
+
{
|
|
174
|
+
"payable": true,
|
|
175
|
+
"stateMutability": "payable",
|
|
176
|
+
"type": "fallback"
|
|
177
|
+
},
|
|
178
|
+
{
|
|
179
|
+
"anonymous": false,
|
|
180
|
+
"inputs": [
|
|
181
|
+
{
|
|
182
|
+
"indexed": true,
|
|
183
|
+
"name": "owner",
|
|
184
|
+
"type": "address"
|
|
185
|
+
},
|
|
186
|
+
{
|
|
187
|
+
"indexed": true,
|
|
188
|
+
"name": "spender",
|
|
189
|
+
"type": "address"
|
|
190
|
+
},
|
|
191
|
+
{
|
|
192
|
+
"indexed": false,
|
|
193
|
+
"name": "value",
|
|
194
|
+
"type": "uint256"
|
|
195
|
+
}
|
|
196
|
+
],
|
|
197
|
+
"name": "Approval",
|
|
198
|
+
"type": "event"
|
|
199
|
+
},
|
|
200
|
+
{
|
|
201
|
+
"anonymous": false,
|
|
202
|
+
"inputs": [
|
|
203
|
+
{
|
|
204
|
+
"indexed": true,
|
|
205
|
+
"name": "from",
|
|
206
|
+
"type": "address"
|
|
207
|
+
},
|
|
208
|
+
{
|
|
209
|
+
"indexed": true,
|
|
210
|
+
"name": "to",
|
|
211
|
+
"type": "address"
|
|
212
|
+
},
|
|
213
|
+
{
|
|
214
|
+
"indexed": false,
|
|
215
|
+
"name": "value",
|
|
216
|
+
"type": "uint256"
|
|
217
|
+
}
|
|
218
|
+
],
|
|
219
|
+
"name": "Transfer",
|
|
220
|
+
"type": "event"
|
|
221
|
+
}
|
|
222
|
+
] as const;
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import {EVMModule} from "../EVMModule";
|
|
2
|
+
import {Wallet, isAddress} from "ethers";
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
export class EVMAddresses extends EVMModule<any> {
|
|
6
|
+
|
|
7
|
+
///////////////////
|
|
8
|
+
//// Address utils
|
|
9
|
+
/**
|
|
10
|
+
* Checks whether an address is a valid EVM address
|
|
11
|
+
*
|
|
12
|
+
* @param value
|
|
13
|
+
*/
|
|
14
|
+
static isValidAddress(value: string): boolean {
|
|
15
|
+
if(value.length!==42) return false;
|
|
16
|
+
try {
|
|
17
|
+
isAddress(value);
|
|
18
|
+
return true;
|
|
19
|
+
} catch (e) {
|
|
20
|
+
return false;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
static randomAddress(): string {
|
|
25
|
+
const wallet = Wallet.createRandom();
|
|
26
|
+
return wallet.address;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
}
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import {EVMModule} from "../EVMModule";
|
|
2
|
+
import {Block} from "ethers";
|
|
3
|
+
|
|
4
|
+
export type EVMBlockTag = "safe" | "pending" | "latest" | "finalized";
|
|
5
|
+
|
|
6
|
+
export class EVMBlocks extends EVMModule<any> {
|
|
7
|
+
|
|
8
|
+
private BLOCK_CACHE_TIME = 5*1000;
|
|
9
|
+
|
|
10
|
+
private blockCache: {
|
|
11
|
+
[key: string]: {
|
|
12
|
+
block: Promise<Block>,
|
|
13
|
+
timestamp: number
|
|
14
|
+
}
|
|
15
|
+
} = {};
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Initiates fetch of a given block & saves it to cache
|
|
19
|
+
*
|
|
20
|
+
* @private
|
|
21
|
+
* @param blockTag
|
|
22
|
+
*/
|
|
23
|
+
private fetchAndSaveBlockTime(blockTag: EVMBlockTag | number): {
|
|
24
|
+
block: Promise<Block>,
|
|
25
|
+
timestamp: number
|
|
26
|
+
} {
|
|
27
|
+
const blockTagStr = blockTag.toString(10);
|
|
28
|
+
|
|
29
|
+
const blockPromise = this.provider.getBlock(blockTag, false);
|
|
30
|
+
const timestamp = Date.now();
|
|
31
|
+
this.blockCache[blockTagStr] = {
|
|
32
|
+
block: blockPromise,
|
|
33
|
+
timestamp
|
|
34
|
+
};
|
|
35
|
+
blockPromise.catch(e => {
|
|
36
|
+
if(this.blockCache[blockTagStr]!=null && this.blockCache[blockTagStr].block===blockPromise) delete this.blockCache[blockTagStr];
|
|
37
|
+
throw e;
|
|
38
|
+
})
|
|
39
|
+
return {
|
|
40
|
+
block: blockPromise,
|
|
41
|
+
timestamp
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
private cleanupBlocks() {
|
|
46
|
+
const currentTime = Date.now();
|
|
47
|
+
//Keys are in order that they were added, so we can stop at the first non-expired block
|
|
48
|
+
for(let key in this.blockCache) {
|
|
49
|
+
const block = this.blockCache[key];
|
|
50
|
+
if(currentTime - block.timestamp > this.BLOCK_CACHE_TIME) {
|
|
51
|
+
delete this.blockCache[key];
|
|
52
|
+
} else {
|
|
53
|
+
break;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
///////////////////
|
|
59
|
+
//// Blocks
|
|
60
|
+
/**
|
|
61
|
+
* Gets the block for a given blocktag, with caching
|
|
62
|
+
*
|
|
63
|
+
* @param blockTag
|
|
64
|
+
*/
|
|
65
|
+
public getBlock(blockTag: EVMBlockTag | number): Promise<Block> {
|
|
66
|
+
this.cleanupBlocks();
|
|
67
|
+
let cachedBlockData = this.blockCache[blockTag.toString(10)];
|
|
68
|
+
|
|
69
|
+
if(cachedBlockData==null || Date.now()-cachedBlockData.timestamp>this.BLOCK_CACHE_TIME) {
|
|
70
|
+
cachedBlockData = this.fetchAndSaveBlockTime(blockTag);
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
return cachedBlockData.block;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Gets the block time for a given blocktag, with caching
|
|
78
|
+
*
|
|
79
|
+
* @param blockTag
|
|
80
|
+
*/
|
|
81
|
+
public async getBlockTime(blockTag: EVMBlockTag | number): Promise<number> {
|
|
82
|
+
const block = await this.getBlock(blockTag);
|
|
83
|
+
return block.timestamp;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
}
|