Skip to content

Commit ad65589

Browse files
committed
solana: Added client-side tx rebroadcast logic
1 parent 5abaa75 commit ad65589

File tree

2 files changed

+65
-28
lines changed

2 files changed

+65
-28
lines changed

platforms/solana/src/platform.ts

+59-24
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,9 @@ import type {
2424
Commitment,
2525
ConnectionConfig,
2626
ParsedAccountData,
27+
RpcResponseAndContext,
2728
SendOptions,
29+
SignatureResult,
2830
} from '@solana/web3.js';
2931
import { Connection, PublicKey } from '@solana/web3.js';
3032
import { SolanaAddress, SolanaZeroAddress } from './address.js';
@@ -176,41 +178,74 @@ export class SolanaPlatform<N extends Network>
176178
stxns: SignedTx[],
177179
opts?: SendOptions,
178180
): 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-
191181
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)),
202183
);
203184

185+
const txhashes = results.map((r) => r.signature);
186+
204187
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);
207190

208191
if (erroredTxs.length > 0)
209192
throw new Error(`Failed to confirm transaction: ${erroredTxs}`);
210193

211194
return txhashes;
212195
}
213196

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+
214249
static async latestBlock(
215250
rpc: Connection,
216251
commitment?: Commitment,

platforms/solana/src/signer.ts

+6-4
Original file line numberDiff line numberDiff line change
@@ -203,25 +203,27 @@ export class SolanaSendSigner<
203203
for (let i = 0; i < this._maxResubmits; i++) {
204204
try {
205205
if (isVersionedTransaction(transaction)) {
206-
if (priorityFeeIx) {
206+
if (priorityFeeIx && i === 0) {
207207
const msg = TransactionMessage.decompile(transaction.message);
208208
msg.instructions.push(...priorityFeeIx);
209209
transaction.message = msg.compileToV0Message();
210210
}
211211
transaction.message.recentBlockhash = blockhash;
212212
transaction.sign([this._keypair, ...(extraSigners ?? [])]);
213213
} else {
214-
if (priorityFeeIx) transaction.add(...priorityFeeIx);
214+
if (priorityFeeIx && i === 0) transaction.add(...priorityFeeIx);
215215
transaction.recentBlockhash = blockhash;
216+
transaction.lastValidBlockHeight = lastValidBlockHeight;
216217
transaction.partialSign(this._keypair, ...(extraSigners ?? []));
217218
}
218219

219220
if (this._debug) console.log('Submitting transactions ');
220-
const txid = await this._rpc.sendRawTransaction(
221+
const { signature } = await SolanaPlatform.sendTxWithRetry(
222+
this._rpc,
221223
transaction.serialize(),
222224
this._sendOpts,
223225
);
224-
txids.push(txid);
226+
txids.push(signature);
225227
break;
226228
} catch (e) {
227229
// No point checking if retryable if we're on the last retry

0 commit comments

Comments
 (0)