@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.
Files changed (146) hide show
  1. package/LICENSE +201 -0
  2. package/dist/chains/citrea/CitreaChainType.d.ts +13 -0
  3. package/dist/chains/citrea/CitreaChainType.js +2 -0
  4. package/dist/chains/citrea/CitreaInitializer.d.ts +30 -0
  5. package/dist/chains/citrea/CitreaInitializer.js +120 -0
  6. package/dist/evm/btcrelay/BtcRelayAbi.d.ts +198 -0
  7. package/dist/evm/btcrelay/BtcRelayAbi.js +261 -0
  8. package/dist/evm/btcrelay/BtcRelayTypechain.d.ts +172 -0
  9. package/dist/evm/btcrelay/BtcRelayTypechain.js +2 -0
  10. package/dist/evm/btcrelay/EVMBtcRelay.d.ts +188 -0
  11. package/dist/evm/btcrelay/EVMBtcRelay.js +419 -0
  12. package/dist/evm/btcrelay/headers/EVMBtcHeader.d.ts +33 -0
  13. package/dist/evm/btcrelay/headers/EVMBtcHeader.js +84 -0
  14. package/dist/evm/btcrelay/headers/EVMBtcStoredHeader.d.ts +56 -0
  15. package/dist/evm/btcrelay/headers/EVMBtcStoredHeader.js +123 -0
  16. package/dist/evm/chain/EVMChainInterface.d.ts +51 -0
  17. package/dist/evm/chain/EVMChainInterface.js +90 -0
  18. package/dist/evm/chain/EVMModule.d.ts +9 -0
  19. package/dist/evm/chain/EVMModule.js +13 -0
  20. package/dist/evm/chain/modules/ERC20Abi.d.ts +168 -0
  21. package/dist/evm/chain/modules/ERC20Abi.js +225 -0
  22. package/dist/evm/chain/modules/EVMAddresses.d.ts +9 -0
  23. package/dist/evm/chain/modules/EVMAddresses.js +26 -0
  24. package/dist/evm/chain/modules/EVMBlocks.d.ts +20 -0
  25. package/dist/evm/chain/modules/EVMBlocks.js +64 -0
  26. package/dist/evm/chain/modules/EVMEvents.d.ts +36 -0
  27. package/dist/evm/chain/modules/EVMEvents.js +122 -0
  28. package/dist/evm/chain/modules/EVMFees.d.ts +35 -0
  29. package/dist/evm/chain/modules/EVMFees.js +73 -0
  30. package/dist/evm/chain/modules/EVMSignatures.d.ts +29 -0
  31. package/dist/evm/chain/modules/EVMSignatures.js +68 -0
  32. package/dist/evm/chain/modules/EVMTokens.d.ts +49 -0
  33. package/dist/evm/chain/modules/EVMTokens.js +105 -0
  34. package/dist/evm/chain/modules/EVMTransactions.d.ts +89 -0
  35. package/dist/evm/chain/modules/EVMTransactions.js +216 -0
  36. package/dist/evm/contract/EVMContractBase.d.ts +22 -0
  37. package/dist/evm/contract/EVMContractBase.js +34 -0
  38. package/dist/evm/contract/EVMContractModule.d.ts +8 -0
  39. package/dist/evm/contract/EVMContractModule.js +11 -0
  40. package/dist/evm/contract/modules/EVMContractEvents.d.ts +42 -0
  41. package/dist/evm/contract/modules/EVMContractEvents.js +75 -0
  42. package/dist/evm/events/EVMChainEvents.d.ts +22 -0
  43. package/dist/evm/events/EVMChainEvents.js +67 -0
  44. package/dist/evm/events/EVMChainEventsBrowser.d.ts +86 -0
  45. package/dist/evm/events/EVMChainEventsBrowser.js +294 -0
  46. package/dist/evm/spv_swap/EVMSpvVaultContract.d.ts +64 -0
  47. package/dist/evm/spv_swap/EVMSpvVaultContract.js +410 -0
  48. package/dist/evm/spv_swap/EVMSpvVaultData.d.ts +38 -0
  49. package/dist/evm/spv_swap/EVMSpvVaultData.js +159 -0
  50. package/dist/evm/spv_swap/EVMSpvWithdrawalData.d.ts +19 -0
  51. package/dist/evm/spv_swap/EVMSpvWithdrawalData.js +55 -0
  52. package/dist/evm/spv_swap/SpvVaultContractAbi.d.ts +91 -0
  53. package/dist/evm/spv_swap/SpvVaultContractAbi.js +849 -0
  54. package/dist/evm/spv_swap/SpvVaultContractTypechain.d.ts +450 -0
  55. package/dist/evm/spv_swap/SpvVaultContractTypechain.js +2 -0
  56. package/dist/evm/swaps/EVMSwapContract.d.ts +192 -0
  57. package/dist/evm/swaps/EVMSwapContract.js +373 -0
  58. package/dist/evm/swaps/EVMSwapData.d.ts +64 -0
  59. package/dist/evm/swaps/EVMSwapData.js +254 -0
  60. package/dist/evm/swaps/EVMSwapModule.d.ts +9 -0
  61. package/dist/evm/swaps/EVMSwapModule.js +11 -0
  62. package/dist/evm/swaps/EscrowManagerAbi.d.ts +120 -0
  63. package/dist/evm/swaps/EscrowManagerAbi.js +985 -0
  64. package/dist/evm/swaps/EscrowManagerTypechain.d.ts +475 -0
  65. package/dist/evm/swaps/EscrowManagerTypechain.js +2 -0
  66. package/dist/evm/swaps/handlers/IHandler.d.ts +13 -0
  67. package/dist/evm/swaps/handlers/IHandler.js +2 -0
  68. package/dist/evm/swaps/handlers/claim/ClaimHandlers.d.ts +10 -0
  69. package/dist/evm/swaps/handlers/claim/ClaimHandlers.js +13 -0
  70. package/dist/evm/swaps/handlers/claim/HashlockClaimHandler.d.ts +20 -0
  71. package/dist/evm/swaps/handlers/claim/HashlockClaimHandler.js +39 -0
  72. package/dist/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.d.ts +24 -0
  73. package/dist/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.js +59 -0
  74. package/dist/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.d.ts +25 -0
  75. package/dist/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.js +51 -0
  76. package/dist/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.d.ts +21 -0
  77. package/dist/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.js +28 -0
  78. package/dist/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.d.ts +48 -0
  79. package/dist/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.js +63 -0
  80. package/dist/evm/swaps/handlers/refund/TimelockRefundHandler.d.ts +17 -0
  81. package/dist/evm/swaps/handlers/refund/TimelockRefundHandler.js +28 -0
  82. package/dist/evm/swaps/modules/EVMLpVault.d.ts +69 -0
  83. package/dist/evm/swaps/modules/EVMLpVault.js +131 -0
  84. package/dist/evm/swaps/modules/EVMSwapClaim.d.ts +53 -0
  85. package/dist/evm/swaps/modules/EVMSwapClaim.js +101 -0
  86. package/dist/evm/swaps/modules/EVMSwapInit.d.ts +88 -0
  87. package/dist/evm/swaps/modules/EVMSwapInit.js +241 -0
  88. package/dist/evm/swaps/modules/EVMSwapRefund.d.ts +62 -0
  89. package/dist/evm/swaps/modules/EVMSwapRefund.js +132 -0
  90. package/dist/evm/typechain/common.d.ts +50 -0
  91. package/dist/evm/typechain/common.js +2 -0
  92. package/dist/evm/wallet/EVMSigner.d.ts +9 -0
  93. package/dist/evm/wallet/EVMSigner.js +16 -0
  94. package/dist/index.d.ts +37 -0
  95. package/dist/index.js +53 -0
  96. package/dist/utils/Utils.d.ts +15 -0
  97. package/dist/utils/Utils.js +71 -0
  98. package/package.json +37 -0
  99. package/src/chains/citrea/CitreaChainType.ts +28 -0
  100. package/src/chains/citrea/CitreaInitializer.ts +167 -0
  101. package/src/evm/btcrelay/BtcRelayAbi.ts +258 -0
  102. package/src/evm/btcrelay/BtcRelayTypechain.ts +371 -0
  103. package/src/evm/btcrelay/EVMBtcRelay.ts +517 -0
  104. package/src/evm/btcrelay/headers/EVMBtcHeader.ts +110 -0
  105. package/src/evm/btcrelay/headers/EVMBtcStoredHeader.ts +153 -0
  106. package/src/evm/chain/EVMChainInterface.ts +157 -0
  107. package/src/evm/chain/EVMModule.ts +21 -0
  108. package/src/evm/chain/modules/ERC20Abi.ts +222 -0
  109. package/src/evm/chain/modules/EVMAddresses.ts +24 -0
  110. package/src/evm/chain/modules/EVMBlocks.ts +75 -0
  111. package/src/evm/chain/modules/EVMEvents.ts +139 -0
  112. package/src/evm/chain/modules/EVMFees.ts +105 -0
  113. package/src/evm/chain/modules/EVMSignatures.ts +76 -0
  114. package/src/evm/chain/modules/EVMTokens.ts +115 -0
  115. package/src/evm/chain/modules/EVMTransactions.ts +246 -0
  116. package/src/evm/contract/EVMContractBase.ts +63 -0
  117. package/src/evm/contract/EVMContractModule.ts +16 -0
  118. package/src/evm/contract/modules/EVMContractEvents.ts +102 -0
  119. package/src/evm/events/EVMChainEvents.ts +81 -0
  120. package/src/evm/events/EVMChainEventsBrowser.ts +390 -0
  121. package/src/evm/spv_swap/EVMSpvVaultContract.ts +533 -0
  122. package/src/evm/spv_swap/EVMSpvVaultData.ts +201 -0
  123. package/src/evm/spv_swap/EVMSpvWithdrawalData.ts +70 -0
  124. package/src/evm/spv_swap/SpvVaultContractAbi.ts +846 -0
  125. package/src/evm/spv_swap/SpvVaultContractTypechain.ts +685 -0
  126. package/src/evm/swaps/EVMSwapContract.ts +590 -0
  127. package/src/evm/swaps/EVMSwapData.ts +367 -0
  128. package/src/evm/swaps/EVMSwapModule.ts +16 -0
  129. package/src/evm/swaps/EscrowManagerAbi.ts +982 -0
  130. package/src/evm/swaps/EscrowManagerTypechain.ts +723 -0
  131. package/src/evm/swaps/handlers/IHandler.ts +17 -0
  132. package/src/evm/swaps/handlers/claim/ClaimHandlers.ts +20 -0
  133. package/src/evm/swaps/handlers/claim/HashlockClaimHandler.ts +47 -0
  134. package/src/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.ts +82 -0
  135. package/src/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.ts +76 -0
  136. package/src/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.ts +46 -0
  137. package/src/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.ts +115 -0
  138. package/src/evm/swaps/handlers/refund/TimelockRefundHandler.ts +38 -0
  139. package/src/evm/swaps/modules/EVMLpVault.ts +153 -0
  140. package/src/evm/swaps/modules/EVMSwapClaim.ts +141 -0
  141. package/src/evm/swaps/modules/EVMSwapInit.ts +292 -0
  142. package/src/evm/swaps/modules/EVMSwapRefund.ts +198 -0
  143. package/src/evm/typechain/common.ts +131 -0
  144. package/src/evm/wallet/EVMSigner.ts +23 -0
  145. package/src/index.ts +44 -0
  146. 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
+ }