@@ -24,7 +24,9 @@ import type {
24
24
Commitment ,
25
25
ConnectionConfig ,
26
26
ParsedAccountData ,
27
+ RpcResponseAndContext ,
27
28
SendOptions ,
29
+ SignatureResult ,
28
30
} from '@solana/web3.js' ;
29
31
import { Connection , PublicKey } from '@solana/web3.js' ;
30
32
import { SolanaAddress , SolanaZeroAddress } from './address.js' ;
@@ -176,41 +178,74 @@ export class SolanaPlatform<N extends Network>
176
178
stxns : SignedTx [ ] ,
177
179
opts ?: SendOptions ,
178
180
) : Promise < TxHash [ ] > {
179
- const { blockhash, lastValidBlockHeight } = await this . latestBlock ( rpc ) ;
180
- const txhashes = await Promise . all (
181
- stxns . map ( ( stxn ) =>
182
- rpc . sendRawTransaction (
183
- stxn ,
184
- // Set the commitment level to match the rpc commitment level
185
- // otherwise, it defaults to finalized
186
- opts ?? { preflightCommitment : rpc . commitment } ,
187
- ) ,
188
- ) ,
189
- ) ;
190
-
191
181
const results = await Promise . all (
192
- txhashes . map ( ( signature ) => {
193
- return rpc . confirmTransaction (
194
- {
195
- signature,
196
- blockhash,
197
- lastValidBlockHeight,
198
- } ,
199
- rpc . commitment ,
200
- ) ;
201
- } ) ,
182
+ stxns . map ( ( stxn ) => this . sendTxWithRetry ( rpc , stxn , opts ) ) ,
202
183
) ;
203
184
185
+ const txhashes = results . map ( ( r ) => r . signature ) ;
186
+
204
187
const erroredTxs = results
205
- . filter ( ( result ) => result . value . err )
206
- . map ( ( result ) => result . value . err ) ;
188
+ . filter ( ( r ) => r . response . value . err )
189
+ . map ( ( r ) => r . response . value . err ) ;
207
190
208
191
if ( erroredTxs . length > 0 )
209
192
throw new Error ( `Failed to confirm transaction: ${ erroredTxs } ` ) ;
210
193
211
194
return txhashes ;
212
195
}
213
196
197
+ static async sendTxWithRetry (
198
+ rpc : Connection ,
199
+ tx : SignedTx ,
200
+ sendOpts : SendOptions = { } ,
201
+ retryInterval = 5000 ,
202
+ ) : Promise < {
203
+ signature : string ;
204
+ response : RpcResponseAndContext < SignatureResult > ;
205
+ } > {
206
+ const commitment = sendOpts . preflightCommitment ?? rpc . commitment ;
207
+ const signature = await rpc . sendRawTransaction ( tx , {
208
+ ...sendOpts ,
209
+ skipPreflight : false , // The first send should not skip preflight to catch any errors
210
+ maxRetries : 0 ,
211
+ preflightCommitment : commitment ,
212
+ } ) ;
213
+ // TODO: Use the lastValidBlockHeight that corresponds to the blockhash used in the transaction.
214
+ const { blockhash, lastValidBlockHeight } = await rpc . getLatestBlockhash ( ) ;
215
+ const confirmTransactionPromise = rpc . confirmTransaction (
216
+ {
217
+ signature,
218
+ blockhash,
219
+ lastValidBlockHeight,
220
+ } ,
221
+ commitment ,
222
+ ) ;
223
+ // This loop will break once the transaction has been confirmed or the block height is exceeded.
224
+ // An exception will be thrown if the block height is exceeded by the confirmTransactionPromise.
225
+ // The transaction will be resent if it hasn't been confirmed after the interval.
226
+ let confirmedTx : RpcResponseAndContext < SignatureResult > | null = null ;
227
+ while ( ! confirmedTx ) {
228
+ confirmedTx = await Promise . race ( [
229
+ confirmTransactionPromise ,
230
+ new Promise < null > ( ( resolve ) =>
231
+ setTimeout ( ( ) => {
232
+ resolve ( null ) ;
233
+ } , retryInterval ) ,
234
+ ) ,
235
+ ] ) ;
236
+ if ( confirmedTx ) {
237
+ break ;
238
+ }
239
+ await rpc . sendRawTransaction ( tx , {
240
+ ...sendOpts ,
241
+ skipPreflight : true ,
242
+ maxRetries : 0 ,
243
+ preflightCommitment : commitment ,
244
+ } ) ;
245
+ }
246
+ return { signature, response : confirmedTx } ;
247
+ }
248
+
214
249
static async latestBlock (
215
250
rpc : Connection ,
216
251
commitment ?: Commitment ,
0 commit comments