fix: retry transactions on AccountInUse errors
This commit is contained in:
parent
96c685eb5d
commit
90c9df15ef
|
@ -41,7 +41,11 @@ declare module '@solana/web3.js' {
|
|||
userdata: Buffer,
|
||||
}
|
||||
|
||||
declare export type SignatureStatus = 'Confirmed' | 'SignatureNotFound' | 'ProgramRuntimeError' | 'GenericFailure';
|
||||
declare export type SignatureStatus = 'Confirmed'
|
||||
| 'AccountInUse'
|
||||
| 'SignatureNotFound'
|
||||
| 'ProgramRuntimeError'
|
||||
| 'GenericFailure';
|
||||
|
||||
declare export class Connection {
|
||||
constructor(endpoint: string): Connection;
|
||||
|
|
|
@ -101,10 +101,11 @@ const ConfirmTransactionRpcResult = jsonRpcResult('boolean');
|
|||
* Expected JSON RPC response for the "getSignatureStatus" message
|
||||
*/
|
||||
const GetSignatureStatusRpcResult = jsonRpcResult(struct.enum([
|
||||
'AccountInUse',
|
||||
'Confirmed',
|
||||
'SignatureNotFound',
|
||||
'ProgramRuntimeError',
|
||||
'GenericFailure',
|
||||
'ProgramRuntimeError',
|
||||
'SignatureNotFound',
|
||||
]));
|
||||
|
||||
/**
|
||||
|
@ -152,7 +153,11 @@ type AccountInfo = {
|
|||
*
|
||||
* @typedef {string} SignatureStatus
|
||||
*/
|
||||
export type SignatureStatus = 'Confirmed' | 'SignatureNotFound' | 'ProgramRuntimeError' | 'GenericFailure';
|
||||
export type SignatureStatus = 'Confirmed'
|
||||
| 'AccountInUse'
|
||||
| 'SignatureNotFound'
|
||||
| 'ProgramRuntimeError'
|
||||
| 'GenericFailure';
|
||||
|
||||
/**
|
||||
* A connection to a fullnode JSON RPC endpoint
|
||||
|
|
|
@ -15,31 +15,36 @@ export async function sendAndConfirmTransaction(
|
|||
transaction: Transaction,
|
||||
runtimeErrorOk: boolean = false
|
||||
): Promise<void> {
|
||||
|
||||
let sendRetries = 3;
|
||||
for (;;) {
|
||||
const start = Date.now();
|
||||
const signature = await connection.sendTransaction(from, transaction);
|
||||
|
||||
// Wait up to a couple seconds for a confirmation
|
||||
let i = 4;
|
||||
let status = 'SignatureNotFound';
|
||||
let statusRetries = 4;
|
||||
for (;;) {
|
||||
const status = await connection.getSignatureStatus(signature);
|
||||
switch (status) {
|
||||
case 'Confirmed':
|
||||
return;
|
||||
case 'ProgramRuntimeError':
|
||||
if (runtimeErrorOk) return;
|
||||
//fall through
|
||||
case 'GenericError':
|
||||
default:
|
||||
throw new Error(`Transaction ${signature} failed (${status})`);
|
||||
case 'SignatureNotFound':
|
||||
status = await connection.getSignatureStatus(signature);
|
||||
if (status !== 'SignatureNotFound') {
|
||||
break;
|
||||
}
|
||||
|
||||
await sleep(500);
|
||||
if (--i < 0) {
|
||||
if (--statusRetries <= 0) {
|
||||
const duration = (Date.now() - start) / 1000;
|
||||
throw new Error(`Transaction '${signature}' was not confirmed in ${duration.toFixed(2)} seconds (${status})`);
|
||||
}
|
||||
}
|
||||
|
||||
if ( (status === 'Confirmed') ||
|
||||
(status === 'ProgramRuntimeError' && runtimeErrorOk) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (status !== 'AccountInUse' || --sendRetries <= 0) {
|
||||
throw new Error(`Transaction ${signature} failed (${status})`);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue