feat: update getSignatureStatus api

This commit is contained in:
Justin Starry
2020-03-26 21:37:45 +08:00
committed by Michael Vines
parent ce4820a85a
commit a571e3dc65
7 changed files with 128 additions and 67 deletions

11
web3.js/module.d.ts vendored
View File

@ -48,11 +48,10 @@ declare module '@solana/web3.js' {
export type Commitment = 'max' | 'recent'; export type Commitment = 'max' | 'recent';
export type SignatureStatusResult = SignatureSuccess | TransactionError;
export type SignatureStatus = { export type SignatureStatus = {
slot: number; slot: number;
status: SignatureSuccess | TransactionError; status: SignatureSuccess | TransactionError;
confirmations: number | null;
}; };
export type BlockhashAndFeeCalculator = { export type BlockhashAndFeeCalculator = {
@ -90,7 +89,7 @@ declare module '@solana/web3.js' {
fee: number; fee: number;
preBalances: Array<number>; preBalances: Array<number>;
postBalances: Array<number>; postBalances: Array<number>;
status?: SignatureStatusResult; status?: SignatureSuccess | TransactionError;
}; };
}>; }>;
}; };
@ -127,7 +126,7 @@ declare module '@solana/web3.js' {
) => void; ) => void;
export type SlotChangeCallback = (slotInfo: SlotInfo) => void; export type SlotChangeCallback = (slotInfo: SlotInfo) => void;
export type SignatureResultCallback = ( export type SignatureResultCallback = (
signatureResult: SignatureStatusResult, signatureResult: SignatureSuccess | TransactionError,
context: Context, context: Context,
) => void; ) => void;
@ -195,11 +194,11 @@ declare module '@solana/web3.js' {
getSignatureStatus( getSignatureStatus(
signature: TransactionSignature, signature: TransactionSignature,
commitment?: Commitment, commitment?: Commitment,
): Promise<SignatureStatus | null>; ): Promise<RpcResponseAndContext<SignatureStatus | null>>;
getSignatureStatusBatch( getSignatureStatusBatch(
signatures: Array<TransactionSignature>, signatures: Array<TransactionSignature>,
commitment?: Commitment, commitment?: Commitment,
): Promise<Array<SignatureStatus | null>>; ): Promise<RpcResponseAndContext<Array<SignatureStatus | null>>>;
getTransactionCount(commitment?: Commitment): Promise<number>; getTransactionCount(commitment?: Commitment): Promise<number>;
getTotalSupply(commitment?: Commitment): Promise<number>; getTotalSupply(commitment?: Commitment): Promise<number>;
getVersion(): Promise<Version>; getVersion(): Promise<Version>;

View File

@ -61,13 +61,10 @@ declare module '@solana/web3.js' {
declare export type Commitment = 'max' | 'recent'; declare export type Commitment = 'max' | 'recent';
declare export type SignatureStatusResult =
| SignatureSuccess
| TransactionError;
declare export type SignatureStatus = { declare export type SignatureStatus = {
slot: number, slot: number,
status: SignatureSuccess | TransactionError, status: SignatureSuccess | TransactionError,
confirmations: number | null,
}; };
declare export type BlockhashAndFeeCalculator = { declare export type BlockhashAndFeeCalculator = {
@ -105,7 +102,7 @@ declare module '@solana/web3.js' {
fee: number, fee: number,
preBalances: Array<number>, preBalances: Array<number>,
postBalances: Array<number>, postBalances: Array<number>,
status: ?SignatureStatusResult, status: SignatureSuccess | TransactionError | null,
}, },
}>, }>,
}; };
@ -142,7 +139,7 @@ declare module '@solana/web3.js' {
) => void; ) => void;
declare type SlotChangeCallback = (slotInfo: SlotInfo) => void; declare type SlotChangeCallback = (slotInfo: SlotInfo) => void;
declare type SignatureResultCallback = ( declare type SignatureResultCallback = (
signatureResult: SignatureStatusResult, signatureResult: SignatureSuccess | TransactionError,
context: Context, context: Context,
) => void; ) => void;
@ -210,11 +207,11 @@ declare module '@solana/web3.js' {
getSignatureStatus( getSignatureStatus(
signature: TransactionSignature, signature: TransactionSignature,
commitment: ?Commitment, commitment: ?Commitment,
): Promise<SignatureStatus | null>; ): Promise<RpcResponseAndContext<SignatureStatus | null>>;
getSignatureStatusBatch( getSignatureStatusBatch(
signatures: Array<TransactionSignature>, signatures: Array<TransactionSignature>,
commitment: ?Commitment, commitment: ?Commitment,
): Promise<Array<SignatureStatus | null>>; ): Promise<RpcResponseAndContext<Array<SignatureStatus | null>>>;
getTransactionCount(commitment: ?Commitment): Promise<number>; getTransactionCount(commitment: ?Commitment): Promise<number>;
getTotalSupply(commitment: ?Commitment): Promise<number>; getTotalSupply(commitment: ?Commitment): Promise<number>;
getVersion(): Promise<Version>; getVersion(): Promise<Version>;

View File

@ -241,7 +241,7 @@ type ConfirmedBlock = {
fee: number, fee: number,
preBalances: Array<number>, preBalances: Array<number>,
postBalances: Array<number>, postBalances: Array<number>,
status?: SignatureStatusResult, status?: SignatureSuccess | TransactionError,
}, },
}>, }>,
rewards: Array<{ rewards: Array<{
@ -471,12 +471,13 @@ const GetVoteAccounts = jsonRpcResult(
/** /**
* Expected JSON RPC response for the "getSignatureStatus" message * Expected JSON RPC response for the "getSignatureStatus" message
*/ */
const GetSignatureStatusRpcResult = jsonRpcResult( const GetSignatureStatusRpcResult = jsonRpcResultAndContext(
struct.array([ struct.array([
struct.union([ struct.union([
'null', 'null',
struct({ struct({
slot: 'number', slot: 'number',
confirmations: struct.union(['number', 'null']),
status: SignatureStatusResult, status: SignatureStatusResult,
}), }),
]), ]),
@ -662,7 +663,7 @@ type SlotSubscriptionInfo = {
* Callback function for signature notifications * Callback function for signature notifications
*/ */
export type SignatureResultCallback = ( export type SignatureResultCallback = (
signatureResult: SignatureStatusResult, signatureResult: SignatureSuccess | TransactionError,
context: Context, context: Context,
) => void; ) => void;
@ -698,10 +699,12 @@ export type TransactionError = {|
* *
* @typedef {Object} SignatureStatus * @typedef {Object} SignatureStatus
* @property {number} slot when the transaction was processed * @property {number} slot when the transaction was processed
* @property {SignatureStatus | TransactionError} status * @property {number | null} confirmations the number of blocks that have been confirmed and voted on in the fork containing `slot` (TODO)
* @property {SignatureStatus | TransactionError} status success or error
*/ */
export type SignatureStatus = { export type SignatureStatus = {
slot: number, slot: number,
confirmations: number | null,
status: SignatureSuccess | TransactionError, status: SignatureSuccess | TransactionError,
}; };
@ -989,10 +992,13 @@ export class Connection {
async getSignatureStatus( async getSignatureStatus(
signature: TransactionSignature, signature: TransactionSignature,
commitment: ?Commitment, commitment: ?Commitment,
): Promise<SignatureStatus | null> { ): Promise<RpcResponseAndContext<SignatureStatus | null>> {
const res = await this.getSignatureStatusBatch([signature], commitment); const {context, value} = await this.getSignatureStatusBatch(
assert(res.length === 1); [signature],
return res[0]; commitment,
);
assert(value.length === 1);
return {context, value: value[0]};
} }
/** /**
@ -1001,7 +1007,7 @@ export class Connection {
async getSignatureStatusBatch( async getSignatureStatusBatch(
signatures: Array<TransactionSignature>, signatures: Array<TransactionSignature>,
commitment: ?Commitment, commitment: ?Commitment,
): Promise<Array<SignatureStatus | null>> { ): Promise<RpcResponseAndContext<Array<SignatureStatus | null>>> {
const args = this._argsWithCommitment([signatures], commitment); const args = this._argsWithCommitment([signatures], commitment);
const unsafeRes = await this._rpcRequest('getSignatureStatus', args); const unsafeRes = await this._rpcRequest('getSignatureStatus', args);
const res = GetSignatureStatusRpcResult(unsafeRes); const res = GetSignatureStatusRpcResult(unsafeRes);

View File

@ -21,7 +21,7 @@ export async function sendAndConfirmRawTransaction(
let status = null; let status = null;
let statusRetries = 6; let statusRetries = 6;
for (;;) { for (;;) {
status = await connection.getSignatureStatus(signature, commitment); status = (await connection.getSignatureStatus(signature, commitment)).value;
if (status) { if (status) {
break; break;
} }

View File

@ -53,7 +53,8 @@ async function _sendAndConfirmTransaction(
let status = null; let status = null;
let statusRetries = 6; let statusRetries = 6;
for (;;) { for (;;) {
status = await connection.getSignatureStatus(signature, commitment); status = (await connection.getSignatureStatus(signature, commitment))
.value;
if (status) { if (status) {
break; break;
} }

View File

@ -108,12 +108,18 @@ test('get program accounts', async () => {
}, },
{ {
error: null, error: null,
result: [ result: {
{ context: {
slot: 0, slot: 11,
status: {Ok: null},
}, },
], value: [
{
slot: 0,
confirmations: 11,
status: {Ok: null},
},
],
},
}, },
]); ]);
let transaction = SystemProgram.assign({ let transaction = SystemProgram.assign({
@ -146,12 +152,18 @@ test('get program accounts', async () => {
}, },
{ {
error: null, error: null,
result: [ result: {
{ context: {
slot: 0, slot: 11,
status: {Ok: null},
}, },
], value: [
{
slot: 0,
confirmations: 11,
status: {Ok: null},
},
],
},
}, },
]); ]);
transaction = SystemProgram.assign({ transaction = SystemProgram.assign({
@ -1009,6 +1021,9 @@ test('transaction', async () => {
}, },
]); ]);
// Wait for one confirmation
await sleep(500);
let i = 0; let i = 0;
for (;;) { for (;;) {
if (await connection.confirmTransaction(signature)) { if (await connection.confirmTransaction(signature)) {
@ -1033,23 +1048,37 @@ test('transaction', async () => {
}, },
{ {
error: null, error: null,
result: [ result: {
{ context: {
slot: 0, slot: 11,
status: {Ok: null},
}, },
], value: [
{
slot: 0,
confirmations: 11,
status: {Ok: null},
},
],
},
}, },
]); ]);
const response = await connection.getSignatureStatus(signature); const response = (await connection.getSignatureStatus(signature)).value;
if (response !== null) { if (response === null) {
expect(typeof response.slot).toEqual('number');
expect(response.status).toEqual({Ok: null});
} else {
expect(response).not.toBeNull(); expect(response).not.toBeNull();
return;
} }
const responseConfirmations = response.confirmations;
if (typeof responseConfirmations !== 'number') {
expect(typeof responseConfirmations).toEqual('number');
return;
}
expect(response.status).toEqual({Ok: null});
expect(response.slot).toBeGreaterThanOrEqual(0);
expect(responseConfirmations).toBeGreaterThan(0);
const unprocessedSignature = const unprocessedSignature =
'8WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk'; '8WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk';
mockRpc.push([ mockRpc.push([
@ -1066,24 +1095,47 @@ test('transaction', async () => {
}, },
{ {
error: null, error: null,
result: [ result: {
{ context: {
slot: 0, slot: 11,
status: {Ok: null},
}, },
null, value: [
], {
slot: 0,
confirmations: 11,
status: {Ok: null},
},
null,
],
},
}, },
]); ]);
const responses = await connection.getSignatureStatusBatch([ const responses = (
signature, await connection.getSignatureStatusBatch([signature, unprocessedSignature])
unprocessedSignature, ).value;
]);
expect(responses.length).toEqual(2); expect(responses.length).toEqual(2);
expect(responses[0]).toEqual(response);
const firstResponse = responses[0];
expect(responses[1]).toBeNull(); expect(responses[1]).toBeNull();
if (firstResponse === null) {
expect(firstResponse).not.toBeNull();
return;
}
expect(firstResponse.slot).toBeGreaterThanOrEqual(response.slot);
expect(firstResponse.status).toEqual(response.status);
if (typeof firstResponse.confirmations !== 'number') {
expect(typeof firstResponse.confirmations).toEqual('number');
return;
}
expect(firstResponse.confirmations).toBeGreaterThanOrEqual(
responseConfirmations,
);
mockRpc.push([ mockRpc.push([
url, url,
{ {
@ -1181,7 +1233,7 @@ test('multi-instruction transaction', async () => {
await sleep(500); await sleep(500);
} }
const response = await connection.getSignatureStatus(signature); const response = (await connection.getSignatureStatus(signature)).value;
if (response !== null) { if (response !== null) {
expect(typeof response.slot).toEqual('number'); expect(typeof response.slot).toEqual('number');
expect(response.status).toEqual({Ok: null}); expect(response.status).toEqual({Ok: null});

View File

@ -156,20 +156,26 @@ test('transaction-payer', async () => {
}, },
{ {
error: null, error: null,
result: [ result: {
{ context: {
slot: 0, slot: 11,
status: {Ok: null},
}, },
], value: [
{
slot: 0,
confirmations: 11,
status: {Ok: null},
},
],
},
}, },
]); ]);
const response = await connection.getSignatureStatus(signature); const {value} = await connection.getSignatureStatus(signature);
if (response !== null) { if (value !== null) {
expect(typeof response.slot).toEqual('number'); expect(typeof value.slot).toEqual('number');
expect(response.status).toEqual({Ok: null}); expect(value.status).toEqual({Ok: null});
} else { } else {
expect(response).not.toBeNull(); expect(value).not.toBeNull();
} }
mockRpc.push([ mockRpc.push([