@atomiqlabs/chain-evm 1.0.0-dev.22
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/citrea/CitreaChainType.d.ts +13 -0
- package/dist/chains/citrea/CitreaChainType.js +2 -0
- package/dist/chains/citrea/CitreaInitializer.d.ts +30 -0
- package/dist/chains/citrea/CitreaInitializer.js +120 -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 +188 -0
- package/dist/evm/btcrelay/EVMBtcRelay.js +419 -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 +51 -0
- package/dist/evm/chain/EVMChainInterface.js +90 -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 +9 -0
- package/dist/evm/chain/modules/EVMAddresses.js +26 -0
- package/dist/evm/chain/modules/EVMBlocks.d.ts +20 -0
- package/dist/evm/chain/modules/EVMBlocks.js +64 -0
- package/dist/evm/chain/modules/EVMEvents.d.ts +36 -0
- package/dist/evm/chain/modules/EVMEvents.js +122 -0
- package/dist/evm/chain/modules/EVMFees.d.ts +35 -0
- package/dist/evm/chain/modules/EVMFees.js +73 -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 +49 -0
- package/dist/evm/chain/modules/EVMTokens.js +105 -0
- package/dist/evm/chain/modules/EVMTransactions.d.ts +89 -0
- package/dist/evm/chain/modules/EVMTransactions.js +216 -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 +42 -0
- package/dist/evm/contract/modules/EVMContractEvents.js +75 -0
- package/dist/evm/events/EVMChainEvents.d.ts +22 -0
- package/dist/evm/events/EVMChainEvents.js +67 -0
- package/dist/evm/events/EVMChainEventsBrowser.d.ts +86 -0
- package/dist/evm/events/EVMChainEventsBrowser.js +294 -0
- package/dist/evm/spv_swap/EVMSpvVaultContract.d.ts +64 -0
- package/dist/evm/spv_swap/EVMSpvVaultContract.js +410 -0
- package/dist/evm/spv_swap/EVMSpvVaultData.d.ts +38 -0
- package/dist/evm/spv_swap/EVMSpvVaultData.js +159 -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 +192 -0
- package/dist/evm/swaps/EVMSwapContract.js +373 -0
- package/dist/evm/swaps/EVMSwapData.d.ts +64 -0
- package/dist/evm/swaps/EVMSwapData.js +254 -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 +131 -0
- package/dist/evm/swaps/modules/EVMSwapClaim.d.ts +53 -0
- package/dist/evm/swaps/modules/EVMSwapClaim.js +101 -0
- package/dist/evm/swaps/modules/EVMSwapInit.d.ts +88 -0
- package/dist/evm/swaps/modules/EVMSwapInit.js +241 -0
- package/dist/evm/swaps/modules/EVMSwapRefund.d.ts +62 -0
- package/dist/evm/swaps/modules/EVMSwapRefund.js +132 -0
- package/dist/evm/typechain/common.d.ts +50 -0
- package/dist/evm/typechain/common.js +2 -0
- package/dist/evm/wallet/EVMSigner.d.ts +9 -0
- package/dist/evm/wallet/EVMSigner.js +16 -0
- package/dist/index.d.ts +37 -0
- package/dist/index.js +53 -0
- package/dist/utils/Utils.d.ts +15 -0
- package/dist/utils/Utils.js +71 -0
- package/package.json +37 -0
- package/src/chains/citrea/CitreaChainType.ts +28 -0
- package/src/chains/citrea/CitreaInitializer.ts +167 -0
- package/src/evm/btcrelay/BtcRelayAbi.ts +258 -0
- package/src/evm/btcrelay/BtcRelayTypechain.ts +371 -0
- package/src/evm/btcrelay/EVMBtcRelay.ts +517 -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 +157 -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 +24 -0
- package/src/evm/chain/modules/EVMBlocks.ts +75 -0
- package/src/evm/chain/modules/EVMEvents.ts +139 -0
- package/src/evm/chain/modules/EVMFees.ts +105 -0
- package/src/evm/chain/modules/EVMSignatures.ts +76 -0
- package/src/evm/chain/modules/EVMTokens.ts +115 -0
- package/src/evm/chain/modules/EVMTransactions.ts +246 -0
- package/src/evm/contract/EVMContractBase.ts +63 -0
- package/src/evm/contract/EVMContractModule.ts +16 -0
- package/src/evm/contract/modules/EVMContractEvents.ts +102 -0
- package/src/evm/events/EVMChainEvents.ts +81 -0
- package/src/evm/events/EVMChainEventsBrowser.ts +390 -0
- package/src/evm/spv_swap/EVMSpvVaultContract.ts +533 -0
- package/src/evm/spv_swap/EVMSpvVaultData.ts +201 -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 +590 -0
- package/src/evm/swaps/EVMSwapData.ts +367 -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 +153 -0
- package/src/evm/swaps/modules/EVMSwapClaim.ts +141 -0
- package/src/evm/swaps/modules/EVMSwapInit.ts +292 -0
- package/src/evm/swaps/modules/EVMSwapRefund.ts +198 -0
- package/src/evm/typechain/common.ts +131 -0
- package/src/evm/wallet/EVMSigner.ts +23 -0
- package/src/index.ts +44 -0
- package/src/utils/Utils.ts +81 -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,157 @@
|
|
|
1
|
+
import {ChainInterface, TransactionConfirmationOptions} from "@atomiqlabs/base";
|
|
2
|
+
import {getLogger, LoggerType} from "../../utils/Utils";
|
|
3
|
+
import {JsonRpcApiProvider, Transaction, TransactionRequest, Wallet} from "ethers";
|
|
4
|
+
import {EVMBlocks, EVMBlockTag} from "./modules/EVMBlocks";
|
|
5
|
+
import {EVMEvents} from "./modules/EVMEvents";
|
|
6
|
+
import {EVMFees} from "./modules/EVMFees";
|
|
7
|
+
import {EVMTokens} from "./modules/EVMTokens";
|
|
8
|
+
import { EVMTransactions } from "./modules/EVMTransactions";
|
|
9
|
+
import { EVMSignatures } from "./modules/EVMSignatures";
|
|
10
|
+
import {EVMAddresses} from "./modules/EVMAddresses";
|
|
11
|
+
import {EVMSigner} from "../wallet/EVMSigner";
|
|
12
|
+
|
|
13
|
+
export type EVMRetryPolicy = {
|
|
14
|
+
maxRetries?: number,
|
|
15
|
+
delay?: number,
|
|
16
|
+
exponential?: boolean
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export type EVMConfiguration = {
|
|
20
|
+
safeBlockTag: EVMBlockTag,
|
|
21
|
+
maxLogsBlockRange: number
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
export class EVMChainInterface<ChainId extends string = string, EVMChainId extends number = number> implements ChainInterface {
|
|
25
|
+
|
|
26
|
+
readonly chainId: ChainId;
|
|
27
|
+
|
|
28
|
+
readonly provider: JsonRpcApiProvider;
|
|
29
|
+
readonly retryPolicy: EVMRetryPolicy;
|
|
30
|
+
|
|
31
|
+
public readonly evmChainId: EVMChainId;
|
|
32
|
+
|
|
33
|
+
public readonly config: EVMConfiguration;
|
|
34
|
+
|
|
35
|
+
public Fees: EVMFees;
|
|
36
|
+
public readonly Tokens: EVMTokens;
|
|
37
|
+
public readonly Transactions: EVMTransactions;
|
|
38
|
+
public readonly Signatures: EVMSignatures;
|
|
39
|
+
public readonly Events: EVMEvents;
|
|
40
|
+
public readonly Blocks: EVMBlocks;
|
|
41
|
+
|
|
42
|
+
protected readonly logger: LoggerType;
|
|
43
|
+
|
|
44
|
+
constructor(
|
|
45
|
+
chainId: ChainId,
|
|
46
|
+
evmChainId: EVMChainId,
|
|
47
|
+
provider: JsonRpcApiProvider,
|
|
48
|
+
config: EVMConfiguration,
|
|
49
|
+
retryPolicy?: EVMRetryPolicy,
|
|
50
|
+
evmFeeEstimator: EVMFees = new EVMFees(provider)
|
|
51
|
+
) {
|
|
52
|
+
this.chainId = chainId;
|
|
53
|
+
this.evmChainId = evmChainId;
|
|
54
|
+
this.provider = provider;
|
|
55
|
+
this.retryPolicy = retryPolicy;
|
|
56
|
+
this.config = config;
|
|
57
|
+
this.config.safeBlockTag ??= "safe";
|
|
58
|
+
|
|
59
|
+
this.logger = getLogger("EVMChainInterface("+this.evmChainId+"): ");
|
|
60
|
+
|
|
61
|
+
this.Fees = evmFeeEstimator;
|
|
62
|
+
this.Tokens = new EVMTokens(this);
|
|
63
|
+
this.Transactions = new EVMTransactions(this);
|
|
64
|
+
this.Signatures = new EVMSignatures(this);
|
|
65
|
+
this.Events = new EVMEvents(this);
|
|
66
|
+
this.Blocks = new EVMBlocks(this);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
|
|
70
|
+
async getBalance(signer: string, tokenAddress: string): Promise<bigint> {
|
|
71
|
+
//TODO: For native token we should discount the cost of transactions
|
|
72
|
+
return await this.Tokens.getTokenBalance(signer, tokenAddress);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
getNativeCurrencyAddress(): string {
|
|
76
|
+
return this.Tokens.getNativeCurrencyAddress();
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
isValidToken(tokenIdentifier: string): boolean {
|
|
80
|
+
return this.Tokens.isValidToken(tokenIdentifier);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
isValidAddress(address: string): boolean {
|
|
84
|
+
return EVMAddresses.isValidAddress(address);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
///////////////////////////////////
|
|
88
|
+
//// Callbacks & handlers
|
|
89
|
+
offBeforeTxReplace(callback: (oldTx: string, oldTxId: string, newTx: string, newTxId: string) => Promise<void>): boolean {
|
|
90
|
+
return true;
|
|
91
|
+
}
|
|
92
|
+
onBeforeTxReplace(callback: (oldTx: string, oldTxId: string, newTx: string, newTxId: string) => Promise<void>): void {}
|
|
93
|
+
|
|
94
|
+
onBeforeTxSigned(callback: (tx: TransactionRequest) => Promise<void>): void {
|
|
95
|
+
this.Transactions.onBeforeTxSigned(callback);
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
offBeforeTxSigned(callback: (tx: TransactionRequest) => Promise<void>): boolean {
|
|
99
|
+
return this.Transactions.offBeforeTxSigned(callback);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
randomAddress(): string {
|
|
103
|
+
const wallet = Wallet.createRandom();
|
|
104
|
+
return wallet.address;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
randomSigner(): EVMSigner {
|
|
108
|
+
const wallet = Wallet.createRandom();
|
|
109
|
+
return new EVMSigner(wallet, wallet.address);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
////////////////////////////////////////////
|
|
113
|
+
//// Transactions
|
|
114
|
+
sendAndConfirm(
|
|
115
|
+
signer: EVMSigner,
|
|
116
|
+
txs: TransactionRequest[],
|
|
117
|
+
waitForConfirmation?: boolean,
|
|
118
|
+
abortSignal?: AbortSignal,
|
|
119
|
+
parallel?: boolean,
|
|
120
|
+
onBeforePublish?: (txId: string, rawTx: string) => Promise<void>
|
|
121
|
+
): Promise<string[]> {
|
|
122
|
+
return this.Transactions.sendAndConfirm(signer, txs, waitForConfirmation, abortSignal, parallel, onBeforePublish);
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
serializeTx(tx: Transaction): Promise<string> {
|
|
126
|
+
return this.Transactions.serializeTx(tx);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
deserializeTx(txData: string): Promise<Transaction> {
|
|
130
|
+
return this.Transactions.deserializeTx(txData);
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
getTxIdStatus(txId: string): Promise<"not_found" | "pending" | "success" | "reverted"> {
|
|
134
|
+
return this.Transactions.getTxIdStatus(txId);
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
getTxStatus(tx: string): Promise<"not_found" | "pending" | "success" | "reverted"> {
|
|
138
|
+
return this.Transactions.getTxStatus(tx);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
async txsTransfer(signer: string, token: string, amount: bigint, dstAddress: string, feeRate?: string): Promise<TransactionRequest[]> {
|
|
142
|
+
return [await this.Tokens.Transfer(signer, token, amount, dstAddress, feeRate)];
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
async transfer(
|
|
146
|
+
signer: EVMSigner,
|
|
147
|
+
token: string,
|
|
148
|
+
amount: bigint,
|
|
149
|
+
dstAddress: string,
|
|
150
|
+
txOptions?: TransactionConfirmationOptions
|
|
151
|
+
): Promise<string> {
|
|
152
|
+
const tx = await this.Tokens.Transfer(signer.getAddress(), token, amount, dstAddress, txOptions?.feeRate);
|
|
153
|
+
const [txId] = await this.Transactions.sendAndConfirm(signer, [tx], txOptions?.waitForConfirmation, txOptions?.abortSignal, false);
|
|
154
|
+
return txId;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
}
|
|
@@ -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, EVMChainId extends number = number> {
|
|
6
|
+
|
|
7
|
+
protected readonly provider: JsonRpcApiProvider;
|
|
8
|
+
protected readonly retryPolicy: EVMRetryPolicy;
|
|
9
|
+
protected readonly root: EVMChainInterface<ChainId, EVMChainId>;
|
|
10
|
+
|
|
11
|
+
protected readonly logger = getLogger(this.constructor.name+": ");
|
|
12
|
+
|
|
13
|
+
constructor(
|
|
14
|
+
root: EVMChainInterface<ChainId, EVMChainId>
|
|
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,24 @@
|
|
|
1
|
+
import {EVMModule} from "../EVMModule";
|
|
2
|
+
import {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
|
+
}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import {EVMModule} from "../EVMModule";
|
|
2
|
+
|
|
3
|
+
export type EVMBlockTag = "safe" | "pending" | "latest" | "finalized";
|
|
4
|
+
|
|
5
|
+
export class EVMBlocks extends EVMModule<any> {
|
|
6
|
+
|
|
7
|
+
private BLOCK_CACHE_TIME = 5*1000;
|
|
8
|
+
|
|
9
|
+
private blockCache: {
|
|
10
|
+
[key: string]: {
|
|
11
|
+
blockTime: Promise<number>,
|
|
12
|
+
timestamp: number
|
|
13
|
+
}
|
|
14
|
+
} = {};
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Initiates fetch of a given block & saves it to cache
|
|
18
|
+
*
|
|
19
|
+
* @private
|
|
20
|
+
* @param blockTag
|
|
21
|
+
*/
|
|
22
|
+
private fetchAndSaveBlockTime(blockTag: EVMBlockTag | number): {
|
|
23
|
+
blockTime: Promise<number>,
|
|
24
|
+
timestamp: number
|
|
25
|
+
} {
|
|
26
|
+
const blockTagStr = blockTag.toString(10);
|
|
27
|
+
|
|
28
|
+
const blockTimePromise = this.provider.getBlock(blockTag, false).then(result => result.timestamp);
|
|
29
|
+
const timestamp = Date.now();
|
|
30
|
+
this.blockCache[blockTagStr] = {
|
|
31
|
+
blockTime: blockTimePromise,
|
|
32
|
+
timestamp
|
|
33
|
+
};
|
|
34
|
+
blockTimePromise.catch(e => {
|
|
35
|
+
if(this.blockCache[blockTagStr]!=null && this.blockCache[blockTagStr].blockTime===blockTimePromise) delete this.blockCache[blockTagStr];
|
|
36
|
+
throw e;
|
|
37
|
+
})
|
|
38
|
+
return {
|
|
39
|
+
blockTime: blockTimePromise,
|
|
40
|
+
timestamp
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
private cleanupBlocks() {
|
|
45
|
+
const currentTime = Date.now();
|
|
46
|
+
//Keys are in order that they were added, so we can stop at the first non-expired block
|
|
47
|
+
for(let key in this.blockCache) {
|
|
48
|
+
const block = this.blockCache[key];
|
|
49
|
+
if(currentTime - block.timestamp > this.BLOCK_CACHE_TIME) {
|
|
50
|
+
delete this.blockCache[key];
|
|
51
|
+
} else {
|
|
52
|
+
break;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
///////////////////
|
|
58
|
+
//// Blocks
|
|
59
|
+
/**
|
|
60
|
+
* Gets the block for a given blocktag, with caching
|
|
61
|
+
*
|
|
62
|
+
* @param blockTag
|
|
63
|
+
*/
|
|
64
|
+
public getBlockTime(blockTag: EVMBlockTag | number): Promise<number> {
|
|
65
|
+
this.cleanupBlocks();
|
|
66
|
+
let cachedBlockData = this.blockCache[blockTag.toString(10)];
|
|
67
|
+
|
|
68
|
+
if(cachedBlockData==null || Date.now()-cachedBlockData.timestamp>this.BLOCK_CACHE_TIME) {
|
|
69
|
+
cachedBlockData = this.fetchAndSaveBlockTime(blockTag);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
return cachedBlockData.blockTime;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
}
|