@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.
Files changed (185) hide show
  1. package/LICENSE +201 -0
  2. package/dist/chains/botanix/BotanixChainType.d.ts +14 -0
  3. package/dist/chains/botanix/BotanixChainType.js +2 -0
  4. package/dist/chains/botanix/BotanixInitializer.d.ts +30 -0
  5. package/dist/chains/botanix/BotanixInitializer.js +125 -0
  6. package/dist/chains/citrea/CitreaBtcRelay.d.ts +21 -0
  7. package/dist/chains/citrea/CitreaBtcRelay.js +43 -0
  8. package/dist/chains/citrea/CitreaChainType.d.ts +14 -0
  9. package/dist/chains/citrea/CitreaChainType.js +2 -0
  10. package/dist/chains/citrea/CitreaFees.d.ts +29 -0
  11. package/dist/chains/citrea/CitreaFees.js +67 -0
  12. package/dist/chains/citrea/CitreaInitializer.d.ts +30 -0
  13. package/dist/chains/citrea/CitreaInitializer.js +132 -0
  14. package/dist/chains/citrea/CitreaSpvVaultContract.d.ts +15 -0
  15. package/dist/chains/citrea/CitreaSpvVaultContract.js +74 -0
  16. package/dist/chains/citrea/CitreaSwapContract.d.ts +22 -0
  17. package/dist/chains/citrea/CitreaSwapContract.js +96 -0
  18. package/dist/chains/citrea/CitreaTokens.d.ts +9 -0
  19. package/dist/chains/citrea/CitreaTokens.js +20 -0
  20. package/dist/evm/btcrelay/BtcRelayAbi.d.ts +198 -0
  21. package/dist/evm/btcrelay/BtcRelayAbi.js +261 -0
  22. package/dist/evm/btcrelay/BtcRelayTypechain.d.ts +172 -0
  23. package/dist/evm/btcrelay/BtcRelayTypechain.js +2 -0
  24. package/dist/evm/btcrelay/EVMBtcRelay.d.ts +197 -0
  25. package/dist/evm/btcrelay/EVMBtcRelay.js +435 -0
  26. package/dist/evm/btcrelay/headers/EVMBtcHeader.d.ts +33 -0
  27. package/dist/evm/btcrelay/headers/EVMBtcHeader.js +84 -0
  28. package/dist/evm/btcrelay/headers/EVMBtcStoredHeader.d.ts +56 -0
  29. package/dist/evm/btcrelay/headers/EVMBtcStoredHeader.js +123 -0
  30. package/dist/evm/chain/EVMChainInterface.d.ts +60 -0
  31. package/dist/evm/chain/EVMChainInterface.js +107 -0
  32. package/dist/evm/chain/EVMModule.d.ts +9 -0
  33. package/dist/evm/chain/EVMModule.js +13 -0
  34. package/dist/evm/chain/modules/ERC20Abi.d.ts +168 -0
  35. package/dist/evm/chain/modules/ERC20Abi.js +225 -0
  36. package/dist/evm/chain/modules/EVMAddresses.d.ts +10 -0
  37. package/dist/evm/chain/modules/EVMAddresses.js +30 -0
  38. package/dist/evm/chain/modules/EVMBlocks.d.ts +27 -0
  39. package/dist/evm/chain/modules/EVMBlocks.js +73 -0
  40. package/dist/evm/chain/modules/EVMEvents.d.ts +46 -0
  41. package/dist/evm/chain/modules/EVMEvents.js +151 -0
  42. package/dist/evm/chain/modules/EVMFees.d.ts +36 -0
  43. package/dist/evm/chain/modules/EVMFees.js +74 -0
  44. package/dist/evm/chain/modules/EVMSignatures.d.ts +29 -0
  45. package/dist/evm/chain/modules/EVMSignatures.js +68 -0
  46. package/dist/evm/chain/modules/EVMTokens.d.ts +70 -0
  47. package/dist/evm/chain/modules/EVMTokens.js +142 -0
  48. package/dist/evm/chain/modules/EVMTransactions.d.ts +94 -0
  49. package/dist/evm/chain/modules/EVMTransactions.js +307 -0
  50. package/dist/evm/contract/EVMContractBase.d.ts +22 -0
  51. package/dist/evm/contract/EVMContractBase.js +34 -0
  52. package/dist/evm/contract/EVMContractModule.d.ts +8 -0
  53. package/dist/evm/contract/EVMContractModule.js +11 -0
  54. package/dist/evm/contract/modules/EVMContractEvents.d.ts +43 -0
  55. package/dist/evm/contract/modules/EVMContractEvents.js +76 -0
  56. package/dist/evm/events/EVMChainEvents.d.ts +22 -0
  57. package/dist/evm/events/EVMChainEvents.js +69 -0
  58. package/dist/evm/events/EVMChainEventsBrowser.d.ts +102 -0
  59. package/dist/evm/events/EVMChainEventsBrowser.js +412 -0
  60. package/dist/evm/providers/JsonRpcProviderWithRetries.d.ts +16 -0
  61. package/dist/evm/providers/JsonRpcProviderWithRetries.js +27 -0
  62. package/dist/evm/providers/ReconnectingWebSocketProvider.d.ts +22 -0
  63. package/dist/evm/providers/ReconnectingWebSocketProvider.js +91 -0
  64. package/dist/evm/providers/SocketProvider.d.ts +111 -0
  65. package/dist/evm/providers/SocketProvider.js +336 -0
  66. package/dist/evm/providers/WebSocketProviderWithRetries.d.ts +17 -0
  67. package/dist/evm/providers/WebSocketProviderWithRetries.js +23 -0
  68. package/dist/evm/spv_swap/EVMSpvVaultContract.d.ts +110 -0
  69. package/dist/evm/spv_swap/EVMSpvVaultContract.js +629 -0
  70. package/dist/evm/spv_swap/EVMSpvVaultData.d.ts +40 -0
  71. package/dist/evm/spv_swap/EVMSpvVaultData.js +184 -0
  72. package/dist/evm/spv_swap/EVMSpvWithdrawalData.d.ts +19 -0
  73. package/dist/evm/spv_swap/EVMSpvWithdrawalData.js +55 -0
  74. package/dist/evm/spv_swap/SpvVaultContractAbi.d.ts +91 -0
  75. package/dist/evm/spv_swap/SpvVaultContractAbi.js +849 -0
  76. package/dist/evm/spv_swap/SpvVaultContractTypechain.d.ts +450 -0
  77. package/dist/evm/spv_swap/SpvVaultContractTypechain.js +2 -0
  78. package/dist/evm/swaps/EVMSwapContract.d.ts +199 -0
  79. package/dist/evm/swaps/EVMSwapContract.js +394 -0
  80. package/dist/evm/swaps/EVMSwapData.d.ts +66 -0
  81. package/dist/evm/swaps/EVMSwapData.js +260 -0
  82. package/dist/evm/swaps/EVMSwapModule.d.ts +9 -0
  83. package/dist/evm/swaps/EVMSwapModule.js +11 -0
  84. package/dist/evm/swaps/EscrowManagerAbi.d.ts +120 -0
  85. package/dist/evm/swaps/EscrowManagerAbi.js +985 -0
  86. package/dist/evm/swaps/EscrowManagerTypechain.d.ts +475 -0
  87. package/dist/evm/swaps/EscrowManagerTypechain.js +2 -0
  88. package/dist/evm/swaps/handlers/IHandler.d.ts +13 -0
  89. package/dist/evm/swaps/handlers/IHandler.js +2 -0
  90. package/dist/evm/swaps/handlers/claim/ClaimHandlers.d.ts +10 -0
  91. package/dist/evm/swaps/handlers/claim/ClaimHandlers.js +13 -0
  92. package/dist/evm/swaps/handlers/claim/HashlockClaimHandler.d.ts +20 -0
  93. package/dist/evm/swaps/handlers/claim/HashlockClaimHandler.js +39 -0
  94. package/dist/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.d.ts +24 -0
  95. package/dist/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.js +59 -0
  96. package/dist/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.d.ts +25 -0
  97. package/dist/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.js +51 -0
  98. package/dist/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.d.ts +21 -0
  99. package/dist/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.js +28 -0
  100. package/dist/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.d.ts +48 -0
  101. package/dist/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.js +63 -0
  102. package/dist/evm/swaps/handlers/refund/TimelockRefundHandler.d.ts +17 -0
  103. package/dist/evm/swaps/handlers/refund/TimelockRefundHandler.js +28 -0
  104. package/dist/evm/swaps/modules/EVMLpVault.d.ts +69 -0
  105. package/dist/evm/swaps/modules/EVMLpVault.js +134 -0
  106. package/dist/evm/swaps/modules/EVMSwapClaim.d.ts +54 -0
  107. package/dist/evm/swaps/modules/EVMSwapClaim.js +137 -0
  108. package/dist/evm/swaps/modules/EVMSwapInit.d.ts +88 -0
  109. package/dist/evm/swaps/modules/EVMSwapInit.js +274 -0
  110. package/dist/evm/swaps/modules/EVMSwapRefund.d.ts +62 -0
  111. package/dist/evm/swaps/modules/EVMSwapRefund.js +167 -0
  112. package/dist/evm/typechain/common.d.ts +50 -0
  113. package/dist/evm/typechain/common.js +2 -0
  114. package/dist/evm/wallet/EVMBrowserSigner.d.ts +5 -0
  115. package/dist/evm/wallet/EVMBrowserSigner.js +11 -0
  116. package/dist/evm/wallet/EVMPersistentSigner.d.ts +29 -0
  117. package/dist/evm/wallet/EVMPersistentSigner.js +230 -0
  118. package/dist/evm/wallet/EVMSigner.d.ts +12 -0
  119. package/dist/evm/wallet/EVMSigner.js +28 -0
  120. package/dist/index.d.ts +44 -0
  121. package/dist/index.js +60 -0
  122. package/dist/utils/Utils.d.ts +19 -0
  123. package/dist/utils/Utils.js +98 -0
  124. package/package.json +39 -0
  125. package/src/chains/botanix/BotanixChainType.ts +30 -0
  126. package/src/chains/botanix/BotanixInitializer.ts +175 -0
  127. package/src/chains/citrea/CitreaBtcRelay.ts +58 -0
  128. package/src/chains/citrea/CitreaChainType.ts +30 -0
  129. package/src/chains/citrea/CitreaFees.ts +77 -0
  130. package/src/chains/citrea/CitreaInitializer.ts +182 -0
  131. package/src/chains/citrea/CitreaSpvVaultContract.ts +75 -0
  132. package/src/chains/citrea/CitreaSwapContract.ts +103 -0
  133. package/src/chains/citrea/CitreaTokens.ts +22 -0
  134. package/src/evm/btcrelay/BtcRelayAbi.ts +258 -0
  135. package/src/evm/btcrelay/BtcRelayTypechain.ts +371 -0
  136. package/src/evm/btcrelay/EVMBtcRelay.ts +537 -0
  137. package/src/evm/btcrelay/headers/EVMBtcHeader.ts +110 -0
  138. package/src/evm/btcrelay/headers/EVMBtcStoredHeader.ts +153 -0
  139. package/src/evm/chain/EVMChainInterface.ts +189 -0
  140. package/src/evm/chain/EVMModule.ts +21 -0
  141. package/src/evm/chain/modules/ERC20Abi.ts +222 -0
  142. package/src/evm/chain/modules/EVMAddresses.ts +29 -0
  143. package/src/evm/chain/modules/EVMBlocks.ts +86 -0
  144. package/src/evm/chain/modules/EVMEvents.ts +182 -0
  145. package/src/evm/chain/modules/EVMFees.ts +104 -0
  146. package/src/evm/chain/modules/EVMSignatures.ts +76 -0
  147. package/src/evm/chain/modules/EVMTokens.ts +155 -0
  148. package/src/evm/chain/modules/EVMTransactions.ts +346 -0
  149. package/src/evm/contract/EVMContractBase.ts +63 -0
  150. package/src/evm/contract/EVMContractModule.ts +16 -0
  151. package/src/evm/contract/modules/EVMContractEvents.ts +104 -0
  152. package/src/evm/events/EVMChainEvents.ts +82 -0
  153. package/src/evm/events/EVMChainEventsBrowser.ts +533 -0
  154. package/src/evm/providers/JsonRpcProviderWithRetries.ts +34 -0
  155. package/src/evm/providers/ReconnectingWebSocketProvider.ts +107 -0
  156. package/src/evm/providers/SocketProvider.ts +371 -0
  157. package/src/evm/providers/WebSocketProviderWithRetries.ts +35 -0
  158. package/src/evm/spv_swap/EVMSpvVaultContract.ts +779 -0
  159. package/src/evm/spv_swap/EVMSpvVaultData.ts +228 -0
  160. package/src/evm/spv_swap/EVMSpvWithdrawalData.ts +70 -0
  161. package/src/evm/spv_swap/SpvVaultContractAbi.ts +846 -0
  162. package/src/evm/spv_swap/SpvVaultContractTypechain.ts +685 -0
  163. package/src/evm/swaps/EVMSwapContract.ts +621 -0
  164. package/src/evm/swaps/EVMSwapData.ts +378 -0
  165. package/src/evm/swaps/EVMSwapModule.ts +16 -0
  166. package/src/evm/swaps/EscrowManagerAbi.ts +982 -0
  167. package/src/evm/swaps/EscrowManagerTypechain.ts +723 -0
  168. package/src/evm/swaps/handlers/IHandler.ts +17 -0
  169. package/src/evm/swaps/handlers/claim/ClaimHandlers.ts +20 -0
  170. package/src/evm/swaps/handlers/claim/HashlockClaimHandler.ts +47 -0
  171. package/src/evm/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.ts +82 -0
  172. package/src/evm/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.ts +76 -0
  173. package/src/evm/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.ts +46 -0
  174. package/src/evm/swaps/handlers/claim/btc/IBitcoinClaimHandler.ts +115 -0
  175. package/src/evm/swaps/handlers/refund/TimelockRefundHandler.ts +38 -0
  176. package/src/evm/swaps/modules/EVMLpVault.ts +155 -0
  177. package/src/evm/swaps/modules/EVMSwapClaim.ts +173 -0
  178. package/src/evm/swaps/modules/EVMSwapInit.ts +329 -0
  179. package/src/evm/swaps/modules/EVMSwapRefund.ts +229 -0
  180. package/src/evm/typechain/common.ts +131 -0
  181. package/src/evm/wallet/EVMBrowserSigner.ts +12 -0
  182. package/src/evm/wallet/EVMPersistentSigner.ts +307 -0
  183. package/src/evm/wallet/EVMSigner.ts +35 -0
  184. package/src/index.ts +53 -0
  185. 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
+ }