@atomiqlabs/chain-starknet 4.0.0-dev.12 → 4.0.0-dev.13

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 -201
  2. package/dist/index.d.ts +38 -38
  3. package/dist/index.js +54 -54
  4. package/dist/starknet/StarknetChainType.d.ts +13 -13
  5. package/dist/starknet/StarknetChainType.js +2 -2
  6. package/dist/starknet/StarknetInitializer.d.ts +27 -27
  7. package/dist/starknet/StarknetInitializer.js +69 -69
  8. package/dist/starknet/btcrelay/BtcRelayAbi.d.ts +250 -250
  9. package/dist/starknet/btcrelay/BtcRelayAbi.js +341 -341
  10. package/dist/starknet/btcrelay/StarknetBtcRelay.d.ts +186 -186
  11. package/dist/starknet/btcrelay/StarknetBtcRelay.js +379 -379
  12. package/dist/starknet/btcrelay/headers/StarknetBtcHeader.d.ts +31 -31
  13. package/dist/starknet/btcrelay/headers/StarknetBtcHeader.js +74 -74
  14. package/dist/starknet/btcrelay/headers/StarknetBtcStoredHeader.d.ts +51 -51
  15. package/dist/starknet/btcrelay/headers/StarknetBtcStoredHeader.js +113 -113
  16. package/dist/starknet/chain/StarknetAction.d.ts +19 -19
  17. package/dist/starknet/chain/StarknetAction.js +73 -73
  18. package/dist/starknet/chain/StarknetChainInterface.d.ts +52 -52
  19. package/dist/starknet/chain/StarknetChainInterface.js +91 -91
  20. package/dist/starknet/chain/StarknetModule.d.ts +14 -14
  21. package/dist/starknet/chain/StarknetModule.js +13 -13
  22. package/dist/starknet/chain/modules/ERC20Abi.d.ts +755 -755
  23. package/dist/starknet/chain/modules/ERC20Abi.js +1032 -1032
  24. package/dist/starknet/chain/modules/StarknetAccounts.d.ts +6 -6
  25. package/dist/starknet/chain/modules/StarknetAccounts.js +24 -24
  26. package/dist/starknet/chain/modules/StarknetAddresses.d.ts +9 -9
  27. package/dist/starknet/chain/modules/StarknetAddresses.js +26 -26
  28. package/dist/starknet/chain/modules/StarknetBlocks.d.ts +20 -20
  29. package/dist/starknet/chain/modules/StarknetBlocks.js +64 -64
  30. package/dist/starknet/chain/modules/StarknetEvents.d.ts +44 -44
  31. package/dist/starknet/chain/modules/StarknetEvents.js +88 -88
  32. package/dist/starknet/chain/modules/StarknetFees.d.ts +77 -77
  33. package/dist/starknet/chain/modules/StarknetFees.js +114 -114
  34. package/dist/starknet/chain/modules/StarknetSignatures.d.ts +29 -29
  35. package/dist/starknet/chain/modules/StarknetSignatures.js +72 -72
  36. package/dist/starknet/chain/modules/StarknetTokens.d.ts +69 -69
  37. package/dist/starknet/chain/modules/StarknetTokens.js +102 -98
  38. package/dist/starknet/chain/modules/StarknetTransactions.d.ts +93 -93
  39. package/dist/starknet/chain/modules/StarknetTransactions.js +261 -260
  40. package/dist/starknet/contract/StarknetContractBase.d.ts +13 -13
  41. package/dist/starknet/contract/StarknetContractBase.js +20 -16
  42. package/dist/starknet/contract/StarknetContractModule.d.ts +8 -8
  43. package/dist/starknet/contract/StarknetContractModule.js +11 -11
  44. package/dist/starknet/contract/modules/StarknetContractEvents.d.ts +51 -51
  45. package/dist/starknet/contract/modules/StarknetContractEvents.js +97 -97
  46. package/dist/starknet/events/StarknetChainEvents.d.ts +21 -21
  47. package/dist/starknet/events/StarknetChainEvents.js +52 -52
  48. package/dist/starknet/events/StarknetChainEventsBrowser.d.ts +89 -90
  49. package/dist/starknet/events/StarknetChainEventsBrowser.js +296 -294
  50. package/dist/starknet/provider/RpcProviderWithRetries.d.ts +21 -21
  51. package/dist/starknet/provider/RpcProviderWithRetries.js +32 -32
  52. package/dist/starknet/spv_swap/SpvVaultContractAbi.d.ts +488 -488
  53. package/dist/starknet/spv_swap/SpvVaultContractAbi.js +656 -656
  54. package/dist/starknet/spv_swap/StarknetSpvVaultContract.d.ts +66 -66
  55. package/dist/starknet/spv_swap/StarknetSpvVaultContract.js +382 -382
  56. package/dist/starknet/spv_swap/StarknetSpvVaultData.d.ts +49 -49
  57. package/dist/starknet/spv_swap/StarknetSpvVaultData.js +145 -145
  58. package/dist/starknet/spv_swap/StarknetSpvWithdrawalData.d.ts +25 -25
  59. package/dist/starknet/spv_swap/StarknetSpvWithdrawalData.js +72 -72
  60. package/dist/starknet/swaps/EscrowManagerAbi.d.ts +431 -431
  61. package/dist/starknet/swaps/EscrowManagerAbi.js +583 -583
  62. package/dist/starknet/swaps/StarknetSwapContract.d.ts +191 -191
  63. package/dist/starknet/swaps/StarknetSwapContract.js +424 -424
  64. package/dist/starknet/swaps/StarknetSwapData.d.ts +74 -74
  65. package/dist/starknet/swaps/StarknetSwapData.js +325 -325
  66. package/dist/starknet/swaps/StarknetSwapModule.d.ts +10 -10
  67. package/dist/starknet/swaps/StarknetSwapModule.js +11 -11
  68. package/dist/starknet/swaps/handlers/IHandler.d.ts +13 -13
  69. package/dist/starknet/swaps/handlers/IHandler.js +2 -2
  70. package/dist/starknet/swaps/handlers/claim/ClaimHandlers.d.ts +13 -13
  71. package/dist/starknet/swaps/handlers/claim/ClaimHandlers.js +13 -13
  72. package/dist/starknet/swaps/handlers/claim/HashlockClaimHandler.d.ts +21 -21
  73. package/dist/starknet/swaps/handlers/claim/HashlockClaimHandler.js +44 -44
  74. package/dist/starknet/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.d.ts +24 -24
  75. package/dist/starknet/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.js +48 -48
  76. package/dist/starknet/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.d.ts +25 -25
  77. package/dist/starknet/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.js +40 -40
  78. package/dist/starknet/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.d.ts +20 -20
  79. package/dist/starknet/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.js +30 -30
  80. package/dist/starknet/swaps/handlers/claim/btc/IBitcoinClaimHandler.d.ts +45 -45
  81. package/dist/starknet/swaps/handlers/claim/btc/IBitcoinClaimHandler.js +52 -52
  82. package/dist/starknet/swaps/handlers/refund/TimelockRefundHandler.d.ts +17 -17
  83. package/dist/starknet/swaps/handlers/refund/TimelockRefundHandler.js +27 -27
  84. package/dist/starknet/swaps/modules/StarknetLpVault.d.ts +69 -69
  85. package/dist/starknet/swaps/modules/StarknetLpVault.js +122 -122
  86. package/dist/starknet/swaps/modules/StarknetSwapClaim.d.ts +53 -53
  87. package/dist/starknet/swaps/modules/StarknetSwapClaim.js +100 -100
  88. package/dist/starknet/swaps/modules/StarknetSwapInit.d.ts +94 -87
  89. package/dist/starknet/swaps/modules/StarknetSwapInit.js +235 -225
  90. package/dist/starknet/swaps/modules/StarknetSwapRefund.d.ts +62 -62
  91. package/dist/starknet/swaps/modules/StarknetSwapRefund.js +128 -128
  92. package/dist/starknet/wallet/StarknetKeypairWallet.d.ts +7 -7
  93. package/dist/starknet/wallet/StarknetKeypairWallet.js +35 -30
  94. package/dist/starknet/wallet/StarknetSigner.d.ts +12 -12
  95. package/dist/starknet/wallet/StarknetSigner.js +47 -46
  96. package/dist/utils/Utils.d.ts +37 -37
  97. package/dist/utils/Utils.js +260 -261
  98. package/package.json +43 -37
  99. package/src/index.ts +47 -47
  100. package/src/starknet/StarknetChainType.ts +28 -28
  101. package/src/starknet/StarknetInitializer.ts +108 -108
  102. package/src/starknet/btcrelay/BtcRelayAbi.ts +338 -338
  103. package/src/starknet/btcrelay/StarknetBtcRelay.ts +494 -494
  104. package/src/starknet/btcrelay/headers/StarknetBtcHeader.ts +100 -100
  105. package/src/starknet/btcrelay/headers/StarknetBtcStoredHeader.ts +141 -141
  106. package/src/starknet/chain/StarknetAction.ts +85 -85
  107. package/src/starknet/chain/StarknetChainInterface.ts +149 -149
  108. package/src/starknet/chain/StarknetModule.ts +19 -19
  109. package/src/starknet/chain/modules/ERC20Abi.ts +1029 -1029
  110. package/src/starknet/chain/modules/StarknetAccounts.ts +25 -25
  111. package/src/starknet/chain/modules/StarknetAddresses.ts +22 -22
  112. package/src/starknet/chain/modules/StarknetBlocks.ts +75 -74
  113. package/src/starknet/chain/modules/StarknetEvents.ts +104 -104
  114. package/src/starknet/chain/modules/StarknetFees.ts +154 -154
  115. package/src/starknet/chain/modules/StarknetSignatures.ts +91 -91
  116. package/src/starknet/chain/modules/StarknetTokens.ts +120 -116
  117. package/src/starknet/chain/modules/StarknetTransactions.ts +285 -283
  118. package/src/starknet/contract/StarknetContractBase.ts +30 -26
  119. package/src/starknet/contract/StarknetContractModule.ts +16 -16
  120. package/src/starknet/contract/modules/StarknetContractEvents.ts +134 -134
  121. package/src/starknet/events/StarknetChainEvents.ts +67 -67
  122. package/src/starknet/events/StarknetChainEventsBrowser.ts +411 -411
  123. package/src/starknet/provider/RpcProviderWithRetries.ts +43 -43
  124. package/src/starknet/spv_swap/SpvVaultContractAbi.ts +656 -656
  125. package/src/starknet/spv_swap/StarknetSpvVaultContract.ts +483 -483
  126. package/src/starknet/spv_swap/StarknetSpvVaultData.ts +195 -195
  127. package/src/starknet/spv_swap/StarknetSpvWithdrawalData.ts +79 -79
  128. package/src/starknet/swaps/EscrowManagerAbi.ts +582 -582
  129. package/src/starknet/swaps/StarknetSwapContract.ts +647 -647
  130. package/src/starknet/swaps/StarknetSwapData.ts +455 -455
  131. package/src/starknet/swaps/StarknetSwapModule.ts +17 -17
  132. package/src/starknet/swaps/handlers/IHandler.ts +20 -20
  133. package/src/starknet/swaps/handlers/claim/ClaimHandlers.ts +23 -23
  134. package/src/starknet/swaps/handlers/claim/HashlockClaimHandler.ts +53 -53
  135. package/src/starknet/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.ts +73 -73
  136. package/src/starknet/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.ts +67 -67
  137. package/src/starknet/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.ts +50 -50
  138. package/src/starknet/swaps/handlers/claim/btc/IBitcoinClaimHandler.ts +102 -102
  139. package/src/starknet/swaps/handlers/refund/TimelockRefundHandler.ts +38 -38
  140. package/src/starknet/swaps/modules/StarknetLpVault.ts +147 -147
  141. package/src/starknet/swaps/modules/StarknetSwapClaim.ts +141 -141
  142. package/src/starknet/swaps/modules/StarknetSwapInit.ts +300 -287
  143. package/src/starknet/swaps/modules/StarknetSwapRefund.ts +196 -196
  144. package/src/starknet/wallet/StarknetKeypairWallet.ts +44 -39
  145. package/src/starknet/wallet/StarknetSigner.ts +55 -55
  146. package/src/utils/Utils.ts +251 -252
@@ -1,284 +1,286 @@
1
- import {StarknetModule} from "../StarknetModule";
2
- import {
3
- Call,
4
- DeployAccountContractPayload, DeployAccountContractTransaction,
5
- Invocation, InvocationsSignerDetails,
6
- BigNumberish
7
- } from "starknet";
8
- import {StarknetSigner} from "../../wallet/StarknetSigner";
9
- import {calculateHash, timeoutPromise, toBigInt, toHex, tryWithRetries} from "../../../utils/Utils";
10
-
11
- export type StarknetTx = ({
12
- type: "DEPLOY_ACCOUNT",
13
- tx: DeployAccountContractPayload,
14
- signed?: DeployAccountContractTransaction
15
- } | {
16
- type: "INVOKE",
17
- tx: Array<Call>,
18
- signed?: Invocation
19
- }) & {
20
- details: InvocationsSignerDetails & {maxFee?: BigNumberish},
21
- txId?: string
22
- };
23
-
24
- const MAX_UNCONFIRMED_TXS = 25;
25
-
26
- export class StarknetTransactions extends StarknetModule {
27
-
28
- private readonly latestConfirmedNonces: {[address: string]: bigint} = {};
29
-
30
- private cbkBeforeTxSigned: (tx: StarknetTx) => Promise<void>;
31
-
32
- /**
33
- * Waits for transaction confirmation using WS subscription and occasional HTTP polling, also re-sends
34
- * the transaction at regular interval
35
- *
36
- * @param tx starknet transaction to wait for confirmation for & keep re-sending until it confirms
37
- * @param abortSignal signal to abort waiting for tx confirmation
38
- * @private
39
- */
40
- private async confirmTransaction(tx: StarknetTx, abortSignal?: AbortSignal) {
41
- let state = "pending";
42
- while(state==="pending" || state==="not_found") {
43
- await timeoutPromise(3000, abortSignal);
44
- state = await this._getTxIdStatus(tx.txId);
45
- if(state==="not_found" && tx.signed!=null) await this.sendSignedTransaction(tx).catch(e => {
46
- if(e.baseError?.code === 59) return; //Transaction already in the mempool
47
- this.logger.error("confirmTransaction(): Error on transaction re-send: ", e);
48
- });
49
- }
50
- if(state==="rejected") throw new Error("Transaction rejected!");
51
- const nextAccountNonce = toBigInt(tx.details.nonce) + 1n;
52
- const currentNonce = this.latestConfirmedNonces[tx.details.walletAddress];
53
- if(currentNonce==null || nextAccountNonce > currentNonce) {
54
- this.latestConfirmedNonces[tx.details.walletAddress] = nextAccountNonce;
55
- }
56
- if(state==="reverted") throw new Error("Transaction reverted!");
57
- }
58
-
59
- /**
60
- * Prepares starknet transactions, checks if the account is deployed, assigns nonces if needed & calls beforeTxSigned callback
61
- *
62
- * @param signer
63
- * @param txs
64
- * @private
65
- */
66
- private async prepareTransactions(signer: StarknetSigner, txs: StarknetTx[]): Promise<void> {
67
- let nonce: bigint = await signer.getNonce();
68
- const latestConfirmedNonce = this.latestConfirmedNonces[signer.getAddress()];
69
- if(latestConfirmedNonce!=null && latestConfirmedNonce > nonce) {
70
- this.logger.debug("prepareTransactions(): Using nonce from local cache!");
71
- nonce = latestConfirmedNonce;
72
- }
73
- if(nonce===BigInt(0) && signer.isWalletAccount()) {
74
- //Just increment the nonce by one and hope the wallet is smart enough to deploy account first
75
- nonce = BigInt(1);
76
- }
77
- const deployPayload = await signer.checkAndGetDeployPayload(nonce);
78
- if(deployPayload!=null) {
79
- txs.unshift(await this.root.Accounts.getAccountDeployTransaction(deployPayload));
80
- }
81
-
82
- for(let i=0;i<txs.length;i++) {
83
- const tx = txs[i];
84
- if(tx.details.nonce!=null) nonce = BigInt(tx.details.nonce); //Take the nonce from last tx
85
- if(nonce==null) nonce = BigInt(await this.root.provider.getNonceForAddress(signer.getAddress())); //Fetch the nonce
86
- if(tx.details.nonce==null) tx.details.nonce = nonce;
87
-
88
- this.logger.debug("sendAndConfirm(): transaction prepared ("+(i+1)+"/"+txs.length+"), nonce: "+tx.details.nonce);
89
-
90
- nonce += BigInt(1);
91
-
92
- if(this.cbkBeforeTxSigned!=null) await this.cbkBeforeTxSigned(tx);
93
- }
94
- }
95
-
96
- /**
97
- * Sends out a signed transaction to the RPC
98
- *
99
- * @param tx Starknet tx to send
100
- * @param onBeforePublish a callback called before every transaction is published
101
- * @param signer
102
- * @private
103
- */
104
- private async sendSignedTransaction(
105
- tx: StarknetTx,
106
- onBeforePublish?: (txId: string, rawTx: string) => Promise<void>,
107
- signer?: StarknetSigner
108
- ): Promise<string> {
109
- if(onBeforePublish!=null) await onBeforePublish(tx.txId, await this.serializeTx(tx));
110
- this.logger.debug("sendSignedTransaction(): sending transaction: ", tx.txId);
111
-
112
- if(tx.signed==null) {
113
- let txHash: string;
114
- switch(tx.type) {
115
- case "INVOKE":
116
- txHash = (await signer.account.execute(tx.tx, tx.details)).transaction_hash;
117
- break;
118
- case "DEPLOY_ACCOUNT":
119
- txHash = (await signer.account.deployAccount(tx.tx, tx.details)).transaction_hash;
120
- break;
121
- default:
122
- throw new Error("Unsupported tx type!");
123
- }
124
- tx.txId = txHash;
125
- return txHash;
126
- }
127
-
128
- let txResult: string;
129
- switch(tx.type) {
130
- case "INVOKE":
131
- txResult = await this.provider.channel.invoke(tx.signed, tx.details).then(res => res.transaction_hash);
132
- break;
133
- case "DEPLOY_ACCOUNT":
134
- txResult = await this.provider.channel.deployAccount(tx.signed, tx.details).then((res: any) => res.transaction_hash);
135
- break;
136
- default:
137
- throw new Error("Unsupported tx type!");
138
- }
139
- if(tx.txId!==txResult) this.logger.warn("sendSignedTransaction(): sent tx hash not matching the precomputed hash!");
140
- this.logger.info("sendSignedTransaction(): tx sent, expected txHash: "+tx.txId+", txHash: "+txResult);
141
- return txResult;
142
- }
143
-
144
- /**
145
- * Prepares, signs , sends (in parallel or sequentially) & optionally waits for confirmation
146
- * of a batch of starknet transactions
147
- *
148
- * @param signer
149
- * @param txs transactions to send
150
- * @param waitForConfirmation whether to wait for transaction confirmations (this also makes sure the transactions
151
- * are re-sent at regular intervals)
152
- * @param abortSignal abort signal to abort waiting for transaction confirmations
153
- * @param parallel whether the send all the transaction at once in parallel or sequentially (such that transactions
154
- * are executed in order)
155
- * @param onBeforePublish a callback called before every transaction is published
156
- */
157
- public async sendAndConfirm(signer: StarknetSigner, txs: StarknetTx[], waitForConfirmation?: boolean, abortSignal?: AbortSignal, parallel?: boolean, onBeforePublish?: (txId: string, rawTx: string) => Promise<void>): Promise<string[]> {
158
- await this.prepareTransactions(signer, txs);
159
- if(!signer.isWalletAccount()) {
160
- for(let i=0;i<txs.length;i++) {
161
- const tx = txs[i];
162
- switch(tx.type) {
163
- case "INVOKE":
164
- tx.signed = await signer.account.buildInvocation(tx.tx, tx.details);
165
- calculateHash(tx);
166
- break;
167
- case "DEPLOY_ACCOUNT":
168
- tx.signed = await signer.account.buildAccountDeployPayload(tx.tx, tx.details);
169
- calculateHash(tx);
170
- break;
171
- default:
172
- throw new Error("Unsupported tx type!");
173
- }
174
- this.logger.debug("sendAndConfirm(): transaction signed ("+(i+1)+"/"+txs.length+"): "+tx.txId);
175
- }
176
- }
177
-
178
- this.logger.debug("sendAndConfirm(): sending transactions, count: "+txs.length+
179
- " waitForConfirmation: "+waitForConfirmation+" parallel: "+parallel);
180
-
181
- const txIds: string[] = [];
182
- if(parallel) {
183
- let promises: Promise<void>[] = [];
184
- for(let i=0;i<txs.length;i++) {
185
- const signedTx = txs[i];
186
- const txId = await this.sendSignedTransaction(signedTx, onBeforePublish, signer);
187
- if(waitForConfirmation) promises.push(this.confirmTransaction(signedTx, abortSignal));
188
- txIds.push(txId);
189
- this.logger.debug("sendAndConfirm(): transaction sent ("+(i+1)+"/"+txs.length+"): "+signedTx.txId);
190
- if(promises.length >= MAX_UNCONFIRMED_TXS) {
191
- await Promise.all(promises);
192
- promises = [];
193
- }
194
- }
195
- if(promises.length>0) await Promise.all(promises);
196
- } else {
197
- for(let i=0;i<txs.length;i++) {
198
- const signedTx = txs[i];
199
- const txId = await this.sendSignedTransaction(signedTx, onBeforePublish, signer);
200
- const confirmPromise = this.confirmTransaction(signedTx, abortSignal);
201
- this.logger.debug("sendAndConfirm(): transaction sent ("+(i+1)+"/"+txs.length+"): "+signedTx.txId);
202
- //Don't await the last promise when !waitForConfirmation
203
- if(i<txs.length-1 || waitForConfirmation) await confirmPromise;
204
- txIds.push(txId);
205
- }
206
- }
207
-
208
- this.logger.info("sendAndConfirm(): sent transactions, count: "+txs.length+
209
- " waitForConfirmation: "+waitForConfirmation+" parallel: "+parallel);
210
-
211
- return txIds;
212
- }
213
-
214
- /**
215
- * Serializes the solana transaction, saves the transaction, signers & last valid blockheight
216
- *
217
- * @param tx
218
- */
219
- public serializeTx(tx: StarknetTx): Promise<string> {
220
- return Promise.resolve(JSON.stringify(tx, (key, value) => {
221
- if(typeof(value)==="bigint") return toHex(value);
222
- return value;
223
- }));
224
- }
225
-
226
- /**
227
- * Deserializes saved solana transaction, extracting the transaction, signers & last valid blockheight
228
- *
229
- * @param txData
230
- */
231
- public deserializeTx(txData: string): Promise<StarknetTx> {
232
- return Promise.resolve(JSON.parse(txData));
233
- }
234
-
235
- /**
236
- * Gets the status of the raw starknet transaction
237
- *
238
- * @param tx
239
- */
240
- public async getTxStatus(tx: string): Promise<"pending" | "success" | "not_found" | "reverted"> {
241
- const parsedTx: StarknetTx = await this.deserializeTx(tx);
242
- return await this.getTxIdStatus(parsedTx.txId);
243
- }
244
-
245
- /**
246
- * Gets the status of the starknet transaction with a specific txId
247
- *
248
- * @param txId
249
- */
250
- public async _getTxIdStatus(txId: string): Promise<"pending" | "success" | "not_found" | "reverted" | "rejected"> {
251
- const status = await this.provider.getTransactionStatus(txId).catch(e => {
252
- if(e.message!=null && e.message.includes("29: Transaction hash not found")) return null;
253
- throw e;
254
- });
255
- if(status==null) return "not_found";
256
- if(status.finality_status==="RECEIVED") return "pending";
257
- if(status.finality_status==="REJECTED") return "rejected";
258
- if(status.execution_status==="SUCCEEDED"){
259
- return "success";
260
- }
261
- return "reverted";
262
- }
263
-
264
- /**
265
- * Gets the status of the starknet transaction with a specific txId
266
- *
267
- * @param txId
268
- */
269
- public async getTxIdStatus(txId: string): Promise<"pending" | "success" | "not_found" | "reverted"> {
270
- const status = await this._getTxIdStatus(txId);
271
- if(status==="rejected") return "reverted";
272
- return status;
273
- }
274
-
275
- public onBeforeTxSigned(callback: (tx: StarknetTx) => Promise<void>): void {
276
- this.cbkBeforeTxSigned = callback;
277
- }
278
-
279
- public offBeforeTxSigned(callback: (tx: StarknetTx) => Promise<void>): boolean {
280
- this.cbkBeforeTxSigned = null;
281
- return true;
282
- }
283
-
1
+ import {StarknetModule} from "../StarknetModule";
2
+ import {
3
+ Call,
4
+ DeployAccountContractPayload, DeployAccountContractTransaction,
5
+ Invocation, InvocationsSignerDetails,
6
+ BigNumberish,
7
+ ETransactionStatus,
8
+ ETransactionExecutionStatus
9
+ } from "starknet";
10
+ import {StarknetSigner} from "../../wallet/StarknetSigner";
11
+ import {calculateHash, timeoutPromise, toBigInt, toHex, tryWithRetries} from "../../../utils/Utils";
12
+
13
+ export type StarknetTx = ({
14
+ type: "DEPLOY_ACCOUNT",
15
+ tx: DeployAccountContractPayload,
16
+ signed?: DeployAccountContractTransaction
17
+ } | {
18
+ type: "INVOKE",
19
+ tx: Array<Call>,
20
+ signed?: Invocation
21
+ }) & {
22
+ details: InvocationsSignerDetails & {maxFee?: BigNumberish},
23
+ txId?: string
24
+ };
25
+
26
+ const MAX_UNCONFIRMED_TXS = 25;
27
+
28
+ export class StarknetTransactions extends StarknetModule {
29
+
30
+ private readonly latestConfirmedNonces: {[address: string]: bigint} = {};
31
+
32
+ private cbkBeforeTxSigned: (tx: StarknetTx) => Promise<void>;
33
+
34
+ /**
35
+ * Waits for transaction confirmation using WS subscription and occasional HTTP polling, also re-sends
36
+ * the transaction at regular interval
37
+ *
38
+ * @param tx starknet transaction to wait for confirmation for & keep re-sending until it confirms
39
+ * @param abortSignal signal to abort waiting for tx confirmation
40
+ * @private
41
+ */
42
+ private async confirmTransaction(tx: StarknetTx, abortSignal?: AbortSignal) {
43
+ let state = "pending";
44
+ while(state==="pending" || state==="not_found") {
45
+ await timeoutPromise(3000, abortSignal);
46
+ state = await this._getTxIdStatus(tx.txId);
47
+ if(state==="not_found" && tx.signed!=null) await this.sendSignedTransaction(tx).catch(e => {
48
+ if(e.baseError?.code === 59) return; //Transaction already in the mempool
49
+ this.logger.error("confirmTransaction(): Error on transaction re-send: ", e);
50
+ });
51
+ }
52
+ if(state==="rejected") throw new Error("Transaction rejected!");
53
+ const nextAccountNonce = toBigInt(tx.details.nonce) + 1n;
54
+ const currentNonce = this.latestConfirmedNonces[tx.details.walletAddress];
55
+ if(currentNonce==null || nextAccountNonce > currentNonce) {
56
+ this.latestConfirmedNonces[tx.details.walletAddress] = nextAccountNonce;
57
+ }
58
+ if(state==="reverted") throw new Error("Transaction reverted!");
59
+ }
60
+
61
+ /**
62
+ * Prepares starknet transactions, checks if the account is deployed, assigns nonces if needed & calls beforeTxSigned callback
63
+ *
64
+ * @param signer
65
+ * @param txs
66
+ * @private
67
+ */
68
+ private async prepareTransactions(signer: StarknetSigner, txs: StarknetTx[]): Promise<void> {
69
+ let nonce: bigint = await signer.getNonce();
70
+ const latestConfirmedNonce = this.latestConfirmedNonces[signer.getAddress()];
71
+ if(latestConfirmedNonce!=null && latestConfirmedNonce > nonce) {
72
+ this.logger.debug("prepareTransactions(): Using nonce from local cache!");
73
+ nonce = latestConfirmedNonce;
74
+ }
75
+ if(nonce===BigInt(0) && signer.isWalletAccount()) {
76
+ //Just increment the nonce by one and hope the wallet is smart enough to deploy account first
77
+ nonce = BigInt(1);
78
+ }
79
+ const deployPayload = await signer.checkAndGetDeployPayload(nonce);
80
+ if(deployPayload!=null) {
81
+ txs.unshift(await this.root.Accounts.getAccountDeployTransaction(deployPayload));
82
+ }
83
+
84
+ for(let i=0;i<txs.length;i++) {
85
+ const tx = txs[i];
86
+ if(tx.details.nonce!=null) nonce = BigInt(tx.details.nonce); //Take the nonce from last tx
87
+ if(nonce==null) nonce = BigInt(await this.root.provider.getNonceForAddress(signer.getAddress())); //Fetch the nonce
88
+ if(tx.details.nonce==null) tx.details.nonce = nonce;
89
+
90
+ this.logger.debug("sendAndConfirm(): transaction prepared ("+(i+1)+"/"+txs.length+"), nonce: "+tx.details.nonce);
91
+
92
+ nonce += BigInt(1);
93
+
94
+ if(this.cbkBeforeTxSigned!=null) await this.cbkBeforeTxSigned(tx);
95
+ }
96
+ }
97
+
98
+ /**
99
+ * Sends out a signed transaction to the RPC
100
+ *
101
+ * @param tx Starknet tx to send
102
+ * @param onBeforePublish a callback called before every transaction is published
103
+ * @param signer
104
+ * @private
105
+ */
106
+ private async sendSignedTransaction(
107
+ tx: StarknetTx,
108
+ onBeforePublish?: (txId: string, rawTx: string) => Promise<void>,
109
+ signer?: StarknetSigner
110
+ ): Promise<string> {
111
+ if(onBeforePublish!=null) await onBeforePublish(tx.txId, await this.serializeTx(tx));
112
+ this.logger.debug("sendSignedTransaction(): sending transaction: ", tx.txId);
113
+
114
+ if(tx.signed==null) {
115
+ let txHash: string;
116
+ switch(tx.type) {
117
+ case "INVOKE":
118
+ txHash = (await signer.account.execute(tx.tx, tx.details)).transaction_hash;
119
+ break;
120
+ case "DEPLOY_ACCOUNT":
121
+ txHash = (await signer.account.deployAccount(tx.tx, tx.details)).transaction_hash;
122
+ break;
123
+ default:
124
+ throw new Error("Unsupported tx type!");
125
+ }
126
+ tx.txId = txHash;
127
+ return txHash;
128
+ }
129
+
130
+ let txResult: string;
131
+ switch(tx.type) {
132
+ case "INVOKE":
133
+ txResult = await this.provider.channel.invoke(tx.signed, tx.details).then(res => res.transaction_hash);
134
+ break;
135
+ case "DEPLOY_ACCOUNT":
136
+ txResult = await this.provider.channel.deployAccount(tx.signed, tx.details).then((res: any) => res.transaction_hash);
137
+ break;
138
+ default:
139
+ throw new Error("Unsupported tx type!");
140
+ }
141
+ if(tx.txId!==txResult) this.logger.warn("sendSignedTransaction(): sent tx hash not matching the precomputed hash!");
142
+ this.logger.info("sendSignedTransaction(): tx sent, expected txHash: "+tx.txId+", txHash: "+txResult);
143
+ return txResult;
144
+ }
145
+
146
+ /**
147
+ * Prepares, signs , sends (in parallel or sequentially) & optionally waits for confirmation
148
+ * of a batch of starknet transactions
149
+ *
150
+ * @param signer
151
+ * @param txs transactions to send
152
+ * @param waitForConfirmation whether to wait for transaction confirmations (this also makes sure the transactions
153
+ * are re-sent at regular intervals)
154
+ * @param abortSignal abort signal to abort waiting for transaction confirmations
155
+ * @param parallel whether the send all the transaction at once in parallel or sequentially (such that transactions
156
+ * are executed in order)
157
+ * @param onBeforePublish a callback called before every transaction is published
158
+ */
159
+ public async sendAndConfirm(signer: StarknetSigner, txs: StarknetTx[], waitForConfirmation?: boolean, abortSignal?: AbortSignal, parallel?: boolean, onBeforePublish?: (txId: string, rawTx: string) => Promise<void>): Promise<string[]> {
160
+ await this.prepareTransactions(signer, txs);
161
+ if(!signer.isWalletAccount()) {
162
+ for(let i=0;i<txs.length;i++) {
163
+ const tx = txs[i];
164
+ switch(tx.type) {
165
+ case "INVOKE":
166
+ tx.signed = await signer.account.buildInvocation(tx.tx, tx.details);
167
+ calculateHash(tx);
168
+ break;
169
+ case "DEPLOY_ACCOUNT":
170
+ tx.signed = await signer.account.buildAccountDeployPayload(tx.tx, tx.details);
171
+ calculateHash(tx);
172
+ break;
173
+ default:
174
+ throw new Error("Unsupported tx type!");
175
+ }
176
+ this.logger.debug("sendAndConfirm(): transaction signed ("+(i+1)+"/"+txs.length+"): "+tx.txId);
177
+ }
178
+ }
179
+
180
+ this.logger.debug("sendAndConfirm(): sending transactions, count: "+txs.length+
181
+ " waitForConfirmation: "+waitForConfirmation+" parallel: "+parallel);
182
+
183
+ const txIds: string[] = [];
184
+ if(parallel) {
185
+ let promises: Promise<void>[] = [];
186
+ for(let i=0;i<txs.length;i++) {
187
+ const signedTx = txs[i];
188
+ const txId = await this.sendSignedTransaction(signedTx, onBeforePublish, signer);
189
+ if(waitForConfirmation) promises.push(this.confirmTransaction(signedTx, abortSignal));
190
+ txIds.push(txId);
191
+ this.logger.debug("sendAndConfirm(): transaction sent ("+(i+1)+"/"+txs.length+"): "+signedTx.txId);
192
+ if(promises.length >= MAX_UNCONFIRMED_TXS) {
193
+ await Promise.all(promises);
194
+ promises = [];
195
+ }
196
+ }
197
+ if(promises.length>0) await Promise.all(promises);
198
+ } else {
199
+ for(let i=0;i<txs.length;i++) {
200
+ const signedTx = txs[i];
201
+ const txId = await this.sendSignedTransaction(signedTx, onBeforePublish, signer);
202
+ const confirmPromise = this.confirmTransaction(signedTx, abortSignal);
203
+ this.logger.debug("sendAndConfirm(): transaction sent ("+(i+1)+"/"+txs.length+"): "+signedTx.txId);
204
+ //Don't await the last promise when !waitForConfirmation
205
+ if(i<txs.length-1 || waitForConfirmation) await confirmPromise;
206
+ txIds.push(txId);
207
+ }
208
+ }
209
+
210
+ this.logger.info("sendAndConfirm(): sent transactions, count: "+txs.length+
211
+ " waitForConfirmation: "+waitForConfirmation+" parallel: "+parallel);
212
+
213
+ return txIds;
214
+ }
215
+
216
+ /**
217
+ * Serializes the solana transaction, saves the transaction, signers & last valid blockheight
218
+ *
219
+ * @param tx
220
+ */
221
+ public serializeTx(tx: StarknetTx): Promise<string> {
222
+ return Promise.resolve(JSON.stringify(tx, (key, value) => {
223
+ if(typeof(value)==="bigint") return toHex(value);
224
+ return value;
225
+ }));
226
+ }
227
+
228
+ /**
229
+ * Deserializes saved solana transaction, extracting the transaction, signers & last valid blockheight
230
+ *
231
+ * @param txData
232
+ */
233
+ public deserializeTx(txData: string): Promise<StarknetTx> {
234
+ return Promise.resolve(JSON.parse(txData));
235
+ }
236
+
237
+ /**
238
+ * Gets the status of the raw starknet transaction
239
+ *
240
+ * @param tx
241
+ */
242
+ public async getTxStatus(tx: string): Promise<"pending" | "success" | "not_found" | "reverted"> {
243
+ const parsedTx: StarknetTx = await this.deserializeTx(tx);
244
+ return await this.getTxIdStatus(parsedTx.txId);
245
+ }
246
+
247
+ /**
248
+ * Gets the status of the starknet transaction with a specific txId
249
+ *
250
+ * @param txId
251
+ */
252
+ public async _getTxIdStatus(txId: string): Promise<"pending" | "success" | "not_found" | "reverted" | "rejected"> {
253
+ const status = await this.provider.getTransactionStatus(txId).catch(e => {
254
+ if(e.message!=null && e.message.includes("29: Transaction hash not found")) return null;
255
+ throw e;
256
+ });
257
+ if(status==null) return "not_found";
258
+ if(status.finality_status===ETransactionStatus.RECEIVED) return "pending";
259
+ if(status.finality_status===ETransactionStatus.REJECTED) return "rejected";
260
+ if(status.execution_status===ETransactionExecutionStatus.SUCCEEDED){
261
+ return "success";
262
+ }
263
+ return "reverted";
264
+ }
265
+
266
+ /**
267
+ * Gets the status of the starknet transaction with a specific txId
268
+ *
269
+ * @param txId
270
+ */
271
+ public async getTxIdStatus(txId: string): Promise<"pending" | "success" | "not_found" | "reverted"> {
272
+ const status = await this._getTxIdStatus(txId);
273
+ if(status==="rejected") return "reverted";
274
+ return status;
275
+ }
276
+
277
+ public onBeforeTxSigned(callback: (tx: StarknetTx) => Promise<void>): void {
278
+ this.cbkBeforeTxSigned = callback;
279
+ }
280
+
281
+ public offBeforeTxSigned(callback: (tx: StarknetTx) => Promise<void>): boolean {
282
+ this.cbkBeforeTxSigned = null;
283
+ return true;
284
+ }
285
+
284
286
  }
@@ -1,26 +1,30 @@
1
- import {StarknetChainInterface} from "../chain/StarknetChainInterface";
2
- import {Contract, TypedContractV2} from "starknet";
3
- import {Abi} from "abi-wan-kanabi";
4
- import {StarknetContractEvents} from "./modules/StarknetContractEvents";
5
-
6
- /**
7
- * Base class providing program specific utilities
8
- */
9
- export class StarknetContractBase<T extends Abi> {
10
-
11
- contract: TypedContractV2<T>;
12
-
13
- public readonly Events: StarknetContractEvents<T>;
14
- public readonly Chain: StarknetChainInterface;
15
-
16
- constructor(
17
- chainInterface: StarknetChainInterface,
18
- contractAddress: string,
19
- contractAbi: T
20
- ) {
21
- this.Chain = chainInterface;
22
- this.contract = new Contract(contractAbi, contractAddress, chainInterface.provider).typedv2(contractAbi);
23
- this.Events = new StarknetContractEvents(chainInterface, this, contractAbi);
24
- }
25
-
26
- }
1
+ import {StarknetChainInterface} from "../chain/StarknetChainInterface";
2
+ import {Contract, TypedContractV2} from "starknet";
3
+ import {Abi} from "abi-wan-kanabi";
4
+ import {StarknetContractEvents} from "./modules/StarknetContractEvents";
5
+
6
+ /**
7
+ * Base class providing program specific utilities
8
+ */
9
+ export class StarknetContractBase<T extends Abi> {
10
+
11
+ contract: TypedContractV2<T>;
12
+
13
+ public readonly Events: StarknetContractEvents<T>;
14
+ public readonly Chain: StarknetChainInterface;
15
+
16
+ constructor(
17
+ chainInterface: StarknetChainInterface,
18
+ contractAddress: string,
19
+ contractAbi: T
20
+ ) {
21
+ this.Chain = chainInterface;
22
+ this.contract = new Contract({
23
+ abi: contractAbi,
24
+ address: contractAddress,
25
+ providerOrAccount: chainInterface.provider
26
+ }).typedv2(contractAbi);
27
+ this.Events = new StarknetContractEvents(chainInterface, this, contractAbi);
28
+ }
29
+
30
+ }