2590 lines
64 KiB
JavaScript
2590 lines
64 KiB
JavaScript
// @flow
|
|
import bs58 from 'bs58';
|
|
import {Token, u64} from '@solana/spl-token';
|
|
|
|
import {
|
|
Account,
|
|
Authorized,
|
|
Connection,
|
|
SystemProgram,
|
|
Transaction,
|
|
sendAndConfirmTransaction,
|
|
LAMPORTS_PER_SOL,
|
|
Lockup,
|
|
PublicKey,
|
|
StakeProgram,
|
|
} from '../src';
|
|
import {DEFAULT_TICKS_PER_SLOT, NUM_TICKS_PER_SECOND} from '../src/timing';
|
|
import {mockRpc, mockRpcEnabled} from './__mocks__/node-fetch';
|
|
import {mockGetRecentBlockhash} from './mockrpc/get-recent-blockhash';
|
|
import {url} from './url';
|
|
import {sleep} from '../src/util/sleep';
|
|
import {BLOCKHASH_CACHE_TIMEOUT_MS} from '../src/connection';
|
|
import type {TransactionSignature} from '../src/transaction';
|
|
import type {SignatureStatus, TransactionError} from '../src/connection';
|
|
import {mockConfirmTransaction} from './mockrpc/confirm-transaction';
|
|
import {mockRpcSocket} from './__mocks__/rpc-websockets';
|
|
|
|
// Testing tokens and blockhash cache each take around 30s to complete
|
|
jest.setTimeout(40000);
|
|
|
|
const errorMessage = 'Invalid';
|
|
const errorResponse = {
|
|
error: {
|
|
code: -32602,
|
|
message: errorMessage,
|
|
},
|
|
result: undefined,
|
|
};
|
|
|
|
const verifySignatureStatus = (
|
|
status: SignatureStatus | null,
|
|
err?: TransactionError,
|
|
): SignatureStatus => {
|
|
if (status === null) {
|
|
expect(status).not.toBeNull();
|
|
throw new Error(); // unreachable
|
|
}
|
|
|
|
const expectedErr = err || null;
|
|
expect(status.err).toEqual(expectedErr);
|
|
expect(status.slot).toBeGreaterThanOrEqual(0);
|
|
if (expectedErr !== null) return status;
|
|
|
|
const confirmations = status.confirmations;
|
|
if (typeof confirmations === 'number') {
|
|
expect(confirmations).toBeGreaterThanOrEqual(0);
|
|
} else {
|
|
expect(confirmations).toBeNull();
|
|
}
|
|
return status;
|
|
};
|
|
|
|
test('get account info - not found', async () => {
|
|
const account = new Account();
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getAccountInfo',
|
|
params: [account.publicKey.toBase58(), {encoding: 'base64'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: null,
|
|
},
|
|
},
|
|
]);
|
|
|
|
expect(await connection.getAccountInfo(account.publicKey)).toBeNull();
|
|
|
|
if (!mockRpcEnabled) {
|
|
expect(
|
|
(await connection.getParsedAccountInfo(account.publicKey)).value,
|
|
).toBeNull();
|
|
}
|
|
});
|
|
|
|
test('get program accounts', async () => {
|
|
const connection = new Connection(url, 'recent');
|
|
const account0 = new Account();
|
|
const account1 = new Account();
|
|
const programId = new Account();
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'requestAirdrop',
|
|
params: [account0.publicKey.toBase58(), LAMPORTS_PER_SOL],
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'2WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'requestAirdrop',
|
|
params: [account1.publicKey.toBase58(), 0.5 * LAMPORTS_PER_SOL],
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'2WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
await connection.requestAirdrop(account0.publicKey, LAMPORTS_PER_SOL);
|
|
await connection.requestAirdrop(account1.publicKey, 0.5 * LAMPORTS_PER_SOL);
|
|
|
|
mockGetRecentBlockhash('max');
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
let transaction = new Transaction().add(
|
|
SystemProgram.assign({
|
|
accountPubkey: account0.publicKey,
|
|
programId: programId.publicKey,
|
|
}),
|
|
);
|
|
|
|
mockConfirmTransaction(
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
);
|
|
await sendAndConfirmTransaction(connection, transaction, [account0], {
|
|
commitment: 'single',
|
|
skipPreflight: true,
|
|
});
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
transaction = new Transaction().add(
|
|
SystemProgram.assign({
|
|
accountPubkey: account1.publicKey,
|
|
programId: programId.publicKey,
|
|
}),
|
|
);
|
|
|
|
mockConfirmTransaction(
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
);
|
|
await sendAndConfirmTransaction(connection, transaction, [account1], {
|
|
commitment: 'single',
|
|
skipPreflight: true,
|
|
});
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getFeeCalculatorForBlockhash',
|
|
params: [transaction.recentBlockhash, {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: {
|
|
feeCalculator: {
|
|
lamportsPerSignature: 42,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
]);
|
|
|
|
if (!transaction.recentBlockhash) {
|
|
expect(transaction.recentBlockhash).toBeTruthy();
|
|
return;
|
|
}
|
|
|
|
const feeCalculator = (
|
|
await connection.getFeeCalculatorForBlockhash(transaction.recentBlockhash)
|
|
).value;
|
|
|
|
if (feeCalculator === null) {
|
|
expect(feeCalculator).not.toBeNull();
|
|
return;
|
|
}
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getProgramAccounts',
|
|
params: [
|
|
programId.publicKey.toBase58(),
|
|
{commitment: 'recent', encoding: 'base64'},
|
|
],
|
|
},
|
|
{
|
|
error: null,
|
|
result: [
|
|
{
|
|
account: {
|
|
data: ['', 'base64'],
|
|
executable: false,
|
|
lamports: LAMPORTS_PER_SOL - feeCalculator.lamportsPerSignature,
|
|
owner: programId.publicKey.toBase58(),
|
|
rentEpoch: 20,
|
|
},
|
|
pubkey: account0.publicKey.toBase58(),
|
|
},
|
|
{
|
|
account: {
|
|
data: ['', 'base64'],
|
|
executable: false,
|
|
lamports:
|
|
0.5 * LAMPORTS_PER_SOL - feeCalculator.lamportsPerSignature,
|
|
owner: programId.publicKey.toBase58(),
|
|
rentEpoch: 20,
|
|
},
|
|
pubkey: account1.publicKey.toBase58(),
|
|
},
|
|
],
|
|
},
|
|
]);
|
|
const programAccounts = await connection.getProgramAccounts(
|
|
programId.publicKey,
|
|
);
|
|
expect(programAccounts.length).toBe(2);
|
|
|
|
programAccounts.forEach(function (element) {
|
|
if (element.pubkey.equals(account0.publicKey)) {
|
|
expect(element.account.lamports).toBe(
|
|
LAMPORTS_PER_SOL - feeCalculator.lamportsPerSignature,
|
|
);
|
|
} else if (element.pubkey.equals(account1.publicKey)) {
|
|
expect(element.account.lamports).toBe(
|
|
0.5 * LAMPORTS_PER_SOL - feeCalculator.lamportsPerSignature,
|
|
);
|
|
} else {
|
|
expect(element.pubkey.equals(account1.publicKey)).toBe(true);
|
|
}
|
|
});
|
|
|
|
if (!mockRpcEnabled) {
|
|
const programAccounts = await connection.getParsedProgramAccounts(
|
|
programId.publicKey,
|
|
);
|
|
expect(programAccounts.length).toBe(2);
|
|
|
|
programAccounts.forEach(function (element) {
|
|
if (element.pubkey.equals(account0.publicKey)) {
|
|
expect(element.account.lamports).toBe(
|
|
LAMPORTS_PER_SOL - feeCalculator.lamportsPerSignature,
|
|
);
|
|
} else if (element.pubkey.equals(account1.publicKey)) {
|
|
expect(element.account.lamports).toBe(
|
|
0.5 * LAMPORTS_PER_SOL - feeCalculator.lamportsPerSignature,
|
|
);
|
|
} else {
|
|
expect(element.pubkey.equals(account1.publicKey)).toBe(true);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
test('validatorExit', async () => {
|
|
if (!mockRpcEnabled) {
|
|
console.log('validatorExit skipped on live node');
|
|
return;
|
|
}
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'validatorExit',
|
|
},
|
|
{
|
|
error: null,
|
|
result: false,
|
|
},
|
|
]);
|
|
|
|
const result = await connection.validatorExit();
|
|
expect(result).toBe(false);
|
|
});
|
|
|
|
test('get balance', async () => {
|
|
const account = new Account();
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBalance',
|
|
params: [account.publicKey.toBase58()],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: 0,
|
|
},
|
|
},
|
|
]);
|
|
|
|
const balance = await connection.getBalance(account.publicKey);
|
|
expect(balance).toBeGreaterThanOrEqual(0);
|
|
});
|
|
|
|
test('get inflation', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getInflationGovernor',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
foundation: 0.05,
|
|
foundationTerm: 7.0,
|
|
initial: 0.15,
|
|
taper: 0.15,
|
|
terminal: 0.015,
|
|
},
|
|
},
|
|
]);
|
|
|
|
const inflation = await connection.getInflationGovernor();
|
|
|
|
for (const key of [
|
|
'initial',
|
|
'terminal',
|
|
'taper',
|
|
'foundation',
|
|
'foundationTerm',
|
|
]) {
|
|
expect(inflation).toHaveProperty(key);
|
|
expect(inflation[key]).toBeGreaterThan(0);
|
|
}
|
|
});
|
|
|
|
test('get epoch info', async () => {
|
|
const connection = new Connection(url, 'recent');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getEpochInfo',
|
|
params: [{commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
epoch: 0,
|
|
slotIndex: 1,
|
|
slotsInEpoch: 8192,
|
|
absoluteSlot: 1,
|
|
blockHeight: 1,
|
|
},
|
|
},
|
|
]);
|
|
|
|
const epochInfo = await connection.getEpochInfo();
|
|
|
|
for (const key of [
|
|
'epoch',
|
|
'slotIndex',
|
|
'slotsInEpoch',
|
|
'absoluteSlot' /*, 'blockHeight'*/, // Uncomment blockHeight after 1.1.20 ships
|
|
]) {
|
|
expect(epochInfo).toHaveProperty(key);
|
|
expect(epochInfo[key]).toBeGreaterThanOrEqual(0);
|
|
}
|
|
});
|
|
|
|
test('get epoch schedule', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getEpochSchedule',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
firstNormalEpoch: 8,
|
|
firstNormalSlot: 8160,
|
|
leaderScheduleSlotOffset: 8192,
|
|
slotsPerEpoch: 8192,
|
|
warmup: true,
|
|
},
|
|
},
|
|
]);
|
|
|
|
const epochSchedule = await connection.getEpochSchedule();
|
|
|
|
for (const key of [
|
|
'firstNormalEpoch',
|
|
'firstNormalSlot',
|
|
'leaderScheduleSlotOffset',
|
|
'slotsPerEpoch',
|
|
]) {
|
|
expect(epochSchedule).toHaveProperty('warmup');
|
|
expect(epochSchedule).toHaveProperty(key);
|
|
if (epochSchedule.warmup) {
|
|
expect(epochSchedule[key]).toBeGreaterThan(0);
|
|
}
|
|
}
|
|
});
|
|
|
|
test('get leader schedule', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getLeaderSchedule',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
'123vij84ecQEKUvQ7gYMKxKwKF6PbYSzCzzURYA4xULY': [0, 1, 2, 3],
|
|
'8PTjAikKoAybKXcEPnDSoy8wSNNikUBJ1iKawJKQwXnB': [4, 5, 6, 7],
|
|
},
|
|
},
|
|
]);
|
|
|
|
const leaderSchedule = await connection.getLeaderSchedule();
|
|
expect(Object.keys(leaderSchedule).length).toBeGreaterThanOrEqual(1);
|
|
for (const key in leaderSchedule) {
|
|
const slots = leaderSchedule[key];
|
|
expect(Array.isArray(slots)).toBe(true);
|
|
expect(slots.length).toBeGreaterThanOrEqual(4);
|
|
}
|
|
});
|
|
|
|
test('get slot', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSlot',
|
|
},
|
|
{
|
|
error: null,
|
|
result: 123,
|
|
},
|
|
]);
|
|
|
|
const slotLeader = await connection.getSlot();
|
|
if (mockRpcEnabled) {
|
|
expect(slotLeader).toBe(123);
|
|
} else {
|
|
// No idea what the correct slot value should be on a live cluster, so
|
|
// just check the type
|
|
expect(typeof slotLeader).toBe('number');
|
|
}
|
|
});
|
|
|
|
test('get slot leader', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSlotLeader',
|
|
},
|
|
{
|
|
error: null,
|
|
result: '11111111111111111111111111111111',
|
|
},
|
|
]);
|
|
|
|
const slotLeader = await connection.getSlotLeader();
|
|
if (mockRpcEnabled) {
|
|
expect(slotLeader).toBe('11111111111111111111111111111111');
|
|
} else {
|
|
// No idea what the correct slotLeader value should be on a live cluster, so
|
|
// just check the type
|
|
expect(typeof slotLeader).toBe('string');
|
|
}
|
|
});
|
|
|
|
test('get cluster nodes', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getClusterNodes',
|
|
},
|
|
{
|
|
error: null,
|
|
result: [
|
|
{
|
|
pubkey: '11111111111111111111111111111111',
|
|
gossip: '127.0.0.0:1234',
|
|
tpu: '127.0.0.0:1235',
|
|
rpc: null,
|
|
version: '1.1.10',
|
|
},
|
|
],
|
|
},
|
|
]);
|
|
|
|
const clusterNodes = await connection.getClusterNodes();
|
|
if (mockRpcEnabled) {
|
|
expect(clusterNodes).toHaveLength(1);
|
|
expect(clusterNodes[0].pubkey).toBe('11111111111111111111111111111111');
|
|
expect(typeof clusterNodes[0].gossip).toBe('string');
|
|
expect(typeof clusterNodes[0].tpu).toBe('string');
|
|
expect(clusterNodes[0].rpc).toBeNull();
|
|
} else {
|
|
// There should be at least one node (the node that we're talking to)
|
|
expect(clusterNodes.length).toBeGreaterThan(0);
|
|
}
|
|
});
|
|
|
|
test('getVoteAccounts', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url);
|
|
const voteAccounts = await connection.getVoteAccounts();
|
|
expect(
|
|
voteAccounts.current.concat(voteAccounts.delinquent).length,
|
|
).toBeGreaterThan(0);
|
|
});
|
|
|
|
test('confirm transaction - error', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
const badTransactionSignature = 'bad transaction signature';
|
|
|
|
await expect(
|
|
connection.confirmTransaction(badTransactionSignature),
|
|
).rejects.toThrow('signature must be base58 encoded');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSignatureStatuses',
|
|
params: [[badTransactionSignature]],
|
|
},
|
|
errorResponse,
|
|
]);
|
|
|
|
await expect(
|
|
connection.getSignatureStatus(badTransactionSignature),
|
|
).rejects.toThrow(errorMessage);
|
|
});
|
|
|
|
test('get transaction count', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getTransactionCount',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 1000000,
|
|
},
|
|
]);
|
|
|
|
const count = await connection.getTransactionCount();
|
|
expect(count).toBeGreaterThanOrEqual(0);
|
|
});
|
|
|
|
test('get total supply', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getTotalSupply',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 1000000,
|
|
},
|
|
]);
|
|
|
|
const count = await connection.getTotalSupply();
|
|
expect(count).toBeGreaterThanOrEqual(0);
|
|
});
|
|
|
|
test('get minimum balance for rent exemption', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getMinimumBalanceForRentExemption',
|
|
params: [512],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 1000000,
|
|
},
|
|
]);
|
|
|
|
const count = await connection.getMinimumBalanceForRentExemption(512);
|
|
expect(count).toBeGreaterThanOrEqual(0);
|
|
});
|
|
|
|
test('get confirmed signatures for address', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSlot',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 1,
|
|
},
|
|
]);
|
|
|
|
while ((await connection.getSlot()) <= 0) {
|
|
continue;
|
|
}
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedBlock',
|
|
params: [1],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
blockhash: '57zQNBZBEiHsCZFqsaY6h176ioXy5MsSLmcvHkEyaLGy',
|
|
previousBlockhash: 'H5nJ91eGag3B5ZSRHZ7zG5ZwXJ6ywCt2hyR8xCsV7xMo',
|
|
parentSlot: 0,
|
|
transactions: [
|
|
{
|
|
meta: {
|
|
fee: 10000,
|
|
postBalances: [499260347380, 15298080, 1, 1, 1],
|
|
preBalances: [499260357380, 15298080, 1, 1, 1],
|
|
status: {Ok: null},
|
|
err: null,
|
|
},
|
|
transaction: {
|
|
message: {
|
|
accountKeys: [
|
|
'va12u4o9DipLEB2z4fuoHszroq1U9NcAB9aooFDPJSf',
|
|
'57zQNBZBEiHsCZFqsaY6h176ioXy5MsSLmcvHkEyaLGy',
|
|
'SysvarS1otHashes111111111111111111111111111',
|
|
'SysvarC1ock11111111111111111111111111111111',
|
|
'Vote111111111111111111111111111111111111111',
|
|
],
|
|
header: {
|
|
numReadonlySignedAccounts: 0,
|
|
numReadonlyUnsignedAccounts: 3,
|
|
numRequiredSignatures: 2,
|
|
},
|
|
instructions: [
|
|
{
|
|
accounts: [1, 2, 3],
|
|
data:
|
|
'37u9WtQpcm6ULa3VtWDFAWoQc1hUvybPrA3dtx99tgHvvcE7pKRZjuGmn7VX2tC3JmYDYGG7',
|
|
programIdIndex: 4,
|
|
},
|
|
],
|
|
recentBlockhash: 'GeyAFFRY3WGpmam2hbgrKw4rbU2RKzfVLm5QLSeZwTZE',
|
|
},
|
|
signatures: [
|
|
'w2Zeq8YkpyB463DttvfzARD7k9ZxGEwbsEw4boEK7jDp3pfoxZbTdLFSsEPhzXhpCcjGi2kHtHFobgX49MMhbWt',
|
|
'4oCEqwGrMdBeMxpzuWiukCYqSfV4DsSKXSiVVCh1iJ6pS772X7y219JZP3mgqBz5PhsvprpKyhzChjYc3VSBQXzG',
|
|
],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
},
|
|
]);
|
|
|
|
// Find a block that has a transaction, usually Block 1
|
|
let slot = 0;
|
|
let address: ?PublicKey;
|
|
let expectedSignature: ?string;
|
|
while (!address || !expectedSignature) {
|
|
slot++;
|
|
const block = await connection.getConfirmedBlock(slot);
|
|
if (block.transactions.length > 0) {
|
|
const {
|
|
signature,
|
|
publicKey,
|
|
} = block.transactions[0].transaction.signatures[0];
|
|
if (signature) {
|
|
address = publicKey;
|
|
expectedSignature = bs58.encode(signature);
|
|
}
|
|
}
|
|
}
|
|
|
|
// getConfirmedSignaturesForAddress tests...
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedSignaturesForAddress',
|
|
params: [address.toBase58(), slot, slot + 1],
|
|
},
|
|
{
|
|
error: null,
|
|
result: [expectedSignature],
|
|
},
|
|
]);
|
|
|
|
const confirmedSignatures = await connection.getConfirmedSignaturesForAddress(
|
|
address,
|
|
slot,
|
|
slot + 1,
|
|
);
|
|
expect(confirmedSignatures.includes(expectedSignature)).toBe(true);
|
|
|
|
const badSlot = Number.MAX_SAFE_INTEGER - 1;
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedSignaturesForAddress',
|
|
params: [address.toBase58(), badSlot, badSlot + 1],
|
|
},
|
|
{
|
|
error: null,
|
|
result: [],
|
|
},
|
|
]);
|
|
|
|
const emptySignatures = await connection.getConfirmedSignaturesForAddress(
|
|
address,
|
|
badSlot,
|
|
badSlot + 1,
|
|
);
|
|
expect(emptySignatures.length).toBe(0);
|
|
|
|
// getConfirmedSignaturesForAddress2 tests...
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedSignaturesForAddress2',
|
|
params: [address.toBase58(), {limit: 1}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: [
|
|
{
|
|
signature: expectedSignature,
|
|
slot,
|
|
err: null,
|
|
memo: null,
|
|
},
|
|
],
|
|
},
|
|
]);
|
|
|
|
const confirmedSignatures2 = await connection.getConfirmedSignaturesForAddress2(
|
|
address,
|
|
{limit: 1},
|
|
);
|
|
expect(confirmedSignatures2.length).toBe(1);
|
|
if (mockRpcEnabled) {
|
|
expect(confirmedSignatures2[0].signature).toBe(expectedSignature);
|
|
expect(confirmedSignatures2[0].slot).toBe(slot);
|
|
expect(confirmedSignatures2[0].err).toBeNull();
|
|
expect(confirmedSignatures2[0].memo).toBeNull();
|
|
}
|
|
});
|
|
|
|
test('get confirmed transaction', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSlot',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 1,
|
|
},
|
|
]);
|
|
|
|
while ((await connection.getSlot()) <= 0) {
|
|
continue;
|
|
}
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedBlock',
|
|
params: [1],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
blockhash: '57zQNBZBEiHsCZFqsaY6h176ioXy5MsSLmcvHkEyaLGy',
|
|
previousBlockhash: 'H5nJ91eGag3B5ZSRHZ7zG5ZwXJ6ywCt2hyR8xCsV7xMo',
|
|
parentSlot: 0,
|
|
transactions: [
|
|
{
|
|
meta: {
|
|
fee: 10000,
|
|
postBalances: [499260347380, 15298080, 1, 1, 1],
|
|
preBalances: [499260357380, 15298080, 1, 1, 1],
|
|
status: {Ok: null},
|
|
err: null,
|
|
},
|
|
transaction: {
|
|
message: {
|
|
accountKeys: [
|
|
'va12u4o9DipLEB2z4fuoHszroq1U9NcAB9aooFDPJSf',
|
|
'57zQNBZBEiHsCZFqsaY6h176ioXy5MsSLmcvHkEyaLGy',
|
|
'SysvarS1otHashes111111111111111111111111111',
|
|
'SysvarC1ock11111111111111111111111111111111',
|
|
'Vote111111111111111111111111111111111111111',
|
|
],
|
|
header: {
|
|
numReadonlySignedAccounts: 0,
|
|
numReadonlyUnsignedAccounts: 3,
|
|
numRequiredSignatures: 2,
|
|
},
|
|
instructions: [
|
|
{
|
|
accounts: [1, 2, 3],
|
|
data:
|
|
'37u9WtQpcm6ULa3VtWDFAWoQc1hUvybPrA3dtx99tgHvvcE7pKRZjuGmn7VX2tC3JmYDYGG7',
|
|
programIdIndex: 4,
|
|
},
|
|
],
|
|
recentBlockhash: 'GeyAFFRY3WGpmam2hbgrKw4rbU2RKzfVLm5QLSeZwTZE',
|
|
},
|
|
signatures: [
|
|
'w2Zeq8YkpyB463DttvfzARD7k9ZxGEwbsEw4boEK7jDp3pfoxZbTdLFSsEPhzXhpCcjGi2kHtHFobgX49MMhbWt',
|
|
'4oCEqwGrMdBeMxpzuWiukCYqSfV4DsSKXSiVVCh1iJ6pS772X7y219JZP3mgqBz5PhsvprpKyhzChjYc3VSBQXzG',
|
|
],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
},
|
|
]);
|
|
|
|
// Find a block that has a transaction, usually Block 1
|
|
let slot = 0;
|
|
let confirmedTransaction: ?string;
|
|
while (!confirmedTransaction) {
|
|
slot++;
|
|
const block = await connection.getConfirmedBlock(slot);
|
|
for (const tx of block.transactions) {
|
|
if (tx.transaction.signature) {
|
|
confirmedTransaction = bs58.encode(tx.transaction.signature);
|
|
}
|
|
}
|
|
}
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedTransaction',
|
|
params: [confirmedTransaction],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
slot,
|
|
transaction: {
|
|
message: {
|
|
accountKeys: [
|
|
'va12u4o9DipLEB2z4fuoHszroq1U9NcAB9aooFDPJSf',
|
|
'57zQNBZBEiHsCZFqsaY6h176ioXy5MsSLmcvHkEyaLGy',
|
|
'SysvarS1otHashes111111111111111111111111111',
|
|
'SysvarC1ock11111111111111111111111111111111',
|
|
'Vote111111111111111111111111111111111111111',
|
|
],
|
|
header: {
|
|
numReadonlySignedAccounts: 0,
|
|
numReadonlyUnsignedAccounts: 3,
|
|
numRequiredSignatures: 2,
|
|
},
|
|
instructions: [
|
|
{
|
|
accounts: [1, 2, 3],
|
|
data:
|
|
'37u9WtQpcm6ULa3VtWDFAWoQc1hUvybPrA3dtx99tgHvvcE7pKRZjuGmn7VX2tC3JmYDYGG7',
|
|
programIdIndex: 4,
|
|
},
|
|
],
|
|
recentBlockhash: 'GeyAFFRY3WGpmam2hbgrKw4rbU2RKzfVLm5QLSeZwTZE',
|
|
},
|
|
signatures: [
|
|
'w2Zeq8YkpyB463DttvfzARD7k9ZxGEwbsEw4boEK7jDp3pfoxZbTdLFSsEPhzXhpCcjGi2kHtHFobgX49MMhbWt',
|
|
'4oCEqwGrMdBeMxpzuWiukCYqSfV4DsSKXSiVVCh1iJ6pS772X7y219JZP3mgqBz5PhsvprpKyhzChjYc3VSBQXzG',
|
|
],
|
|
},
|
|
meta: {
|
|
fee: 10000,
|
|
postBalances: [499260347380, 15298080, 1, 1, 1],
|
|
preBalances: [499260357380, 15298080, 1, 1, 1],
|
|
status: {Ok: null},
|
|
err: null,
|
|
},
|
|
},
|
|
},
|
|
]);
|
|
|
|
const result = await connection.getConfirmedTransaction(confirmedTransaction);
|
|
|
|
if (!result) {
|
|
expect(result).toBeDefined();
|
|
expect(result).not.toBeNull();
|
|
return;
|
|
}
|
|
|
|
if (result.transaction.signature === null) {
|
|
expect(result.transaction.signature).not.toBeNull();
|
|
return;
|
|
}
|
|
|
|
const resultSignature = bs58.encode(result.transaction.signature);
|
|
expect(resultSignature).toEqual(confirmedTransaction);
|
|
|
|
const newAddress = new Account().publicKey;
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'requestAirdrop',
|
|
params: [newAddress.toBase58(), 1],
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'1WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
const recentSignature = await connection.requestAirdrop(newAddress, 1);
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedTransaction',
|
|
params: [recentSignature],
|
|
},
|
|
{
|
|
error: null,
|
|
result: null,
|
|
},
|
|
]);
|
|
|
|
const nullResponse = await connection.getConfirmedTransaction(
|
|
recentSignature,
|
|
);
|
|
expect(nullResponse).toBeNull();
|
|
});
|
|
|
|
test('get confirmed block', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSlot',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 1,
|
|
},
|
|
]);
|
|
|
|
while ((await connection.getSlot()) <= 0) {
|
|
continue;
|
|
}
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedBlock',
|
|
params: [0],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
blockhash: 'H5nJ91eGag3B5ZSRHZ7zG5ZwXJ6ywCt2hyR8xCsV7xMo',
|
|
previousBlockhash: 'H5nJ91eGag3B5ZSRHZ7zG5ZwXJ6ywCt2hyR8xCsV7xMo',
|
|
parentSlot: 0,
|
|
transactions: [],
|
|
},
|
|
},
|
|
]);
|
|
|
|
// Block 0 never has any transactions in automation localnet
|
|
const block0 = await connection.getConfirmedBlock(0);
|
|
const blockhash0 = block0.blockhash;
|
|
expect(block0.transactions.length).toBe(0);
|
|
expect(blockhash0).not.toBeNull();
|
|
expect(block0.previousBlockhash).not.toBeNull();
|
|
expect(block0.parentSlot).toBe(0);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedBlock',
|
|
params: [1],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
blockhash: '57zQNBZBEiHsCZFqsaY6h176ioXy5MsSLmcvHkEyaLGy',
|
|
previousBlockhash: 'H5nJ91eGag3B5ZSRHZ7zG5ZwXJ6ywCt2hyR8xCsV7xMo',
|
|
parentSlot: 0,
|
|
transactions: [
|
|
{
|
|
meta: {
|
|
fee: 10000,
|
|
postBalances: [499260347380, 15298080, 1, 1, 1],
|
|
preBalances: [499260357380, 15298080, 1, 1, 1],
|
|
status: {Ok: null},
|
|
err: null,
|
|
},
|
|
transaction: {
|
|
message: {
|
|
accountKeys: [
|
|
'va12u4o9DipLEB2z4fuoHszroq1U9NcAB9aooFDPJSf',
|
|
'57zQNBZBEiHsCZFqsaY6h176ioXy5MsSLmcvHkEyaLGy',
|
|
'SysvarS1otHashes111111111111111111111111111',
|
|
'SysvarC1ock11111111111111111111111111111111',
|
|
'Vote111111111111111111111111111111111111111',
|
|
],
|
|
header: {
|
|
numReadonlySignedAccounts: 0,
|
|
numReadonlyUnsignedAccounts: 3,
|
|
numRequiredSignatures: 2,
|
|
},
|
|
instructions: [
|
|
{
|
|
accounts: [1, 2, 3],
|
|
data:
|
|
'37u9WtQpcm6ULa3VtWDFAWoQc1hUvybPrA3dtx99tgHvvcE7pKRZjuGmn7VX2tC3JmYDYGG7',
|
|
programIdIndex: 4,
|
|
},
|
|
],
|
|
recentBlockhash: 'GeyAFFRY3WGpmam2hbgrKw4rbU2RKzfVLm5QLSeZwTZE',
|
|
},
|
|
signatures: [
|
|
'w2Zeq8YkpyB463DttvfzARD7k9ZxGEwbsEw4boEK7jDp3pfoxZbTdLFSsEPhzXhpCcjGi2kHtHFobgX49MMhbWt',
|
|
'4oCEqwGrMdBeMxpzuWiukCYqSfV4DsSKXSiVVCh1iJ6pS772X7y219JZP3mgqBz5PhsvprpKyhzChjYc3VSBQXzG',
|
|
],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
},
|
|
]);
|
|
|
|
// Find a block that has a transaction, usually Block 1
|
|
let x = 1;
|
|
while (x < 10) {
|
|
const block1 = await connection.getConfirmedBlock(x);
|
|
if (block1.transactions.length >= 1) {
|
|
expect(block1.previousBlockhash).toBe(blockhash0);
|
|
expect(block1.blockhash).not.toBeNull();
|
|
expect(block1.parentSlot).toBe(0);
|
|
expect(block1.transactions[0].transaction).not.toBeNull();
|
|
break;
|
|
}
|
|
x++;
|
|
}
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getConfirmedBlock',
|
|
params: [Number.MAX_SAFE_INTEGER],
|
|
},
|
|
{
|
|
error: {
|
|
message: `Block not available for slot ${Number.MAX_SAFE_INTEGER}`,
|
|
},
|
|
result: null,
|
|
},
|
|
]);
|
|
await expect(
|
|
connection.getConfirmedBlock(Number.MAX_SAFE_INTEGER),
|
|
).rejects.toThrow(`Block not available for slot ${Number.MAX_SAFE_INTEGER}`);
|
|
});
|
|
|
|
test('get recent blockhash', async () => {
|
|
const connection = new Connection(url);
|
|
for (const commitment of [
|
|
'max',
|
|
'recent',
|
|
'root',
|
|
'single',
|
|
'singleGossip',
|
|
]) {
|
|
mockGetRecentBlockhash(commitment);
|
|
|
|
const {blockhash, feeCalculator} = await connection.getRecentBlockhash(
|
|
commitment,
|
|
);
|
|
expect(blockhash.length).toBeGreaterThanOrEqual(43);
|
|
expect(feeCalculator.lamportsPerSignature).toBeGreaterThanOrEqual(0);
|
|
}
|
|
});
|
|
|
|
test('get fee calculator', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockGetRecentBlockhash('recent');
|
|
const {blockhash} = await connection.getRecentBlockhash('recent');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getFeeCalculatorForBlockhash',
|
|
params: [blockhash, {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: {
|
|
feeCalculator: {
|
|
lamportsPerSignature: 5000,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
]);
|
|
|
|
const feeCalculator = (
|
|
await connection.getFeeCalculatorForBlockhash(blockhash, 'recent')
|
|
).value;
|
|
if (feeCalculator === null) {
|
|
expect(feeCalculator).not.toBeNull();
|
|
return;
|
|
}
|
|
expect(feeCalculator.lamportsPerSignature).toEqual(5000);
|
|
});
|
|
|
|
test('get block time', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBlockTime',
|
|
params: [1],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 10000,
|
|
},
|
|
]);
|
|
|
|
const blockTime = await connection.getBlockTime(1);
|
|
if (blockTime === null) {
|
|
// TODO: enable after https://github.com/solana-labs/solana/issues/11849 fixed
|
|
// expect(blockTime).not.toBeNull();
|
|
} else {
|
|
expect(blockTime).toBeGreaterThan(0);
|
|
}
|
|
});
|
|
|
|
test('get minimum ledger slot', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'minimumLedgerSlot',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 0,
|
|
},
|
|
]);
|
|
|
|
const minimumLedgerSlot = await connection.getMinimumLedgerSlot();
|
|
expect(minimumLedgerSlot).toBeGreaterThanOrEqual(0);
|
|
});
|
|
|
|
test('get first available block', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getFirstAvailableBlock',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 0,
|
|
},
|
|
]);
|
|
|
|
const firstAvailableBlock = await connection.getFirstAvailableBlock();
|
|
expect(firstAvailableBlock).toBeGreaterThanOrEqual(0);
|
|
});
|
|
|
|
test('get supply', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSupply',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 1,
|
|
},
|
|
value: {
|
|
total: 1000,
|
|
circulating: 100,
|
|
nonCirculating: 900,
|
|
nonCirculatingAccounts: [new Account().publicKey.toBase58()],
|
|
},
|
|
},
|
|
},
|
|
]);
|
|
|
|
const supply = (await connection.getSupply()).value;
|
|
expect(supply.total).toBeGreaterThan(0);
|
|
expect(supply.circulating).toBeGreaterThan(0);
|
|
expect(supply.nonCirculating).toBeGreaterThan(0);
|
|
expect(supply.nonCirculatingAccounts.length).toBeGreaterThan(0);
|
|
});
|
|
|
|
test('get performance samples', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
if (mockRpcEnabled) {
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getRecentPerformanceSamples',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: [
|
|
{
|
|
slot: 1234,
|
|
numTransactions: 1000,
|
|
numSlots: 60,
|
|
samplePeriodSecs: 60,
|
|
},
|
|
],
|
|
},
|
|
]);
|
|
}
|
|
|
|
const perfSamples = await connection.getRecentPerformanceSamples();
|
|
expect(Array.isArray(perfSamples)).toBe(true);
|
|
|
|
if (perfSamples.length > 0) {
|
|
expect(perfSamples[0].slot).toBeGreaterThan(0);
|
|
expect(perfSamples[0].numTransactions).toBeGreaterThan(0);
|
|
expect(perfSamples[0].numSlots).toBeGreaterThan(0);
|
|
expect(perfSamples[0].samplePeriodSecs).toBeGreaterThan(0);
|
|
}
|
|
});
|
|
|
|
test('get performance samples limit too high', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
if (mockRpcEnabled) {
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getRecentPerformanceSamples',
|
|
params: [100000],
|
|
},
|
|
{
|
|
error: {
|
|
code: -32602,
|
|
message: 'Invalid limit; max 720',
|
|
},
|
|
result: null,
|
|
},
|
|
]);
|
|
}
|
|
|
|
await expect(
|
|
connection.getRecentPerformanceSamples(100000),
|
|
).rejects.toThrow();
|
|
});
|
|
|
|
const TOKEN_PROGRAM_ID = new PublicKey(
|
|
'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
);
|
|
|
|
describe('token methods', () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url, 'recent');
|
|
const newAccount = new Account().publicKey;
|
|
|
|
let testToken: Token;
|
|
let testTokenAccount: PublicKey;
|
|
let testSignature: TransactionSignature;
|
|
let testOwner: Account;
|
|
|
|
// Setup token mints and accounts for token tests
|
|
beforeAll(async () => {
|
|
const payerAccount = new Account();
|
|
await connection.confirmTransaction(
|
|
await connection.requestAirdrop(payerAccount.publicKey, 100000000000),
|
|
);
|
|
|
|
const mintOwner = new Account();
|
|
const accountOwner = new Account();
|
|
const token = await Token.createMint(
|
|
connection,
|
|
payerAccount,
|
|
mintOwner.publicKey,
|
|
null,
|
|
2,
|
|
TOKEN_PROGRAM_ID,
|
|
);
|
|
|
|
const tokenAccount = await token.createAccount(accountOwner.publicKey);
|
|
await token.mintTo(tokenAccount, mintOwner, [], 11111);
|
|
|
|
const token2 = await Token.createMint(
|
|
connection,
|
|
payerAccount,
|
|
mintOwner.publicKey,
|
|
null,
|
|
2,
|
|
TOKEN_PROGRAM_ID,
|
|
);
|
|
|
|
const token2Account = await token2.createAccount(accountOwner.publicKey);
|
|
await token2.mintTo(token2Account, mintOwner, [], 100);
|
|
|
|
const tokenAccountDest = await token.createAccount(accountOwner.publicKey);
|
|
testSignature = await token.transfer(
|
|
tokenAccount,
|
|
tokenAccountDest,
|
|
accountOwner,
|
|
[],
|
|
new u64(1),
|
|
);
|
|
|
|
await connection.confirmTransaction(testSignature, 'max');
|
|
|
|
testOwner = accountOwner;
|
|
testToken = token;
|
|
testTokenAccount = tokenAccount;
|
|
});
|
|
|
|
test('get token supply', async () => {
|
|
const supply = (await connection.getTokenSupply(testToken.publicKey)).value;
|
|
expect(supply.uiAmount).toEqual(111.11);
|
|
expect(supply.decimals).toEqual(2);
|
|
expect(supply.amount).toEqual('11111');
|
|
|
|
await expect(connection.getTokenSupply(newAccount)).rejects.toThrow();
|
|
});
|
|
|
|
test('get token largest accounts', async () => {
|
|
const largestAccounts = (
|
|
await connection.getTokenLargestAccounts(testToken.publicKey)
|
|
).value;
|
|
|
|
expect(largestAccounts.length).toEqual(2);
|
|
const largestAccount = largestAccounts[0];
|
|
expect(largestAccount.address.equals(testTokenAccount)).toBe(true);
|
|
expect(largestAccount.amount).toEqual('11110');
|
|
expect(largestAccount.decimals).toEqual(2);
|
|
expect(largestAccount.uiAmount).toEqual(111.1);
|
|
|
|
await expect(
|
|
connection.getTokenLargestAccounts(newAccount),
|
|
).rejects.toThrow();
|
|
});
|
|
|
|
test('get confirmed token transaction', async () => {
|
|
const parsedTx = await connection.getParsedConfirmedTransaction(
|
|
testSignature,
|
|
);
|
|
if (parsedTx === null) {
|
|
expect(parsedTx).not.toBeNull();
|
|
return;
|
|
}
|
|
const {signatures, message} = parsedTx.transaction;
|
|
expect(signatures[0]).toEqual(testSignature);
|
|
const ix = message.instructions[0];
|
|
if (ix.parsed) {
|
|
expect(ix.program).toEqual('spl-token');
|
|
expect(ix.programId.equals(TOKEN_PROGRAM_ID)).toBe(true);
|
|
} else {
|
|
expect('parsed' in ix).toBe(true);
|
|
}
|
|
|
|
const missingSignature =
|
|
'45pGoC4Rr3fJ1TKrsiRkhHRbdUeX7633XAGVec6XzVdpRbzQgHhe6ZC6Uq164MPWtiqMg7wCkC6Wy3jy2BqsDEKf';
|
|
const nullResponse = await connection.getParsedConfirmedTransaction(
|
|
missingSignature,
|
|
);
|
|
|
|
expect(nullResponse).toBeNull();
|
|
});
|
|
|
|
test('get token account balance', async () => {
|
|
const balance = (await connection.getTokenAccountBalance(testTokenAccount))
|
|
.value;
|
|
expect(balance.amount).toEqual('11110');
|
|
expect(balance.decimals).toEqual(2);
|
|
expect(balance.uiAmount).toEqual(111.1);
|
|
|
|
await expect(
|
|
connection.getTokenAccountBalance(newAccount),
|
|
).rejects.toThrow();
|
|
});
|
|
|
|
test('get parsed token account info', async () => {
|
|
const accountInfo = (
|
|
await connection.getParsedAccountInfo(testTokenAccount)
|
|
).value;
|
|
if (accountInfo) {
|
|
const data = accountInfo.data;
|
|
if (data instanceof Buffer) {
|
|
expect(data instanceof Buffer).toBe(false);
|
|
} else {
|
|
expect(data.program).toEqual('spl-token');
|
|
expect(data.parsed).toBeTruthy();
|
|
}
|
|
}
|
|
});
|
|
|
|
test('get parsed token program accounts', async () => {
|
|
const tokenAccounts = await connection.getParsedProgramAccounts(
|
|
TOKEN_PROGRAM_ID,
|
|
);
|
|
tokenAccounts.forEach(({account}) => {
|
|
expect(account.owner.equals(TOKEN_PROGRAM_ID)).toBe(true);
|
|
const data = account.data;
|
|
if (data instanceof Buffer) {
|
|
expect(data instanceof Buffer).toBe(false);
|
|
} else {
|
|
expect(data.parsed).toBeTruthy();
|
|
expect(data.program).toEqual('spl-token');
|
|
}
|
|
});
|
|
});
|
|
|
|
test('get parsed token accounts by owner', async () => {
|
|
const tokenAccounts = (
|
|
await connection.getParsedTokenAccountsByOwner(testOwner.publicKey, {
|
|
mint: testToken.publicKey,
|
|
})
|
|
).value;
|
|
tokenAccounts.forEach(({account}) => {
|
|
expect(account.owner.equals(TOKEN_PROGRAM_ID)).toBe(true);
|
|
const data = account.data;
|
|
if (data instanceof Buffer) {
|
|
expect(data instanceof Buffer).toBe(false);
|
|
} else {
|
|
expect(data.parsed).toBeTruthy();
|
|
expect(data.program).toEqual('spl-token');
|
|
}
|
|
});
|
|
});
|
|
|
|
test('get token accounts by owner', async () => {
|
|
const accountsWithMintFilter = (
|
|
await connection.getTokenAccountsByOwner(testOwner.publicKey, {
|
|
mint: testToken.publicKey,
|
|
})
|
|
).value;
|
|
expect(accountsWithMintFilter.length).toEqual(2);
|
|
|
|
const accountsWithProgramFilter = (
|
|
await connection.getTokenAccountsByOwner(testOwner.publicKey, {
|
|
programId: TOKEN_PROGRAM_ID,
|
|
})
|
|
).value;
|
|
expect(accountsWithProgramFilter.length).toEqual(3);
|
|
|
|
const noAccounts = (
|
|
await connection.getTokenAccountsByOwner(newAccount, {
|
|
mint: testToken.publicKey,
|
|
})
|
|
).value;
|
|
expect(noAccounts.length).toEqual(0);
|
|
|
|
await expect(
|
|
connection.getTokenAccountsByOwner(testOwner.publicKey, {
|
|
mint: newAccount,
|
|
}),
|
|
).rejects.toThrow();
|
|
|
|
await expect(
|
|
connection.getTokenAccountsByOwner(testOwner.publicKey, {
|
|
programId: newAccount,
|
|
}),
|
|
).rejects.toThrow();
|
|
});
|
|
});
|
|
|
|
test('get largest accounts', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getLargestAccounts',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 1,
|
|
},
|
|
value: new Array(20).fill(0).map(() => ({
|
|
address: new Account().publicKey.toBase58(),
|
|
lamports: 1000,
|
|
})),
|
|
},
|
|
},
|
|
]);
|
|
|
|
const largestAccounts = (await connection.getLargestAccounts()).value;
|
|
expect(largestAccounts.length).toEqual(20);
|
|
});
|
|
|
|
test('stake activation should throw when called for not delegated account', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
const publicKey = new Account().publicKey;
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getStakeActivation',
|
|
params: [publicKey.toBase58(), {}],
|
|
},
|
|
{
|
|
error: {message: 'account not delegated'},
|
|
result: undefined,
|
|
},
|
|
]);
|
|
|
|
await expect(connection.getStakeActivation(publicKey)).rejects.toThrow();
|
|
});
|
|
|
|
test('stake activation should return activating for new accounts', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url, 'recent');
|
|
const voteAccounts = await connection.getVoteAccounts();
|
|
const voteAccount = voteAccounts.current.concat(voteAccounts.delinquent)[0];
|
|
const votePubkey = new PublicKey(voteAccount.votePubkey);
|
|
|
|
const authorized = new Account();
|
|
await connection.requestAirdrop(authorized.publicKey, 2 * LAMPORTS_PER_SOL);
|
|
|
|
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
|
|
StakeProgram.space,
|
|
'recent',
|
|
);
|
|
|
|
const newStakeAccount = new Account();
|
|
let createAndInitialize = StakeProgram.createAccount({
|
|
fromPubkey: authorized.publicKey,
|
|
stakePubkey: newStakeAccount.publicKey,
|
|
authorized: new Authorized(authorized.publicKey, authorized.publicKey),
|
|
lockup: new Lockup(0, 0, new PublicKey(0)),
|
|
lamports: minimumAmount + 42,
|
|
});
|
|
|
|
await sendAndConfirmTransaction(
|
|
connection,
|
|
createAndInitialize,
|
|
[authorized, newStakeAccount],
|
|
{commitment: 'single', skipPreflight: true},
|
|
);
|
|
let delegation = StakeProgram.delegate({
|
|
stakePubkey: newStakeAccount.publicKey,
|
|
authorizedPubkey: authorized.publicKey,
|
|
votePubkey,
|
|
});
|
|
await sendAndConfirmTransaction(connection, delegation, [authorized], {
|
|
commitment: 'single',
|
|
skipPreflight: true,
|
|
});
|
|
|
|
const LARGE_EPOCH = 4000;
|
|
await expect(
|
|
connection.getStakeActivation(
|
|
newStakeAccount.publicKey,
|
|
'recent',
|
|
LARGE_EPOCH,
|
|
),
|
|
).rejects.toThrow(
|
|
`failed to get Stake Activation ${newStakeAccount.publicKey.toBase58()}: Invalid param: epoch ${LARGE_EPOCH} has not yet started`,
|
|
);
|
|
|
|
const activationState = await connection.getStakeActivation(
|
|
newStakeAccount.publicKey,
|
|
);
|
|
expect(activationState.state).toBe('activating');
|
|
expect(activationState.inactive).toBe(42);
|
|
expect(activationState.active).toBe(0);
|
|
});
|
|
|
|
test('stake activation should only accept state with valid string literals', async () => {
|
|
if (!mockRpcEnabled) {
|
|
console.log('live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url, 'recent');
|
|
const publicKey = new Account().publicKey;
|
|
|
|
const addStakeActivationMock = state => {
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getStakeActivation',
|
|
params: [publicKey.toBase58(), {}],
|
|
},
|
|
{
|
|
error: undefined,
|
|
result: {
|
|
state: state,
|
|
active: 0,
|
|
inactive: 80,
|
|
},
|
|
},
|
|
]);
|
|
};
|
|
|
|
addStakeActivationMock('active');
|
|
let activation = await connection.getStakeActivation(publicKey);
|
|
expect(activation.state).toBe('active');
|
|
expect(activation.active).toBe(0);
|
|
expect(activation.inactive).toBe(80);
|
|
|
|
addStakeActivationMock('invalid');
|
|
await expect(connection.getStakeActivation(publicKey)).rejects.toThrow();
|
|
});
|
|
|
|
test('getVersion', async () => {
|
|
const connection = new Connection(url);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getVersion',
|
|
params: [],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {'solana-core': '0.20.4'},
|
|
},
|
|
]);
|
|
|
|
const version = await connection.getVersion();
|
|
expect(version['solana-core']).toBeTruthy();
|
|
});
|
|
|
|
test('request airdrop', async () => {
|
|
const account = new Account();
|
|
const connection = new Connection(url, 'recent');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getMinimumBalanceForRentExemption',
|
|
params: [0, {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 50,
|
|
},
|
|
]);
|
|
|
|
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
|
|
0,
|
|
'recent',
|
|
);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'requestAirdrop',
|
|
params: [account.publicKey.toBase58(), minimumAmount + 42],
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'1WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
const signature = await connection.requestAirdrop(
|
|
account.publicKey,
|
|
minimumAmount + 42,
|
|
);
|
|
|
|
mockConfirmTransaction(signature);
|
|
await connection.confirmTransaction(signature, 'single');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBalance',
|
|
params: [account.publicKey.toBase58(), {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: minimumAmount + 42,
|
|
},
|
|
},
|
|
]);
|
|
|
|
const balance = await connection.getBalance(account.publicKey);
|
|
expect(balance).toBe(minimumAmount + 42);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getAccountInfo',
|
|
params: [
|
|
account.publicKey.toBase58(),
|
|
{commitment: 'recent', encoding: 'base64'},
|
|
],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: {
|
|
owner: '11111111111111111111111111111111',
|
|
lamports: minimumAmount + 42,
|
|
data: ['', 'base64'],
|
|
executable: false,
|
|
},
|
|
},
|
|
},
|
|
]);
|
|
|
|
const accountInfo = await connection.getAccountInfo(account.publicKey);
|
|
if (accountInfo === null) {
|
|
expect(accountInfo).not.toBeNull();
|
|
return;
|
|
}
|
|
expect(accountInfo.lamports).toBe(minimumAmount + 42);
|
|
expect(accountInfo.data).toHaveLength(0);
|
|
expect(accountInfo.owner).toEqual(SystemProgram.programId);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getAccountInfo',
|
|
params: [
|
|
account.publicKey.toBase58(),
|
|
{commitment: 'recent', encoding: 'jsonParsed'},
|
|
],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: {
|
|
owner: '11111111111111111111111111111111',
|
|
lamports: minimumAmount + 42,
|
|
data: ['', 'base64'],
|
|
executable: false,
|
|
},
|
|
},
|
|
},
|
|
]);
|
|
|
|
const parsedAccountInfo = (
|
|
await connection.getParsedAccountInfo(account.publicKey)
|
|
).value;
|
|
if (parsedAccountInfo === null) {
|
|
expect(parsedAccountInfo).not.toBeNull();
|
|
return;
|
|
} else if (parsedAccountInfo.data.parsed) {
|
|
expect(parsedAccountInfo.data.parsed).not.toBeTruthy();
|
|
return;
|
|
}
|
|
expect(parsedAccountInfo.lamports).toBe(minimumAmount + 42);
|
|
expect(parsedAccountInfo.data).toHaveLength(0);
|
|
expect(parsedAccountInfo.owner).toEqual(SystemProgram.programId);
|
|
});
|
|
|
|
test('transaction failure', async () => {
|
|
const account = new Account();
|
|
const connection = new Connection(url, 'recent');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getMinimumBalanceForRentExemption',
|
|
params: [0, {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 50,
|
|
},
|
|
]);
|
|
|
|
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
|
|
0,
|
|
'recent',
|
|
);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'requestAirdrop',
|
|
params: [account.publicKey.toBase58(), minimumAmount + 100010],
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'2WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
const airdropSignature = await connection.requestAirdrop(
|
|
account.publicKey,
|
|
minimumAmount + 100010,
|
|
);
|
|
|
|
mockConfirmTransaction(airdropSignature);
|
|
await connection.confirmTransaction(airdropSignature, 'single');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBalance',
|
|
params: [account.publicKey.toBase58(), {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: minimumAmount + 100010,
|
|
},
|
|
},
|
|
]);
|
|
expect(await connection.getBalance(account.publicKey)).toBe(
|
|
minimumAmount + 100010,
|
|
);
|
|
|
|
mockGetRecentBlockhash('max');
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
const newAccount = new Account();
|
|
let transaction = new Transaction().add(
|
|
SystemProgram.createAccount({
|
|
fromPubkey: account.publicKey,
|
|
newAccountPubkey: newAccount.publicKey,
|
|
lamports: 1000,
|
|
space: 0,
|
|
programId: SystemProgram.programId,
|
|
}),
|
|
);
|
|
|
|
mockConfirmTransaction(
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
);
|
|
await sendAndConfirmTransaction(
|
|
connection,
|
|
transaction,
|
|
[account, newAccount],
|
|
{commitment: 'single', skipPreflight: true},
|
|
);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
// This should fail because the account is already created
|
|
transaction = new Transaction().add(
|
|
SystemProgram.createAccount({
|
|
fromPubkey: account.publicKey,
|
|
newAccountPubkey: newAccount.publicKey,
|
|
lamports: 10,
|
|
space: 0,
|
|
programId: SystemProgram.programId,
|
|
}),
|
|
);
|
|
const signature = await connection.sendTransaction(
|
|
transaction,
|
|
[account, newAccount],
|
|
{skipPreflight: true},
|
|
);
|
|
|
|
const expectedErr = {InstructionError: [0, {Custom: 0}]};
|
|
mockRpcSocket.push([
|
|
{
|
|
method: 'signatureSubscribe',
|
|
params: [signature, {commitment: 'single'}],
|
|
},
|
|
{
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: {err: expectedErr},
|
|
},
|
|
]);
|
|
|
|
// Wait for one confirmation
|
|
const confirmResult = (
|
|
await connection.confirmTransaction(signature, 'single')
|
|
).value;
|
|
expect(confirmResult.err).toEqual(expectedErr);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getSignatureStatuses',
|
|
params: [
|
|
[
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
],
|
|
],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: [
|
|
{
|
|
slot: 0,
|
|
confirmations: 11,
|
|
status: {Err: expectedErr},
|
|
err: expectedErr,
|
|
},
|
|
],
|
|
},
|
|
},
|
|
]);
|
|
|
|
const response = (await connection.getSignatureStatus(signature)).value;
|
|
verifySignatureStatus(response, expectedErr);
|
|
});
|
|
|
|
test('transaction', async () => {
|
|
const accountFrom = new Account();
|
|
const accountTo = new Account();
|
|
const connection = new Connection(url, 'recent');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getMinimumBalanceForRentExemption',
|
|
params: [0, {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: 50,
|
|
},
|
|
]);
|
|
|
|
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
|
|
0,
|
|
'recent',
|
|
);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'requestAirdrop',
|
|
params: [accountFrom.publicKey.toBase58(), minimumAmount + 100010],
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'2WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
const airdropFromSig = await connection.requestAirdrop(
|
|
accountFrom.publicKey,
|
|
minimumAmount + 100010,
|
|
);
|
|
|
|
mockConfirmTransaction(airdropFromSig);
|
|
await connection.confirmTransaction(airdropFromSig, 'single');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBalance',
|
|
params: [accountFrom.publicKey.toBase58(), {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: minimumAmount + 100010,
|
|
},
|
|
},
|
|
]);
|
|
expect(await connection.getBalance(accountFrom.publicKey)).toBe(
|
|
minimumAmount + 100010,
|
|
);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'requestAirdrop',
|
|
params: [accountTo.publicKey.toBase58(), minimumAmount],
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'8WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
const airdropToSig = await connection.requestAirdrop(
|
|
accountTo.publicKey,
|
|
minimumAmount,
|
|
);
|
|
|
|
mockConfirmTransaction(airdropToSig);
|
|
await connection.confirmTransaction(airdropToSig, 'single');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBalance',
|
|
params: [accountTo.publicKey.toBase58(), {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: minimumAmount,
|
|
},
|
|
},
|
|
]);
|
|
|
|
expect(await connection.getBalance(accountTo.publicKey)).toBe(minimumAmount);
|
|
|
|
mockGetRecentBlockhash('max');
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'1WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
const transaction = new Transaction().add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: accountFrom.publicKey,
|
|
toPubkey: accountTo.publicKey,
|
|
lamports: 10,
|
|
}),
|
|
);
|
|
const signature = await connection.sendTransaction(
|
|
transaction,
|
|
[accountFrom],
|
|
{skipPreflight: true},
|
|
);
|
|
|
|
mockConfirmTransaction(signature);
|
|
let confirmResult = (await connection.confirmTransaction(signature, 'single'))
|
|
.value;
|
|
expect(confirmResult.err).toBeNull();
|
|
|
|
mockGetRecentBlockhash('max');
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'2WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
// Send again and ensure that new blockhash is used
|
|
const lastFetch = Date.now();
|
|
const transaction2 = new Transaction().add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: accountFrom.publicKey,
|
|
toPubkey: accountTo.publicKey,
|
|
lamports: 10,
|
|
}),
|
|
);
|
|
const signature2 = await connection.sendTransaction(
|
|
transaction2,
|
|
[accountFrom],
|
|
{skipPreflight: true},
|
|
);
|
|
expect(signature).not.toEqual(signature2);
|
|
expect(transaction.recentBlockhash).not.toEqual(transaction2.recentBlockhash);
|
|
|
|
mockConfirmTransaction(signature2);
|
|
await connection.confirmTransaction(signature2, 'single');
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
// Send new transaction and ensure that same blockhash is used
|
|
const transaction3 = new Transaction().add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: accountFrom.publicKey,
|
|
toPubkey: accountTo.publicKey,
|
|
lamports: 9,
|
|
}),
|
|
);
|
|
const signature3 = await connection.sendTransaction(
|
|
transaction3,
|
|
[accountFrom],
|
|
{
|
|
skipPreflight: true,
|
|
},
|
|
);
|
|
expect(transaction2.recentBlockhash).toEqual(transaction3.recentBlockhash);
|
|
|
|
mockConfirmTransaction(signature3);
|
|
await connection.confirmTransaction(signature3, 'single');
|
|
|
|
// Sleep until blockhash cache times out
|
|
await sleep(
|
|
Math.max(0, 1000 + BLOCKHASH_CACHE_TIMEOUT_MS - (Date.now() - lastFetch)),
|
|
);
|
|
|
|
mockGetRecentBlockhash('max');
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'sendTransaction',
|
|
},
|
|
{
|
|
error: null,
|
|
result:
|
|
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
|
|
},
|
|
]);
|
|
|
|
const transaction4 = new Transaction().add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: accountFrom.publicKey,
|
|
toPubkey: accountTo.publicKey,
|
|
lamports: 13,
|
|
}),
|
|
);
|
|
|
|
const signature4 = await connection.sendTransaction(
|
|
transaction4,
|
|
[accountFrom],
|
|
{
|
|
skipPreflight: true,
|
|
},
|
|
);
|
|
mockConfirmTransaction(signature4);
|
|
await connection.confirmTransaction(signature4, 'single');
|
|
|
|
expect(transaction4.recentBlockhash).not.toEqual(
|
|
transaction3.recentBlockhash,
|
|
);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBalance',
|
|
params: [accountFrom.publicKey.toBase58(), {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: minimumAmount + 2,
|
|
},
|
|
},
|
|
]);
|
|
|
|
// accountFrom may have less than 100000 due to transaction fees
|
|
const balance = await connection.getBalance(accountFrom.publicKey);
|
|
expect(balance).toBeGreaterThan(0);
|
|
expect(balance).toBeLessThanOrEqual(minimumAmount + 100000);
|
|
|
|
mockRpc.push([
|
|
url,
|
|
{
|
|
method: 'getBalance',
|
|
params: [accountTo.publicKey.toBase58(), {commitment: 'recent'}],
|
|
},
|
|
{
|
|
error: null,
|
|
result: {
|
|
context: {
|
|
slot: 11,
|
|
},
|
|
value: minimumAmount + 42,
|
|
},
|
|
},
|
|
]);
|
|
expect(await connection.getBalance(accountTo.publicKey)).toBe(
|
|
minimumAmount + 42,
|
|
);
|
|
});
|
|
|
|
test('multi-instruction transaction', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const accountFrom = new Account();
|
|
const accountTo = new Account();
|
|
const connection = new Connection(url, 'recent');
|
|
|
|
let signature = await connection.requestAirdrop(
|
|
accountFrom.publicKey,
|
|
LAMPORTS_PER_SOL,
|
|
);
|
|
await connection.confirmTransaction(signature, 'single');
|
|
expect(await connection.getBalance(accountFrom.publicKey)).toBe(
|
|
LAMPORTS_PER_SOL,
|
|
);
|
|
|
|
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
|
|
0,
|
|
'recent',
|
|
);
|
|
|
|
signature = await connection.requestAirdrop(
|
|
accountTo.publicKey,
|
|
minimumAmount + 21,
|
|
);
|
|
await connection.confirmTransaction(signature, 'single');
|
|
expect(await connection.getBalance(accountTo.publicKey)).toBe(
|
|
minimumAmount + 21,
|
|
);
|
|
|
|
// 1. Move(accountFrom, accountTo)
|
|
// 2. Move(accountTo, accountFrom)
|
|
const transaction = new Transaction()
|
|
.add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: accountFrom.publicKey,
|
|
toPubkey: accountTo.publicKey,
|
|
lamports: 100,
|
|
}),
|
|
)
|
|
.add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: accountTo.publicKey,
|
|
toPubkey: accountFrom.publicKey,
|
|
lamports: 100,
|
|
}),
|
|
);
|
|
signature = await connection.sendTransaction(
|
|
transaction,
|
|
[accountFrom, accountTo],
|
|
{skipPreflight: true},
|
|
);
|
|
|
|
await connection.confirmTransaction(signature, 'single');
|
|
|
|
const response = (await connection.getSignatureStatus(signature)).value;
|
|
if (response !== null) {
|
|
expect(typeof response.slot).toEqual('number');
|
|
expect(response.err).toBeNull();
|
|
} else {
|
|
expect(response).not.toBeNull();
|
|
}
|
|
|
|
// accountFrom may have less than LAMPORTS_PER_SOL due to transaction fees
|
|
expect(await connection.getBalance(accountFrom.publicKey)).toBeGreaterThan(0);
|
|
expect(
|
|
await connection.getBalance(accountFrom.publicKey),
|
|
).toBeLessThanOrEqual(LAMPORTS_PER_SOL);
|
|
|
|
expect(await connection.getBalance(accountTo.publicKey)).toBe(
|
|
minimumAmount + 21,
|
|
);
|
|
});
|
|
|
|
test('account change notification', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url, 'recent');
|
|
const owner = new Account();
|
|
const programAccount = new Account();
|
|
|
|
const mockCallback = jest.fn();
|
|
|
|
const subscriptionId = connection.onAccountChange(
|
|
programAccount.publicKey,
|
|
mockCallback,
|
|
'recent',
|
|
);
|
|
|
|
const balanceNeeded = Math.max(
|
|
await connection.getMinimumBalanceForRentExemption(0),
|
|
1,
|
|
);
|
|
|
|
await connection.requestAirdrop(owner.publicKey, LAMPORTS_PER_SOL);
|
|
try {
|
|
const transaction = new Transaction().add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: owner.publicKey,
|
|
toPubkey: programAccount.publicKey,
|
|
lamports: balanceNeeded,
|
|
}),
|
|
);
|
|
await sendAndConfirmTransaction(connection, transaction, [owner], {
|
|
commitment: 'single',
|
|
skipPreflight: true,
|
|
});
|
|
} catch (err) {
|
|
await connection.removeAccountChangeListener(subscriptionId);
|
|
throw err;
|
|
}
|
|
|
|
// Wait for mockCallback to receive a call
|
|
let i = 0;
|
|
for (;;) {
|
|
if (mockCallback.mock.calls.length > 0) {
|
|
break;
|
|
}
|
|
|
|
if (++i === 30) {
|
|
throw new Error('Account change notification not observed');
|
|
}
|
|
// Sleep for a 1/4 of a slot, notifications only occur after a block is
|
|
// processed
|
|
await sleep((250 * DEFAULT_TICKS_PER_SLOT) / NUM_TICKS_PER_SECOND);
|
|
}
|
|
|
|
await connection.removeAccountChangeListener(subscriptionId);
|
|
|
|
expect(mockCallback.mock.calls[0][0].lamports).toBe(balanceNeeded);
|
|
expect(mockCallback.mock.calls[0][0].owner).toEqual(SystemProgram.programId);
|
|
});
|
|
|
|
test('program account change notification', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url, 'recent');
|
|
const owner = new Account();
|
|
const programAccount = new Account();
|
|
|
|
// const mockCallback = jest.fn();
|
|
|
|
const balanceNeeded = await connection.getMinimumBalanceForRentExemption(0);
|
|
|
|
let notified = false;
|
|
const subscriptionId = connection.onProgramAccountChange(
|
|
SystemProgram.programId,
|
|
keyedAccountInfo => {
|
|
if (keyedAccountInfo.accountId !== programAccount.publicKey.toString()) {
|
|
//console.log('Ignoring another account', keyedAccountInfo);
|
|
return;
|
|
}
|
|
expect(keyedAccountInfo.accountInfo.lamports).toBe(balanceNeeded);
|
|
expect(keyedAccountInfo.accountInfo.owner).toEqual(
|
|
SystemProgram.programId,
|
|
);
|
|
notified = true;
|
|
},
|
|
);
|
|
|
|
await connection.requestAirdrop(owner.publicKey, LAMPORTS_PER_SOL);
|
|
try {
|
|
const transaction = new Transaction().add(
|
|
SystemProgram.transfer({
|
|
fromPubkey: owner.publicKey,
|
|
toPubkey: programAccount.publicKey,
|
|
lamports: balanceNeeded,
|
|
}),
|
|
);
|
|
await sendAndConfirmTransaction(connection, transaction, [owner], {
|
|
commitment: 'single',
|
|
skipPreflight: true,
|
|
});
|
|
} catch (err) {
|
|
await connection.removeProgramAccountChangeListener(subscriptionId);
|
|
throw err;
|
|
}
|
|
|
|
// Wait for mockCallback to receive a call
|
|
let i = 0;
|
|
while (!notified) {
|
|
//for (;;) {
|
|
if (++i === 30) {
|
|
throw new Error('Program change notification not observed');
|
|
}
|
|
// Sleep for a 1/4 of a slot, notifications only occur after a block is
|
|
// processed
|
|
await sleep((250 * DEFAULT_TICKS_PER_SLOT) / NUM_TICKS_PER_SECOND);
|
|
}
|
|
|
|
await connection.removeProgramAccountChangeListener(subscriptionId);
|
|
});
|
|
|
|
test('slot notification', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url, 'recent');
|
|
|
|
let notified = false;
|
|
const subscriptionId = connection.onSlotChange(slotInfo => {
|
|
expect(slotInfo.parent).toBeDefined();
|
|
expect(slotInfo.slot).toBeDefined();
|
|
expect(slotInfo.root).toBeDefined();
|
|
expect(slotInfo.slot).toBeGreaterThan(slotInfo.parent);
|
|
expect(slotInfo.slot).toBeGreaterThanOrEqual(slotInfo.root);
|
|
notified = true;
|
|
});
|
|
|
|
// Wait for mockCallback to receive a call
|
|
let i = 0;
|
|
while (!notified) {
|
|
if (++i === 30) {
|
|
throw new Error('Slot change notification not observed');
|
|
}
|
|
// Sleep for a 1/4 of a slot, notifications only occur after a block is
|
|
// processed
|
|
await sleep((250 * DEFAULT_TICKS_PER_SLOT) / NUM_TICKS_PER_SECOND);
|
|
}
|
|
|
|
await connection.removeSlotChangeListener(subscriptionId);
|
|
});
|
|
|
|
test('root notification', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection(url, 'recent');
|
|
|
|
let roots = [];
|
|
const subscriptionId = connection.onRootChange(root => {
|
|
roots.push(root);
|
|
});
|
|
|
|
// Wait for mockCallback to receive a call
|
|
let i = 0;
|
|
while (roots.length < 2) {
|
|
if (++i === 30) {
|
|
throw new Error('Root change notification not observed');
|
|
}
|
|
// Sleep for a 1/4 of a slot, notifications only occur after a block is
|
|
// processed
|
|
await sleep((250 * DEFAULT_TICKS_PER_SLOT) / NUM_TICKS_PER_SECOND);
|
|
}
|
|
|
|
expect(roots[1]).toBeGreaterThan(roots[0]);
|
|
await connection.removeRootChangeListener(subscriptionId);
|
|
});
|
|
|
|
test('https request', async () => {
|
|
if (mockRpcEnabled) {
|
|
console.log('non-live test skipped');
|
|
return;
|
|
}
|
|
|
|
const connection = new Connection('https://devnet.solana.com');
|
|
const version = await connection.getVersion();
|
|
expect(version['solana-core']).toBeTruthy();
|
|
});
|