chore: replace jest with mocha

This commit is contained in:
Justin Starry 2021-02-06 10:59:00 +08:00 committed by Justin Starry
parent 612958ece0
commit c675c67c26
41 changed files with 4635 additions and 6665 deletions

View File

@ -7,4 +7,4 @@
/module.flow.js
/.eslintrc.js
/test/.eslintrc.js
/jest-environment.js
/test/dist

6
web3.js/.mocharc.js Normal file
View File

@ -0,0 +1,6 @@
'use strict';
// Configure Node.js tests
module.exports = {
require: ['@babel/register', 'esm'],
};

1
web3.js/.prettierignore Normal file
View File

@ -0,0 +1 @@
test/dist

View File

@ -15,3 +15,4 @@ npm run codecov
make -C examples/bpf-c-noop/
cargo build-bpf --manifest-path examples/bpf-rust-noop/Cargo.toml
npm run test:live-with-test-validator
npm run test:browser-with-test-validator

View File

@ -24,7 +24,7 @@ eslint and flow-type are used.
Helpful link: https://www.saltycrane.com/flow-type-cheat-sheet/latest/
### Testing Framework
https://jestjs.io/
https://mochajs.org/
### API Documentation
ESDoc is used to document the public API. See

View File

@ -5,11 +5,5 @@
],
"plugins": [
"@babel/plugin-proposal-class-properties"
],
"env": {
"test": {
"plugins": ["@babel/plugin-transform-runtime"],
"presets": ["@babel/preset-flow"]
}
}
]
}

File diff suppressed because it is too large Load Diff

View File

@ -1,27 +0,0 @@
const NodeEnvironment = require('jest-environment-node');
class CustomEnvironment extends NodeEnvironment {
constructor(config, context) {
Object.assign(config.globals, {
Uint8Array,
ArrayBuffer,
});
super(config, context);
this.testPath = context.testPath;
this.docblockPragmas = context.docblockPragmas;
}
async setup() {
await super.setup();
}
async teardown() {
await super.teardown();
}
runScript(script) {
return super.runScript(script);
}
}
module.exports = CustomEnvironment;

View File

@ -1,5 +0,0 @@
{
"collectCoverage": true,
"collectCoverageFrom": ["src/**"],
"coverageReporters": ["json", "lcov", "text-summary", "html"]
}

21
web3.js/mocha.html Normal file
View File

@ -0,0 +1,21 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<title>Mocha Tests</title>
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<link rel="stylesheet" href="./node_modules/mocha/mocha.css" />
</head>
<body>
<div id="mocha"></div>
<script src="./node_modules/mocha/mocha.js"></script>
<script class="mocha-init">
mocha.setup('bdd');
mocha.checkLeaks();
</script>
<script type="module" src="./test/dist/bundle.js"></script>
<script type="module" class="mocha-exec">
mocha.run();
</script>
</body>
</html>

View File

@ -46,6 +46,7 @@
"bpf-sdk:install": "npm run clean:fixtures; bin/bpf-sdk-install.sh .",
"bpf-sdk:remove-symlinks": "find bpf-sdk -type l -print -exec cp {} {}.tmp \\; -exec mv {}.tmp {} \\;",
"build": "cross-env NODE_ENV=production rollup -c",
"build:browser-test": "rollup -c test/rollup.config.js",
"build:fixtures": "set -ex; ./test/fixtures/noop-c/build.sh; ./test/fixtures/noop-rust/build.sh",
"clean:fixtures": "make -C examples/bpf-c-noop clean ",
"clean": "rimraf ./coverage ./lib",
@ -70,14 +71,13 @@
"pretty": "prettier --check '{,{examples,src,test}/**/}*.{j,t}s'",
"pretty:fix": "prettier --write '{,{examples,src,test}/**/}*.{j,t}s'",
"re": "semantic-release --repository-url git@github.com:solana-labs/solana-web3.js.git",
"test": "npm run build:fixtures && cross-env NODE_ENV=test jest --useStderr",
"test:cover": "npm run build:fixtures && cross-env NODE_ENV=test jest --coverage --useStderr",
"test:live": "npm run build:fixtures && cross-env NODE_ENV=test TEST_LIVE=1 jest --useStderr",
"test": "npm run build:fixtures && mocha './test/**/*.test.js'",
"test:browser": "TEST_LIVE=1 npm run build:fixtures && npm run build:browser-test && mocha-headless-chrome -f http://localhost:8080/mocha.html --timeout 180000",
"test:browser-with-server": "start-server-and-test 'http-server -p 8080' 8080 test:browser",
"test:browser-with-test-validator": "start-server-and-test 'solana-test-validator --reset --quiet' http://localhost:8899/health test:browser-with-server",
"test:live": "TEST_LIVE=1 npm run test",
"test:live-with-test-validator": "start-server-and-test 'solana-test-validator --reset --quiet' http://localhost:8899/health test:live",
"test:watch": "npm run build:fixtures && cross-env NODE_ENV=test jest --watch --useStderr"
},
"jest": {
"testEnvironment": "./jest-environment"
"test:watch": "npm run build:fixtures && npm run test"
},
"dependencies": {
"@babel/runtime": "^7.12.5",
@ -95,15 +95,19 @@
"tweetnacl": "^1.0.0"
},
"devDependencies": {
"@babel/core": "^7.12.13",
"@babel/plugin-proposal-class-properties": "^7.12.1",
"@babel/plugin-transform-runtime": "^7.12.10",
"@babel/preset-env": "^7.12.11",
"@babel/preset-flow": "^7.12.1",
"@babel/register": "^7.12.13",
"@commitlint/config-conventional": "^11.0.0",
"@commitlint/travis-cli": "^11.0.0",
"@rollup/plugin-alias": "^3.1.2",
"@rollup/plugin-babel": "^5.2.3",
"@rollup/plugin-commonjs": "^17.1.0",
"@rollup/plugin-json": "^4.1.0",
"@rollup/plugin-multi-entry": "^4.0.0",
"@rollup/plugin-node-resolve": "^11.1.1",
"@rollup/plugin-replace": "^2.3.4",
"@solana/spl-token": "^0.0.13",
@ -111,6 +115,8 @@
"@typescript-eslint/parser": "^4.14.2",
"acorn": "^8.0.1",
"babel-eslint": "^11.0.0-beta.2",
"chai": "^4.3.0",
"chai-as-promised": "^7.1.1",
"codecov": "^3.0.4",
"cross-env": "7.0.3",
"elfy": "^1.0.0",
@ -121,22 +127,21 @@
"esdoc-importpath-plugin": "^1.0.2",
"esdoc-inject-style-plugin": "^1.0.0",
"esdoc-standard-plugin": "^1.0.0",
<<<<<<< HEAD
"eslint": "7.19.0",
=======
"eslint": "^7.19.0",
>>>>>>> 37215dbc2... feat: add support for browser es modules
"eslint-config-prettier": "^7.0.0",
"eslint-plugin-flowtype": "^5.2.0",
"eslint-plugin-import": "2.22.1",
"eslint-plugin-jest": "22.19.0",
"eslint-plugin-prettier": "^3.0.0",
"esm": "^3.2.25",
"flow-bin": "0.130.0",
"flow-remove-types": "^2.143.1",
"flow-typed": "3.2.1",
"fs-file-tree": "1.1.1",
"jest": "26.6.3",
"http-server": "^0.12.3",
"marked": "^1.1.0",
"mocha": "^8.2.1",
"mocha-headless-chrome": "^3.1.0",
"mockttp": "^1.1.0",
"mz": "^2.7.0",
"npm-run-all": "^4.1.5",
"prettier": "^2.0.0",
@ -146,8 +151,11 @@
"rollup-plugin-node-polyfills": "^0.2.1",
"rollup-plugin-terser": "^7.0.2",
"semantic-release": "^17.0.2",
"start-server-and-test": "^1.11.6",
"sinon": "^9.2.4",
"start-server-and-test": "^1.12.0",
"typescript": "^4.1.3",
"watch": "^1.0.2"
"watch": "^1.0.2",
"webpack": "^5.21.0",
"webpack-cli": "^4.5.0"
}
}

View File

@ -19,7 +19,11 @@ function generateConfig(configType, format) {
plugins: [
flow(),
commonjs(),
nodeResolve({browser, preferBuiltins: !browser, dedupe: ['bn.js']}),
nodeResolve({
browser,
preferBuiltins: !browser,
dedupe: ['bn.js', 'buffer'],
}),
babel({
exclude: '**/node_modules/**',
babelHelpers: bundle ? 'bundled' : 'runtime',

View File

@ -1,4 +0,0 @@
module.exports = {
// eslint-disable-line import/no-commonjs
extends: ['plugin:jest/recommended', '../.eslintrc.js'],
};

1
web3.js/test/.gitignore vendored Normal file
View File

@ -0,0 +1 @@
dist

View File

@ -1,86 +0,0 @@
// @flow
import fetch from 'node-fetch';
type RpcRequest = {
method: string,
params?: Array<any>,
};
type RpcResponseError = {
message: string,
};
type RpcResponseResult = any;
type RpcResponse = {
error: ?RpcResponseError,
result: ?RpcResponseResult,
};
export const mockRpc: Array<[string, RpcRequest, RpcResponse]> = [];
// Define TEST_LIVE in the environment to test against the real full node
// identified by `url` instead of using the mock
export const mockRpcEnabled = !process.env.TEST_LIVE;
let mockNotice = true;
// Suppress lint: 'JestMockFn' is not defined
// eslint-disable-next-line no-undef
const mock: JestMockFn<any, any> = jest.fn((fetchUrl, fetchOptions) => {
if (!mockRpcEnabled) {
if (mockNotice) {
console.log(
`Note: node-fetch mock is disabled, testing live against ${fetchUrl}`,
);
mockNotice = false;
}
return fetch(fetchUrl, fetchOptions);
}
expect(mockRpc.length).toBeGreaterThanOrEqual(1);
const [mockUrl, mockRequest, mockResponse] = mockRpc.shift();
expect(fetchUrl).toBe(mockUrl);
expect(fetchOptions).toMatchObject({
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
});
expect(fetchOptions.body).toBeDefined();
const body = JSON.parse(fetchOptions.body);
expect(body).toMatchObject(
Object.assign(
{},
{
jsonrpc: '2.0',
method: 'invalid',
},
mockRequest,
),
);
const response = Object.assign(
{},
{
jsonrpc: '2.0',
id: body.id,
error: {
message: 'invalid error message',
},
result: 'invalid response',
},
mockResponse,
);
return {
ok: true,
status: 200,
statusText: 'OK',
text: () => {
return Promise.resolve(JSON.stringify(response));
},
};
});
export default mock;

View File

@ -1,77 +0,0 @@
// @flow
import {Client as LiveClient} from 'rpc-websockets';
import EventEmitter from 'events';
type RpcRequest = {
method: string,
params?: Array<any>,
};
type RpcResponse = {
context: {
slot: number,
},
value: any,
};
// Define TEST_LIVE in the environment to test against the real full node
// identified by `url` instead of using the mock
export const mockRpcEnabled = !process.env.TEST_LIVE;
export const mockRpcSocket: Array<[RpcRequest, RpcResponse]> = [];
class MockClient extends EventEmitter {
mockOpen = false;
subscriptionCounter = 0;
constructor() {
super();
}
connect() {
if (!this.mockOpen) {
this.mockOpen = true;
this.emit('open');
}
}
close() {
if (this.mockOpen) {
this.mockOpen = false;
this.emit('close');
}
}
notify(): Promise<any> {
return Promise.resolve();
}
on(event: string, callback: Function): this {
return super.on(event, callback);
}
call(method: string, params: Array<any>): Promise<Object> {
expect(mockRpcSocket.length).toBeGreaterThanOrEqual(1);
const [mockRequest, mockResponse] = mockRpcSocket.shift();
expect(method).toBe(mockRequest.method);
expect(params).toMatchObject(mockRequest.params);
let id = this.subscriptionCounter++;
const response = {
subscription: id,
result: mockResponse,
};
setImmediate(() => {
const eventName = method.replace('Subscribe', 'Notification');
this.emit(eventName, response);
});
return Promise.resolve(id);
}
}
const Client = mockRpcEnabled ? MockClient : LiveClient;
export {Client};

View File

@ -1,12 +1,15 @@
// @flow
import {Account} from '../src/account';
import {expect} from 'chai';
import {Buffer} from 'buffer';
test('generate new account', () => {
describe('Account', () => {
it('generate new account', () => {
const account = new Account();
expect(account.secretKey).toHaveLength(64);
});
expect(account.secretKey).to.have.length(64);
});
test('account from secret key', () => {
it('account from secret key', () => {
const secretKey = Buffer.from([
153,
218,
@ -74,7 +77,8 @@ test('account from secret key', () => {
34,
]);
const account = new Account(secretKey);
expect(account.publicKey.toBase58()).toBe(
expect(account.publicKey.toBase58()).to.eq(
'2q7pyhPwAwZ3QMfZrnAbDhnh9mDUqycszcpf86VgQxhF',
);
});
});

View File

@ -1,40 +1,41 @@
// @flow
import {AgentManager, DESTROY_TIMEOUT_MS} from '../src/agent-manager';
import {expect} from 'chai';
import {sleep} from '../src/util/sleep';
jest.setTimeout(10 * 1000);
test('agent manager', async () => {
describe('AgentManager', () => {
it('works', async () => {
const manager = new AgentManager();
const agent = manager._agent;
expect(manager._activeRequests).toBe(0);
expect(manager._destroyTimeout).toBeNull();
expect(manager._activeRequests).to.eq(0);
expect(manager._destroyTimeout).to.be.null;
manager.requestStart();
expect(manager._activeRequests).toBe(1);
expect(manager._destroyTimeout).toBeNull();
expect(manager._activeRequests).to.eq(1);
expect(manager._destroyTimeout).to.be.null;
manager.requestEnd();
expect(manager._activeRequests).toBe(0);
expect(manager._destroyTimeout).not.toBeNull();
expect(manager._activeRequests).to.eq(0);
expect(manager._destroyTimeout).not.to.be.null;
manager.requestStart();
manager.requestStart();
expect(manager._activeRequests).toBe(2);
expect(manager._destroyTimeout).toBeNull();
expect(manager._activeRequests).to.eq(2);
expect(manager._destroyTimeout).to.be.null;
manager.requestEnd();
manager.requestEnd();
expect(manager._activeRequests).toBe(0);
expect(manager._destroyTimeout).not.toBeNull();
expect(manager._agent).toBe(agent);
expect(manager._activeRequests).to.eq(0);
expect(manager._destroyTimeout).not.to.be.null;
expect(manager._agent).to.eq(agent);
await sleep(DESTROY_TIMEOUT_MS);
expect(manager._agent).not.toBe(agent);
expect(manager._agent).not.to.eq(agent);
}).timeout(2 * DESTROY_TIMEOUT_MS);
});

View File

@ -1,6 +1,8 @@
// @flow
import fs from 'mz/fs';
import {expect, use} from 'chai';
import chaiAsPromised from 'chai-as-promised';
import {
Connection,
@ -9,22 +11,15 @@ import {
sendAndConfirmTransaction,
Account,
} from '../src';
import {mockRpcEnabled} from './__mocks__/node-fetch';
import {url} from './url';
import {newAccountWithLamports} from './new-account-with-lamports';
import {BPF_LOADER_PROGRAM_ID} from '../src/bpf-loader';
import {helpers} from './mocks/rpc-http';
if (!mockRpcEnabled) {
// The default of 5 seconds is too slow for live testing sometimes
jest.setTimeout(240000);
}
test('load BPF C program', async () => {
if (mockRpcEnabled) {
console.log('non-live test skipped');
return;
}
use(chaiAsPromised);
if (process.env.TEST_LIVE) {
describe('BPF Loader', () => {
it('load BPF C program', async () => {
const data = await fs.readFile('test/fixtures/noop-c/noop.so');
const connection = new Connection(url, 'singleGossip');
@ -32,21 +27,32 @@ test('load BPF C program', async () => {
const fees =
feeCalculator.lamportsPerSignature *
BpfLoader.getMinNumSignatures(data.length);
const payerBalance = await connection.getMinimumBalanceForRentExemption(0);
const payerBalance = await connection.getMinimumBalanceForRentExemption(
0,
);
const executableBalance = await connection.getMinimumBalanceForRentExemption(
data.length,
);
const from = await newAccountWithLamports(
const from = new Account();
await helpers.airdrop({
connection,
payerBalance + fees + executableBalance,
);
address: from.publicKey,
amount: payerBalance + fees + executableBalance,
});
const program = new Account();
await BpfLoader.load(connection, from, program, data, BPF_LOADER_PROGRAM_ID);
await BpfLoader.load(
connection,
from,
program,
data,
BPF_LOADER_PROGRAM_ID,
);
// Check that program loading costed exactly `fees + executableBalance`
const fromBalance = await connection.getBalance(from.publicKey);
expect(fromBalance).toEqual(payerBalance);
expect(fromBalance).to.eq(payerBalance);
const transaction = new Transaction().add({
keys: [{pubkey: from.publicKey, isSigner: true, isWritable: true}],
@ -56,21 +62,17 @@ test('load BPF C program', async () => {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
});
describe('load BPF Rust program', () => {
if (mockRpcEnabled) {
console.log('non-live test skipped');
return;
}
}).timeout(5000);
describe('load BPF Rust program', () => {
const connection = new Connection(url, 'singleGossip');
let program: Account;
let payerAccount: Account;
let program = new Account();
let payerAccount = new Account();
let programData: Buffer;
beforeAll(async () => {
before(async function () {
this.timeout(60_000);
programData = await fs.readFile(
'test/fixtures/noop-rust/solana_bpf_rust_noop.so',
);
@ -79,24 +81,33 @@ describe('load BPF Rust program', () => {
const fees =
feeCalculator.lamportsPerSignature *
BpfLoader.getMinNumSignatures(programData.length);
const payerBalance = await connection.getMinimumBalanceForRentExemption(0);
const payerBalance = await connection.getMinimumBalanceForRentExemption(
0,
);
const executableBalance = await connection.getMinimumBalanceForRentExemption(
programData.length,
);
payerAccount = await newAccountWithLamports(
await helpers.airdrop({
connection,
payerBalance + fees + executableBalance,
);
address: payerAccount.publicKey,
amount: payerBalance + fees + executableBalance,
});
// Create program account with low balance
program = await newAccountWithLamports(connection, executableBalance - 1);
await helpers.airdrop({
connection,
address: program.publicKey,
amount: executableBalance - 1,
});
// First load will fail part way due to lack of funds
const insufficientPayerAccount = await newAccountWithLamports(
const insufficientPayerAccount = new Account();
await helpers.airdrop({
connection,
2 * feeCalculator.lamportsPerSignature * 8,
);
address: insufficientPayerAccount.publicKey,
amount: 2 * feeCalculator.lamportsPerSignature * 8,
});
const failedLoad = BpfLoader.load(
connection,
@ -105,7 +116,7 @@ describe('load BPF Rust program', () => {
programData,
BPF_LOADER_PROGRAM_ID,
);
await expect(failedLoad).rejects.toThrow();
await expect(failedLoad).to.be.rejected;
// Second load will succeed
await BpfLoader.load(
@ -117,7 +128,7 @@ describe('load BPF Rust program', () => {
);
});
test('get confirmed transaction', async () => {
it('get confirmed transaction', async () => {
const transaction = new Transaction().add({
keys: [
{pubkey: payerAccount.publicKey, isSigner: true, isWritable: true},
@ -135,23 +146,25 @@ describe('load BPF Rust program', () => {
},
);
const parsedTx = await connection.getParsedConfirmedTransaction(signature);
const parsedTx = await connection.getParsedConfirmedTransaction(
signature,
);
if (parsedTx === null) {
expect(parsedTx).not.toBeNull();
expect(parsedTx).not.to.be.null;
return;
}
const {signatures, message} = parsedTx.transaction;
expect(signatures[0]).toEqual(signature);
expect(signatures[0]).to.eq(signature);
const ix = message.instructions[0];
if (ix.parsed) {
expect('parsed' in ix).toBe(false);
expect('parsed' in ix).to.eq(false);
} else {
expect(ix.programId.equals(program.publicKey)).toBe(true);
expect(ix.data).toEqual('');
expect(ix.programId).to.eql(program.publicKey);
expect(ix.data).to.eq('');
}
});
}).timeout(30000);
test('simulate transaction', async () => {
it('simulate transaction', async () => {
const simulatedTransaction = new Transaction().add({
keys: [
{pubkey: payerAccount.publicKey, isSigner: true, isWritable: true},
@ -160,25 +173,27 @@ describe('load BPF Rust program', () => {
});
const {err, logs} = (
await connection.simulateTransaction(simulatedTransaction, [payerAccount])
await connection.simulateTransaction(simulatedTransaction, [
payerAccount,
])
).value;
expect(err).toBeNull();
expect(err).to.be.null;
if (logs === null) {
expect(logs).not.toBeNull();
expect(logs).not.to.be.null;
return;
}
expect(logs.length).toBeGreaterThanOrEqual(2);
expect(logs[0]).toEqual(
expect(logs.length).to.be.at.least(2);
expect(logs[0]).to.eq(
`Program ${program.publicKey.toBase58()} invoke [1]`,
);
expect(logs[logs.length - 1]).toEqual(
expect(logs[logs.length - 1]).to.eq(
`Program ${program.publicKey.toBase58()} success`,
);
});
test('deprecated - simulate transaction without signature verification', async () => {
it('deprecated - simulate transaction without signature verification', async () => {
const simulatedTransaction = new Transaction().add({
keys: [
{pubkey: payerAccount.publicKey, isSigner: true, isWritable: true},
@ -190,23 +205,23 @@ describe('load BPF Rust program', () => {
const {err, logs} = (
await connection.simulateTransaction(simulatedTransaction)
).value;
expect(err).toBeNull();
expect(err).to.be.null;
if (logs === null) {
expect(logs).not.toBeNull();
expect(logs).not.to.be.null;
return;
}
expect(logs.length).toBeGreaterThanOrEqual(2);
expect(logs[0]).toEqual(
expect(logs.length).to.be.at.least(2);
expect(logs[0]).to.eq(
`Program ${program.publicKey.toBase58()} invoke [1]`,
);
expect(logs[logs.length - 1]).toEqual(
expect(logs[logs.length - 1]).to.eq(
`Program ${program.publicKey.toBase58()} success`,
);
});
test('simulate transaction without signature verification', async () => {
it('simulate transaction without signature verification', async () => {
const simulatedTransaction = new Transaction({
feePayer: payerAccount.publicKey,
}).add({
@ -219,23 +234,23 @@ describe('load BPF Rust program', () => {
const {err, logs} = (
await connection.simulateTransaction(simulatedTransaction)
).value;
expect(err).toBeNull();
expect(err).to.be.null;
if (logs === null) {
expect(logs).not.toBeNull();
expect(logs).not.to.be.null;
return;
}
expect(logs.length).toBeGreaterThanOrEqual(2);
expect(logs[0]).toEqual(
expect(logs.length).to.be.at.least(2);
expect(logs[0]).to.eq(
`Program ${program.publicKey.toBase58()} invoke [1]`,
);
expect(logs[logs.length - 1]).toEqual(
expect(logs[logs.length - 1]).to.eq(
`Program ${program.publicKey.toBase58()} success`,
);
});
test('simulate transaction with bad programId', async () => {
it('simulate transaction with bad programId', async () => {
const simulatedTransaction = new Transaction().add({
keys: [
{pubkey: payerAccount.publicKey, isSigner: true, isWritable: true},
@ -247,17 +262,17 @@ describe('load BPF Rust program', () => {
const {err, logs} = (
await connection.simulateTransaction(simulatedTransaction)
).value;
expect(err).toEqual('ProgramAccountNotFound');
expect(err).to.eq('ProgramAccountNotFound');
if (logs === null) {
expect(logs).not.toBeNull();
expect(logs).not.to.be.null;
return;
}
expect(logs.length).toEqual(0);
expect(logs).to.have.length(0);
});
test('reload program', async () => {
it('reload program', async () => {
expect(
await BpfLoader.load(
connection,
@ -266,6 +281,8 @@ describe('load BPF Rust program', () => {
programData,
BPF_LOADER_PROGRAM_ID,
),
).toBe(false);
).to.eq(false);
});
});
});
});
}

View File

@ -1,16 +1,19 @@
// @flow
import {clusterApiUrl} from '../src/util/cluster';
import {expect} from 'chai';
test('invalid', () => {
describe('Cluster Util', () => {
it('invalid', () => {
expect(() => {
// $FlowExpectedError
clusterApiUrl('abc123');
}).toThrow();
});
}).to.throw();
});
test('devnet', () => {
expect(clusterApiUrl()).toEqual('https://devnet.solana.com');
expect(clusterApiUrl('devnet')).toEqual('https://devnet.solana.com');
expect(clusterApiUrl('devnet', true)).toEqual('https://devnet.solana.com');
expect(clusterApiUrl('devnet', false)).toEqual('http://devnet.solana.com');
it('devnet', () => {
expect(clusterApiUrl()).to.eq('https://devnet.solana.com');
expect(clusterApiUrl('devnet')).to.eq('https://devnet.solana.com');
expect(clusterApiUrl('devnet', true)).to.eq('https://devnet.solana.com');
expect(clusterApiUrl('devnet', false)).to.eq('http://devnet.solana.com');
});
});

File diff suppressed because it is too large Load Diff

View File

@ -1,7 +0,0 @@
#!/usr/bin/env bash
set -ex
cd "$(dirname "$0")"
make -C ../../../examples/bpf-c-noop/
cp ../../../examples/bpf-c-noop/out/noop.so .

View File

@ -1,7 +0,0 @@
#!/usr/bin/env bash
set -ex
cd "$(dirname "$0")"
cargo build-bpf --manifest-path=../../../examples/bpf-rust-noop/Cargo.toml
cp ../../../examples/bpf-rust-noop/target/deploy/solana_bpf_rust_noop.so .

View File

@ -1,83 +0,0 @@
// @flow
import {Account} from '../src/account';
import {SystemProgram} from '../src/system-program';
import {Transaction} from '../src/transaction';
test('verify getConfirmedBlock', () => {
const account0 = new Account();
const account1 = new Account();
const account2 = new Account();
const account3 = new Account();
const recentBlockhash = account1.publicKey.toBase58(); // Fake recentBlockhash
// Create a couple signed transactions
const transfer0 = SystemProgram.transfer({
fromPubkey: account0.publicKey,
toPubkey: account1.publicKey,
lamports: 123,
});
const transaction0 = new Transaction({recentBlockhash}).add(transfer0);
transaction0.sign(account0);
const transfer1 = SystemProgram.transfer({
fromPubkey: account2.publicKey,
toPubkey: account3.publicKey,
lamports: 456,
});
let transaction1 = new Transaction({recentBlockhash}).add(transfer1);
transaction1.sign(account2);
// Build ConfirmedBlock, with dummy data for blockhashes, balances
const confirmedBlock = {
blockhash: recentBlockhash,
previousBlockhash: recentBlockhash,
transactions: [
{
transaction: transaction0,
meta: {
fee: 0,
preBalances: [100000, 100000, 1, 1, 1],
postBalances: [99877, 100123, 1, 1, 1],
status: {Ok: null},
err: null,
},
},
{
transaction: transaction1,
meta: {
fee: 0,
preBalances: [100000, 100000, 1, 1, 1],
postBalances: [99544, 100456, 1, 1, 1],
status: {Ok: null},
err: null,
},
},
],
rewards: [],
};
// Verify signatures in ConfirmedBlock
for (const transactionWithMeta of confirmedBlock.transactions) {
expect(transactionWithMeta.transaction.verifySignatures()).toBe(true);
}
const bogusSignature = {
signature: Buffer.alloc(64, 9),
publicKey: account2.publicKey,
};
transaction1.signatures[0] = bogusSignature;
let badConfirmedBlock = confirmedBlock;
badConfirmedBlock.transactions[1].transaction = transaction1;
// Verify signatures in ConfirmedBlock
const verifications = badConfirmedBlock.transactions.map(
transactionWithMeta => transactionWithMeta.transaction.verifySignatures(),
);
expect(
verifications.reduce(
(accumulator, currentValue) => accumulator && currentValue,
),
).toBe(false);
});

View File

@ -1,19 +0,0 @@
// @flow
import type {TransactionSignature} from '../../src/transaction';
import {mockRpcSocket} from '../__mocks__/rpc-websockets';
export function mockConfirmTransaction(signature: TransactionSignature) {
mockRpcSocket.push([
{
method: 'signatureSubscribe',
params: [signature, {commitment: 'singleGossip'}],
},
{
context: {
slot: 11,
},
value: {err: null},
},
]);
}

View File

@ -1,36 +0,0 @@
// @flow
import {Account} from '../../src';
import type {Commitment} from '../../src/connection';
import {url} from '../url';
import {mockRpc} from '../__mocks__/node-fetch';
export function mockGetRecentBlockhash(commitment: ?Commitment) {
const recentBlockhash = new Account();
const params = [];
if (commitment) {
params.push({commitment});
}
mockRpc.push([
url,
{
method: 'getRecentBlockhash',
params,
},
{
error: null,
result: {
context: {
slot: 11,
},
value: {
blockhash: recentBlockhash.publicKey.toBase58(),
feeCalculator: {
lamportsPerSignature: 42,
},
},
},
},
]);
}

View File

@ -0,0 +1,170 @@
// @flow
import bs58 from 'bs58';
import BN from 'bn.js';
import * as mockttp from 'mockttp';
import {mockRpcMessage} from './rpc-websockets';
import {Connection} from '../../src';
import type {Commitment} from '../../src/connection';
export const mockServer = process.env.TEST_LIVE || mockttp.getLocal();
let uniqueCounter = 0;
export const uniqueSignature = () => {
return bs58.encode(new BN(++uniqueCounter).toArray(null, 64));
};
export const uniqueBlockhash = () => {
return bs58.encode(new BN(++uniqueCounter).toArray(null, 32));
};
export const mockErrorMessage = 'Invalid';
export const mockErrorResponse = {
code: -32602,
message: mockErrorMessage,
};
export const mockRpcResponse = async ({
method,
params,
value,
error,
withContext,
}: {
method: string,
params: Array<any>,
value: any,
error: any,
withContext?: boolean,
}) => {
if (process.env.TEST_LIVE) return;
let result = withContext
? {
context: {
slot: 11,
},
value,
}
: value;
await mockServer
.post('/')
.withJsonBodyIncluding({
jsonrpc: '2.0',
method,
params,
})
.thenReply(
200,
JSON.stringify({
jsonrpc: '2.0',
id: '',
error,
result,
}),
);
};
const recentBlockhash = async ({
connection,
commitment,
}: {
connection: Connection,
commitment: ?Commitment,
}) => {
const blockhash = uniqueBlockhash();
const params = [];
if (commitment) {
params.push({commitment});
}
await mockRpcResponse({
method: 'getRecentBlockhash',
params,
value: {
blockhash,
feeCalculator: {
lamportsPerSignature: 42,
},
},
withContext: true,
});
return await connection.getRecentBlockhash(commitment);
};
const processTransaction = async ({
connection,
transaction,
signers,
commitment,
err,
}: {
connection: Connection,
transaction: Transaction,
signers: Array<Account>,
commitment: Commitment,
err?: any,
}) => {
const blockhash = (await recentBlockhash({connection})).blockhash;
transaction.recentBlockhash = blockhash;
transaction.sign(...signers);
const encoded = transaction.serialize().toString('base64');
const signature = bs58.encode(transaction.signature);
await mockRpcResponse({
method: 'sendTransaction',
params: [encoded],
value: signature,
});
const sendOptions = err
? {
skipPreflight: true,
}
: {
preflightCommitment: commitment,
};
await connection.sendEncodedTransaction(encoded, sendOptions);
await mockRpcMessage({
method: 'signatureSubscribe',
params: [signature, {commitment}],
result: {err: err || null},
});
return await connection.confirmTransaction(signature, commitment);
};
const airdrop = async ({
connection,
address,
amount,
}: {
connection: Connection,
address: PublicKey,
amount: number,
}) => {
await mockRpcResponse({
method: 'requestAirdrop',
params: [address.toBase58(), amount],
value: uniqueSignature(),
});
const signature = await connection.requestAirdrop(address, amount);
await mockRpcMessage({
method: 'signatureSubscribe',
params: [signature, {commitment: 'singleGossip'}],
result: {err: null},
});
await connection.confirmTransaction(signature, 'singleGossip');
return signature;
};
export const helpers = {
airdrop,
processTransaction,
recentBlockhash,
};

View File

@ -0,0 +1,106 @@
// @flow
import {Client as LiveClient} from 'rpc-websockets';
import {expect} from 'chai';
import sinon from 'sinon';
import {Connection} from '../../src';
type RpcRequest = {
method: string,
params?: Array<any>,
};
type RpcResponse = {
context: {
slot: number,
},
value: any,
};
const mockRpcSocket: Array<[RpcRequest, RpcResponse]> = [];
const sandbox = sinon.createSandbox();
export const mockRpcMessage = ({
method,
params,
result,
}: {
method: string,
params: Array<any>,
result: any,
}) => {
mockRpcSocket.push([
{method, params},
{
context: {slot: 11},
value: result,
},
]);
};
export const stubRpcWebSocket = (connection: Connection) => {
const rpcWebSocket = connection._rpcWebSocket;
const mockClient = new MockClient(rpcWebSocket);
sandbox.stub(rpcWebSocket, 'connect').callsFake(() => {
mockClient.connect();
});
sandbox.stub(rpcWebSocket, 'close').callsFake(() => {
mockClient.close();
});
sandbox.stub(rpcWebSocket, 'call').callsFake((method, params) => {
return mockClient.call(method, params);
});
};
export const restoreRpcWebSocket = (connection: Connection) => {
connection._rpcWebSocket.close();
if (connection._rpcWebSocketIdleTimeout !== null) {
clearTimeout(connection._rpcWebSocketIdleTimeout);
connection._rpcWebSocketIdleTimeout = null;
}
sandbox.restore();
};
class MockClient {
mockOpen = false;
subscriptionCounter = 0;
constructor(rpcWebSocket: LiveClient) {
this.client = rpcWebSocket;
}
connect() {
if (!this.mockOpen) {
this.mockOpen = true;
this.client.emit('open');
}
}
close() {
if (this.mockOpen) {
this.mockOpen = false;
this.client.emit('close');
}
}
call(method: string, params: Array<any>): Promise<Object> {
expect(mockRpcSocket.length).to.be.at.least(1);
const [mockRequest, mockResponse] = mockRpcSocket.shift();
expect(method).to.eq(mockRequest.method);
expect(params).to.eql(mockRequest.params);
let id = ++this.subscriptionCounter;
const response = {
subscription: id,
result: mockResponse,
};
setImmediate(() => {
const eventName = method.replace('Subscribe', 'Notification');
this.client.emit(eventName, response);
});
return Promise.resolve(id);
}
}

View File

@ -1,36 +0,0 @@
// @flow
import {Account, Connection} from '../src';
import {mockRpc} from './__mocks__/node-fetch';
import {url} from './url';
export async function newAccountWithLamports(
connection: Connection,
lamports: number = 1000000,
): Promise<Account> {
const account = new Account();
{
mockRpc.push([
url,
{
method: 'requestAirdrop',
params: [account.publicKey.toBase58(), lamports],
},
{
error: null,
// Signature doesn't matter
result:
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
}
const signature = await connection.requestAirdrop(
account.publicKey,
lamports,
);
await connection.confirmTransaction(signature, 'singleGossip');
return account;
}

View File

@ -1,6 +1,8 @@
// @flow
import bs58 from 'bs58';
import {Buffer} from 'buffer';
import {expect} from 'chai';
import {
Account,
@ -10,15 +12,9 @@ import {
PublicKey,
} from '../src';
import {NONCE_ACCOUNT_LENGTH} from '../src/nonce-account';
import {mockRpc, mockRpcEnabled} from './__mocks__/node-fetch';
import {mockGetRecentBlockhash} from './mockrpc/get-recent-blockhash';
import {url} from './url';
import {mockConfirmTransaction} from './mockrpc/confirm-transaction';
if (!mockRpcEnabled) {
// Testing max commitment level takes around 20s to complete
jest.setTimeout(30000);
}
import {MOCK_PORT, url} from './url';
import {helpers, mockRpcResponse, mockServer} from './mocks/rpc-http';
import {stubRpcWebSocket, restoreRpcWebSocket} from './mocks/rpc-websockets';
const expectedData = (authorizedPubkey: PublicKey): [string, string] => {
const expectedData = Buffer.alloc(NONCE_ACCOUNT_LENGTH);
@ -31,79 +27,43 @@ const expectedData = (authorizedPubkey: PublicKey): [string, string] => {
return [expectedData.toString('base64'), 'base64'];
};
test('create and query nonce account', async () => {
describe('Nonce', () => {
let connection: Connection;
beforeEach(() => {
connection = new Connection(url);
});
if (!process.env.TEST_LIVE) {
beforeEach(() => {
mockServer.start(MOCK_PORT);
stubRpcWebSocket(connection);
});
afterEach(() => {
mockServer.stop();
restoreRpcWebSocket(connection);
});
}
it('create and query nonce account', async () => {
const from = new Account();
const nonceAccount = new Account();
const connection = new Connection(url, 'singleGossip');
mockRpc.push([
url,
{
await mockRpcResponse({
method: 'getMinimumBalanceForRentExemption',
params: [NONCE_ACCOUNT_LENGTH, {commitment: 'singleGossip'}],
},
{
error: null,
result: 50,
},
]);
params: [NONCE_ACCOUNT_LENGTH],
value: 50,
});
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
NONCE_ACCOUNT_LENGTH,
);
mockRpc.push([
url,
{
method: 'requestAirdrop',
params: [from.publicKey.toBase58(), minimumAmount * 2],
},
{
error: null,
result:
'1WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
const signature = await connection.requestAirdrop(
from.publicKey,
minimumAmount * 2,
);
mockConfirmTransaction(signature);
await connection.confirmTransaction(signature, 'singleGossip');
mockRpc.push([
url,
{
method: 'getBalance',
params: [from.publicKey.toBase58(), {commitment: 'singleGossip'}],
},
{
error: null,
result: {
context: {
slot: 11,
},
value: minimumAmount * 2,
},
},
]);
const balance = await connection.getBalance(from.publicKey);
expect(balance).toBe(minimumAmount * 2);
mockGetRecentBlockhash('max');
mockRpc.push([
url,
{
method: 'sendTransaction',
},
{
error: null,
result:
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
await helpers.airdrop({
connection,
address: from.publicKey,
amount: minimumAmount * 2,
});
const transaction = new Transaction().add(
SystemProgram.createNonceAccount({
@ -113,51 +73,42 @@ test('create and query nonce account', async () => {
lamports: minimumAmount,
}),
);
const nonceSignature = await connection.sendTransaction(
transaction,
[from, nonceAccount],
{
skipPreflight: true,
},
);
mockConfirmTransaction(nonceSignature);
await connection.confirmTransaction(nonceSignature, 'singleGossip');
mockRpc.push([
url,
{
await helpers.processTransaction({
connection,
transaction,
signers: [from, nonceAccount],
commitment: 'singleGossip',
});
await mockRpcResponse({
method: 'getAccountInfo',
params: [
nonceAccount.publicKey.toBase58(),
{encoding: 'base64', commitment: 'singleGossip'},
],
},
{
error: null,
result: {
context: {
slot: 11,
},
value: {
owner: '11111111111111111111111111111111',
lamports: minimumAmount,
data: expectedData(from.publicKey),
executable: false,
},
},
},
]);
//
const nonceAccountData = await connection.getNonce(nonceAccount.publicKey);
withContext: true,
});
const nonceAccountData = await connection.getNonce(
nonceAccount.publicKey,
'singleGossip',
);
if (nonceAccountData === null) {
expect(nonceAccountData).not.toBeNull();
expect(nonceAccountData).not.to.be.null;
return;
}
expect(nonceAccountData.authorizedPubkey).toEqual(from.publicKey);
expect(bs58.decode(nonceAccountData.nonce).length).toBeGreaterThan(30);
});
expect(nonceAccountData.authorizedPubkey).to.eql(from.publicKey);
expect(bs58.decode(nonceAccountData.nonce).length).to.be.greaterThan(30);
});
test('create and query nonce account with seed', async () => {
it('create and query nonce account with seed', async () => {
const from = new Account();
const seed = 'seed';
const noncePubkey = await PublicKey.createWithSeed(
@ -165,76 +116,22 @@ test('create and query nonce account with seed', async () => {
seed,
SystemProgram.programId,
);
const connection = new Connection(url, 'singleGossip');
mockRpc.push([
url,
{
await mockRpcResponse({
method: 'getMinimumBalanceForRentExemption',
params: [NONCE_ACCOUNT_LENGTH, {commitment: 'singleGossip'}],
},
{
error: null,
result: 50,
},
]);
params: [NONCE_ACCOUNT_LENGTH],
value: 50,
});
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
NONCE_ACCOUNT_LENGTH,
);
mockRpc.push([
url,
{
method: 'requestAirdrop',
params: [from.publicKey.toBase58(), minimumAmount * 2],
},
{
error: null,
result:
'1WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
const signature = await connection.requestAirdrop(
from.publicKey,
minimumAmount * 2,
);
mockConfirmTransaction(signature);
await connection.confirmTransaction(signature, 'singleGossip');
mockRpc.push([
url,
{
method: 'getBalance',
params: [from.publicKey.toBase58(), {commitment: 'singleGossip'}],
},
{
error: null,
result: {
context: {
slot: 11,
},
value: minimumAmount * 2,
},
},
]);
const balance = await connection.getBalance(from.publicKey);
expect(balance).toBe(minimumAmount * 2);
mockGetRecentBlockhash('max');
mockRpc.push([
url,
{
method: 'sendTransaction',
},
{
error: null,
result:
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
await helpers.airdrop({
connection,
address: from.publicKey,
amount: minimumAmount * 2,
});
const transaction = new Transaction().add(
SystemProgram.createNonceAccount({
@ -246,42 +143,38 @@ test('create and query nonce account with seed', async () => {
lamports: minimumAmount,
}),
);
const nonceSignature = await connection.sendTransaction(transaction, [from], {
skipPreflight: true,
});
mockConfirmTransaction(nonceSignature);
await connection.confirmTransaction(nonceSignature, 'singleGossip');
mockRpc.push([
url,
{
await helpers.processTransaction({
connection,
transaction,
signers: [from],
commitment: 'singleGossip',
});
await mockRpcResponse({
method: 'getAccountInfo',
params: [
noncePubkey.toBase58(),
{encoding: 'base64', commitment: 'singleGossip'},
],
},
{
error: null,
result: {
context: {
slot: 11,
},
value: {
owner: '11111111111111111111111111111111',
lamports: minimumAmount,
data: expectedData(from.publicKey),
executable: false,
},
},
},
]);
//
const nonceAccountData = await connection.getNonce(noncePubkey);
withContext: true,
});
const nonceAccountData = await connection.getNonce(
noncePubkey,
'singleGossip',
);
if (nonceAccountData === null) {
expect(nonceAccountData).not.toBeNull();
expect(nonceAccountData).not.to.be.null;
return;
}
expect(nonceAccountData.authorizedPubkey).toEqual(from.publicKey);
expect(bs58.decode(nonceAccountData.nonce).length).toBeGreaterThan(30);
expect(nonceAccountData.authorizedPubkey).to.eql(from.publicKey);
expect(bs58.decode(nonceAccountData.nonce).length).to.be.greaterThan(30);
});
});

View File

@ -1,9 +1,14 @@
// @flow
import BN from 'bn.js';
import {Buffer} from 'buffer';
import {PublicKey, MAX_SEED_LENGTH} from '../src/publickey';
import {expect, use} from 'chai';
import chaiAsPromised from 'chai-as-promised';
test('invalid', () => {
use(chaiAsPromised);
describe('PublicKey', function () {
it('invalid', () => {
expect(() => {
new PublicKey([
3,
@ -40,32 +45,32 @@ test('invalid', () => {
0,
0,
]);
}).toThrow();
}).to.throw();
expect(() => {
new PublicKey(
'0x300000000000000000000000000000000000000000000000000000000000000000000',
);
}).toThrow();
}).to.throw();
expect(() => {
new PublicKey(
'0x300000000000000000000000000000000000000000000000000000000000000',
);
}).toThrow();
}).to.throw();
expect(() => {
new PublicKey(
'135693854574979916511997248057056142015550763280047535983739356259273198796800000',
);
}).toThrow();
}).to.throw();
expect(() => {
new PublicKey('12345');
}).toThrow();
});
}).to.throw();
});
test('equals', () => {
it('equals', () => {
const arrayKey = new PublicKey([
3,
0,
@ -104,20 +109,20 @@ test('equals', () => {
'CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3',
);
expect(arrayKey.equals(base58Key)).toBe(true);
});
expect(arrayKey.equals(base58Key)).to.be.true;
});
test('toBase58', () => {
it('toBase58', () => {
const key = new PublicKey('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
expect(key.toBase58()).toBe('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
expect(key.toString()).toBe('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
expect(key.toBase58()).to.eq('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
expect(key.toString()).to.eq('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
const key2 = new PublicKey('1111111111111111111111111111BukQL');
expect(key2.toBase58()).toBe('1111111111111111111111111111BukQL');
expect(key2.toString()).toBe('1111111111111111111111111111BukQL');
expect(key2.toBase58()).to.eq('1111111111111111111111111111BukQL');
expect(key2.toString()).to.eq('1111111111111111111111111111BukQL');
const key3 = new PublicKey('11111111111111111111111111111111');
expect(key3.toBase58()).toBe('11111111111111111111111111111111');
expect(key3.toBase58()).to.eq('11111111111111111111111111111111');
const key4 = new PublicKey([
0,
@ -153,24 +158,24 @@ test('toBase58', () => {
0,
0,
]);
expect(key4.toBase58()).toBe('11111111111111111111111111111111');
});
expect(key4.toBase58()).to.eq('11111111111111111111111111111111');
});
test('toBuffer', () => {
it('toBuffer', () => {
const key = new PublicKey('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
expect(key.toBuffer()).toHaveLength(32);
expect(key.toBase58()).toBe('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
expect(key.toBuffer()).to.have.length(32);
expect(key.toBase58()).to.eq('CiDwVBFgWV9E5MvXWoLgnEgn2hK7rJikbvfWavzAQz3');
const key2 = new PublicKey('11111111111111111111111111111111');
expect(key2.toBuffer()).toHaveLength(32);
expect(key2.toBase58()).toBe('11111111111111111111111111111111');
expect(key2.toBuffer()).to.have.length(32);
expect(key2.toBase58()).to.eq('11111111111111111111111111111111');
const key3 = new PublicKey(0);
expect(key3.toBuffer()).toHaveLength(32);
expect(key3.toBase58()).toBe('11111111111111111111111111111111');
});
expect(key3.toBuffer()).to.have.length(32);
expect(key3.toBase58()).to.eq('11111111111111111111111111111111');
});
test('equals (II)', () => {
it('equals (II)', () => {
const key1 = new PublicKey([
0,
0,
@ -207,10 +212,10 @@ test('equals (II)', () => {
]);
const key2 = new PublicKey(key1.toBuffer());
expect(key1.equals(key2)).toBe(true);
});
expect(key1.equals(key2)).to.be.true;
});
test('createWithSeed', async () => {
it('createWithSeed', async () => {
const defaultPublicKey = new PublicKey('11111111111111111111111111111111');
const derivedKey = await PublicKey.createWithSeed(
defaultPublicKey,
@ -222,10 +227,10 @@ test('createWithSeed', async () => {
derivedKey.equals(
new PublicKey('9h1HyLCW5dZnBVap8C5egQ9Z6pHyjsh5MNy83iPqqRuq'),
),
).toBe(true);
});
).to.be.true;
});
test('createProgramAddress', async () => {
it('createProgramAddress', async () => {
const programId = new PublicKey(
'BPFLoader1111111111111111111111111111111111',
);
@ -241,7 +246,7 @@ test('createProgramAddress', async () => {
programAddress.equals(
new PublicKey('3gF2KMe9KiC6FNVBmfg9i267aMPvK37FewCip4eGBFcT'),
),
).toBe(true);
).to.be.true;
programAddress = await PublicKey.createProgramAddress(
[Buffer.from('☉', 'utf8')],
@ -251,7 +256,7 @@ test('createProgramAddress', async () => {
programAddress.equals(
new PublicKey('7ytmC1nT1xY4RfxCV2ZgyA7UakC93do5ZdyhdF3EtPj7'),
),
).toBe(true);
).to.be.true;
programAddress = await PublicKey.createProgramAddress(
[Buffer.from('Talking', 'utf8'), Buffer.from('Squirrels', 'utf8')],
@ -261,7 +266,7 @@ test('createProgramAddress', async () => {
programAddress.equals(
new PublicKey('HwRVBufQ4haG5XSgpspwKtNd3PC9GM9m1196uJW36vds'),
),
).toBe(true);
).to.be.true;
programAddress = await PublicKey.createProgramAddress(
[publicKey.toBuffer()],
@ -271,25 +276,27 @@ test('createProgramAddress', async () => {
programAddress.equals(
new PublicKey('GUs5qLUfsEHkcMB9T38vjr18ypEhRuNWiePW2LoK4E3K'),
),
).toBe(true);
).to.be.true;
const programAddress2 = await PublicKey.createProgramAddress(
[Buffer.from('Talking', 'utf8')],
programId,
);
expect(programAddress.equals(programAddress2)).toBe(false);
expect(programAddress.equals(programAddress2)).to.eq(false);
await expect(
PublicKey.createProgramAddress(
[Buffer.alloc(MAX_SEED_LENGTH + 1)],
programId,
),
).rejects.toThrow('Max seed length exceeded');
).to.be.rejectedWith('Max seed length exceeded');
// https://github.com/solana-labs/solana/issues/11950
{
let seeds = [
new PublicKey('H4snTKK9adiU15gP22ErfZYtro3aqR9BTMXiH3AwiUTQ').toBuffer(),
new PublicKey(
'H4snTKK9adiU15gP22ErfZYtro3aqR9BTMXiH3AwiUTQ',
).toBuffer(),
new BN(2).toArrayLike(Buffer, 'le', 8),
];
let programId = new PublicKey(
@ -300,11 +307,11 @@ test('createProgramAddress', async () => {
programAddress.equals(
new PublicKey('12rqwuEgBYiGhBrDJStCiqEtzQpTTiZbh7teNVLuYcFA'),
),
).toBe(true);
).to.be.true;
}
});
});
test('findProgramAddress', async () => {
it('findProgramAddress', async () => {
const programId = new PublicKey(
'BPFLoader1111111111111111111111111111111111',
);
@ -319,5 +326,6 @@ test('findProgramAddress', async () => {
programId,
),
),
).toBe(true);
).to.be.true;
});
});

View File

@ -0,0 +1,76 @@
import alias from '@rollup/plugin-alias';
import babel from '@rollup/plugin-babel';
import commonjs from '@rollup/plugin-commonjs';
import flowRemoveTypes from 'flow-remove-types';
import json from '@rollup/plugin-json';
import multi from '@rollup/plugin-multi-entry';
import nodeResolve from '@rollup/plugin-node-resolve';
import nodePolyfills from 'rollup-plugin-node-polyfills';
import replace from '@rollup/plugin-replace';
export default {
input: {
// include: [
// 'test/account.test.js',
// 'test/cluster.test.js',
// 'test/stake-program.test.js',
// ],
include: ['test/**/*.test.js'],
exclude: ['test/agent-manager.test.js', 'test/bpf-loader.test.js'],
},
external: ['node-forge', 'http2', '_stream_wrap'],
output: {
file: 'test/dist/bundle.js',
format: 'es',
sourcemap: true,
},
plugins: [
flow(),
multi(),
commonjs(),
nodeResolve({
browser: true,
preferBuiltins: false,
dedupe: ['bn.js', 'buffer'],
}),
babel({
exclude: '**/node_modules/**',
babelHelpers: 'runtime',
plugins: ['@babel/plugin-transform-runtime'],
}),
nodePolyfills(),
replace({
'process.env.BROWSER': 'true',
'process.env.TEST_LIVE': 'true',
}),
alias({
entries: [
{
find: /^\.\.\/src\/.*\.js$/,
replacement: './lib/index.browser.esm.js',
},
],
}),
json(),
],
onwarn: function (warning, rollupWarn) {
if (warning.code !== 'CIRCULAR_DEPENDENCY' && warning.code !== 'EVAL') {
rollupWarn(warning);
}
},
treeshake: {
moduleSideEffects: path => path.endsWith('test.js'),
},
};
// Using this instead of rollup-plugin-flow due to
// https://github.com/leebyron/rollup-plugin-flow/issues/5
function flow() {
return {
name: 'flow-remove-types',
transform: code => ({
code: flowRemoveTypes(code).toString(),
map: null,
}),
};
}

View File

@ -1,88 +1,82 @@
// @flow
import {keccak_256} from 'js-sha3';
import secp256k1 from 'secp256k1';
import {randomBytes} from 'crypto';
import {Buffer} from 'buffer';
import createKeccakHash from 'keccak';
import {privateKeyVerify, ecdsaSign, publicKeyCreate} from 'secp256k1';
import {Secp256k1Program} from '../src/secp256k1-program';
import {mockRpcEnabled} from './__mocks__/node-fetch';
import {url} from './url';
import {
Connection,
Account,
sendAndConfirmTransaction,
LAMPORTS_PER_SOL,
Transaction,
Secp256k1Program,
} from '../src';
import {url} from './url';
import {helpers} from './mocks/rpc-http';
const {privateKeyVerify, ecdsaSign, publicKeyCreate} = secp256k1;
if (!mockRpcEnabled) {
jest.setTimeout(20000);
}
test('live create secp256k1 instruction with public key', async () => {
if (mockRpcEnabled) {
console.log('non-live test skipped');
return;
}
const message = Buffer.from('This is a message');
const randomPrivateKey = () => {
let privateKey;
do {
privateKey = randomBytes(32);
privateKey = new Account().secretKey.slice(0, 32);
} while (!privateKeyVerify(privateKey));
return privateKey;
};
if (process.env.TEST_LIVE) {
describe('secp256k1', () => {
it('create secp256k1 instruction with public key', async () => {
const privateKey = randomPrivateKey();
const publicKey = publicKeyCreate(privateKey, false);
const messageHash = Buffer.from(keccak_256.update(message).digest());
const message = Buffer.from('This is a message');
const messageHash = createKeccakHash('keccak256')
.update(message)
.digest();
const {signature, recid: recoveryId} = ecdsaSign(messageHash, privateKey);
const connection = new Connection(url, 'singleGossip');
const instruction = Secp256k1Program.createInstructionWithPublicKey({
const from = new Account();
await connection.confirmTransaction(
await connection.requestAirdrop(from.publicKey, 2 * LAMPORTS_PER_SOL),
'singleGossip',
);
const transaction = new Transaction().add(
Secp256k1Program.createInstructionWithPublicKey({
publicKey,
message,
signature,
recoveryId,
});
const transaction = new Transaction();
transaction.add(instruction);
const connection = new Connection(url, 'recent');
const from = new Account();
await connection.requestAirdrop(from.publicKey, 2 * LAMPORTS_PER_SOL);
}),
);
await sendAndConfirmTransaction(connection, transaction, [from], {
commitment: 'single',
skipPreflight: true,
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
});
});
test('live create secp256k1 instruction with private key', async () => {
if (mockRpcEnabled) {
console.log('non-live test skipped');
return;
}
it('create secp256k1 instruction with private key', async () => {
const privateKey = randomPrivateKey();
const connection = new Connection(url, 'singleGossip');
let privateKey;
do {
privateKey = randomBytes(32);
} while (!privateKeyVerify(privateKey));
const from = new Account();
await connection.confirmTransaction(
await connection.requestAirdrop(from.publicKey, 2 * LAMPORTS_PER_SOL),
'singleGossip',
);
const instruction = Secp256k1Program.createInstructionWithPrivateKey({
const transaction = new Transaction().add(
Secp256k1Program.createInstructionWithPrivateKey({
privateKey,
message: Buffer.from('Test 123'),
});
const transaction = new Transaction();
transaction.add(instruction);
const connection = new Connection(url, 'recent');
const from = new Account();
await connection.requestAirdrop(from.publicKey, 2 * LAMPORTS_PER_SOL);
}),
);
await sendAndConfirmTransaction(connection, transaction, [from], {
commitment: 'single',
skipPreflight: true,
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
});
});
});
}

View File

@ -1,13 +1,27 @@
// @flow
import {expect} from 'chai';
import {decodeLength, encodeLength} from '../src/util/shortvec-encoding';
function checkDecodedArray(array: Array<number>, expectedValue: number) {
expect(decodeLength(array)).toEqual(expectedValue);
expect(array.length).toEqual(0);
expect(decodeLength(array)).to.eq(expectedValue);
expect(array).to.have.length(0);
}
test('shortvec decodeLength', () => {
function checkEncodedArray(
array: Array<number>,
len: number,
prevLength: number,
addedLength: number,
expectedArray: Array<number>,
) {
encodeLength(array, len);
expect(array).to.have.length(prevLength);
expect(array.slice(-addedLength)).to.eql(expectedArray);
}
describe('shortvec', () => {
it('decodeLength', () => {
let array = [];
checkDecodedArray(array, 0);
@ -31,21 +45,9 @@ test('shortvec decodeLength', () => {
array = [0x80, 0x80, 0x80, 0x01];
checkDecodedArray(array, 0x200000);
});
});
function checkEncodedArray(
array: Array<number>,
len: number,
prevLength: number,
addedLength: number,
expectedArray: Array<number>,
) {
encodeLength(array, len);
expect(array.length).toEqual(prevLength);
expect(array.slice(-addedLength)).toEqual(expectedArray);
}
test('shortvec encodeLength', () => {
it('encodeLength', () => {
let array = [];
let prevLength = 1;
checkEncodedArray(array, 0, prevLength, 1, [0]);
@ -68,4 +70,5 @@ test('shortvec encodeLength', () => {
0x80,
0x01,
]);
});
});

View File

@ -1,5 +1,8 @@
// @flow
import {expect, use} from 'chai';
import chaiAsPromised from 'chai-as-promised';
import {
Account,
Authorized,
@ -14,16 +17,13 @@ import {
SystemInstruction,
Transaction,
} from '../src';
import {mockRpcEnabled} from './__mocks__/node-fetch';
import {newAccountWithLamports} from './new-account-with-lamports';
import {helpers} from './mocks/rpc-http';
import {url} from './url';
if (!mockRpcEnabled) {
// Testing max commitment level takes around 20s to complete
jest.setTimeout(60000);
}
use(chaiAsPromised);
test('createAccountWithSeed', async () => {
describe('StakeProgram', () => {
it('createAccountWithSeed', async () => {
const fromPubkey = new Account().publicKey;
const seed = 'test string';
const newAccountPubkey = await PublicKey.createWithSeed(
@ -44,7 +44,7 @@ test('createAccountWithSeed', async () => {
lockup,
lamports,
});
expect(transaction.instructions).toHaveLength(2);
expect(transaction.instructions).to.have.length(2);
const [systemInstruction, stakeInstruction] = transaction.instructions;
const systemParams = {
fromPubkey,
@ -55,16 +55,16 @@ test('createAccountWithSeed', async () => {
space: StakeProgram.space,
programId: StakeProgram.programId,
};
expect(systemParams).toEqual(
expect(systemParams).to.eql(
SystemInstruction.decodeCreateWithSeed(systemInstruction),
);
const initParams = {stakePubkey: newAccountPubkey, authorized, lockup};
expect(initParams).toEqual(
expect(initParams).to.eql(
StakeInstruction.decodeInitialize(stakeInstruction),
);
});
});
test('createAccount', () => {
it('createAccount', () => {
const fromPubkey = new Account().publicKey;
const newAccountPubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
@ -78,7 +78,7 @@ test('createAccount', () => {
lockup,
lamports,
});
expect(transaction.instructions).toHaveLength(2);
expect(transaction.instructions).to.have.length(2);
const [systemInstruction, stakeInstruction] = transaction.instructions;
const systemParams = {
fromPubkey,
@ -87,17 +87,17 @@ test('createAccount', () => {
space: StakeProgram.space,
programId: StakeProgram.programId,
};
expect(systemParams).toEqual(
expect(systemParams).to.eql(
SystemInstruction.decodeCreateAccount(systemInstruction),
);
const initParams = {stakePubkey: newAccountPubkey, authorized, lockup};
expect(initParams).toEqual(
expect(initParams).to.eql(
StakeInstruction.decodeInitialize(stakeInstruction),
);
});
});
test('delegate', () => {
it('delegate', () => {
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
const votePubkey = new Account().publicKey;
@ -107,12 +107,12 @@ test('delegate', () => {
votePubkey,
};
const transaction = StakeProgram.delegate(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(StakeInstruction.decodeDelegate(stakeInstruction));
});
expect(params).to.eql(StakeInstruction.decodeDelegate(stakeInstruction));
});
test('authorize', () => {
it('authorize', () => {
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
const newAuthorizedPubkey = new Account().publicKey;
@ -124,12 +124,12 @@ test('authorize', () => {
stakeAuthorizationType,
};
const transaction = StakeProgram.authorize(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(StakeInstruction.decodeAuthorize(stakeInstruction));
});
expect(params).to.eql(StakeInstruction.decodeAuthorize(stakeInstruction));
});
test('authorize with custodian', () => {
it('authorize with custodian', () => {
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
const newAuthorizedPubkey = new Account().publicKey;
@ -143,12 +143,12 @@ test('authorize with custodian', () => {
custodianPubkey,
};
const transaction = StakeProgram.authorize(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(StakeInstruction.decodeAuthorize(stakeInstruction));
});
expect(params).to.eql(StakeInstruction.decodeAuthorize(stakeInstruction));
});
test('authorizeWithSeed', () => {
it('authorizeWithSeed', () => {
const stakePubkey = new Account().publicKey;
const authorityBase = new Account().publicKey;
const authoritySeed = 'test string';
@ -164,14 +164,14 @@ test('authorizeWithSeed', () => {
stakeAuthorizationType,
};
const transaction = StakeProgram.authorizeWithSeed(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(
expect(params).to.eql(
StakeInstruction.decodeAuthorizeWithSeed(stakeInstruction),
);
});
});
test('authorizeWithSeed with custodian', () => {
it('authorizeWithSeed with custodian', () => {
const stakePubkey = new Account().publicKey;
const authorityBase = new Account().publicKey;
const authoritySeed = 'test string';
@ -189,14 +189,14 @@ test('authorizeWithSeed with custodian', () => {
custodianPubkey,
};
const transaction = StakeProgram.authorizeWithSeed(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(
expect(params).to.eql(
StakeInstruction.decodeAuthorizeWithSeed(stakeInstruction),
);
});
});
test('split', () => {
it('split', () => {
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
const splitStakePubkey = new Account().publicKey;
@ -207,7 +207,7 @@ test('split', () => {
lamports: 123,
};
const transaction = StakeProgram.split(params);
expect(transaction.instructions).toHaveLength(2);
expect(transaction.instructions).to.have.length(2);
const [systemInstruction, stakeInstruction] = transaction.instructions;
const systemParams = {
fromPubkey: authorizedPubkey,
@ -216,13 +216,13 @@ test('split', () => {
space: StakeProgram.space,
programId: StakeProgram.programId,
};
expect(systemParams).toEqual(
expect(systemParams).to.eql(
SystemInstruction.decodeCreateAccount(systemInstruction),
);
expect(params).toEqual(StakeInstruction.decodeSplit(stakeInstruction));
});
expect(params).to.eql(StakeInstruction.decodeSplit(stakeInstruction));
});
test('withdraw', () => {
it('withdraw', () => {
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
const toPubkey = new Account().publicKey;
@ -233,12 +233,12 @@ test('withdraw', () => {
lamports: 123,
};
const transaction = StakeProgram.withdraw(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(StakeInstruction.decodeWithdraw(stakeInstruction));
});
expect(params).to.eql(StakeInstruction.decodeWithdraw(stakeInstruction));
});
test('withdraw with custodian', () => {
it('withdraw with custodian', () => {
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
const toPubkey = new Account().publicKey;
@ -251,22 +251,22 @@ test('withdraw with custodian', () => {
custodianPubkey,
};
const transaction = StakeProgram.withdraw(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(StakeInstruction.decodeWithdraw(stakeInstruction));
});
expect(params).to.eql(StakeInstruction.decodeWithdraw(stakeInstruction));
});
test('deactivate', () => {
it('deactivate', () => {
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
const params = {stakePubkey, authorizedPubkey};
const transaction = StakeProgram.deactivate(params);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [stakeInstruction] = transaction.instructions;
expect(params).toEqual(StakeInstruction.decodeDeactivate(stakeInstruction));
});
expect(params).to.eql(StakeInstruction.decodeDeactivate(stakeInstruction));
});
test('StakeInstructions', async () => {
it('StakeInstructions', async () => {
const from = new Account();
const seed = 'test string';
const newAccountPubkey = await PublicKey.createWithSeed(
@ -290,22 +290,22 @@ test('StakeInstructions', async () => {
createWithSeed,
);
expect(createWithSeedTransaction.instructions).toHaveLength(2);
expect(createWithSeedTransaction.instructions).to.have.length(2);
const systemInstructionType = SystemInstruction.decodeInstructionType(
createWithSeedTransaction.instructions[0],
);
expect(systemInstructionType).toEqual('CreateWithSeed');
expect(systemInstructionType).to.eq('CreateWithSeed');
const stakeInstructionType = StakeInstruction.decodeInstructionType(
createWithSeedTransaction.instructions[1],
);
expect(stakeInstructionType).toEqual('Initialize');
expect(stakeInstructionType).to.eq('Initialize');
expect(() => {
StakeInstruction.decodeInstructionType(
createWithSeedTransaction.instructions[0],
);
}).toThrow();
}).to.throw();
const stake = new Account();
const vote = new Account();
@ -315,38 +315,47 @@ test('StakeInstructions', async () => {
votePubkey: vote.publicKey,
});
const delegateTransaction = new Transaction({recentBlockhash}).add(delegate);
const delegateTransaction = new Transaction({recentBlockhash}).add(
delegate,
);
const anotherStakeInstructionType = StakeInstruction.decodeInstructionType(
delegateTransaction.instructions[0],
);
expect(anotherStakeInstructionType).toEqual('Delegate');
});
test('live staking actions', async () => {
if (mockRpcEnabled) {
console.log('non-live test skipped');
return;
}
expect(anotherStakeInstructionType).to.eq('Delegate');
});
if (process.env.TEST_LIVE) {
it('live staking actions', async () => {
const connection = new Connection(url, 'singleGossip');
const voteAccounts = await connection.getVoteAccounts();
const voteAccount = voteAccounts.current.concat(voteAccounts.delinquent)[0];
const voteAccount = voteAccounts.current.concat(
voteAccounts.delinquent,
)[0];
const votePubkey = new PublicKey(voteAccount.votePubkey);
const from = await newAccountWithLamports(connection, 2 * LAMPORTS_PER_SOL);
const authorized = await newAccountWithLamports(
const payer = new Account();
await helpers.airdrop({
connection,
2 * LAMPORTS_PER_SOL,
);
address: payer.publicKey,
amount: 2 * LAMPORTS_PER_SOL,
});
const authorized = new Account();
await helpers.airdrop({
connection,
address: authorized.publicKey,
amount: 2 * LAMPORTS_PER_SOL,
});
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
StakeProgram.space,
);
expect(await connection.getBalance(from.publicKey)).toEqual(
expect(await connection.getBalance(payer.publicKey)).to.eq(
2 * LAMPORTS_PER_SOL,
);
expect(await connection.getBalance(authorized.publicKey)).toEqual(
expect(await connection.getBalance(authorized.publicKey)).to.eq(
2 * LAMPORTS_PER_SOL,
);
@ -354,9 +363,12 @@ test('live staking actions', async () => {
// Create Stake account without seed
const newStakeAccount = new Account();
let createAndInitialize = StakeProgram.createAccount({
fromPubkey: from.publicKey,
fromPubkey: payer.publicKey,
stakePubkey: newStakeAccount.publicKey,
authorized: new Authorized(authorized.publicKey, authorized.publicKey),
authorized: new Authorized(
authorized.publicKey,
authorized.publicKey,
),
lockup: new Lockup(0, 0, new PublicKey(0)),
lamports: minimumAmount + 42,
});
@ -364,14 +376,14 @@ test('live staking actions', async () => {
await sendAndConfirmTransaction(
connection,
createAndInitialize,
[from, newStakeAccount],
{commitment: 'singleGossip'},
[payer, newStakeAccount],
{preflightCommitment: 'singleGossip'},
);
expect(await connection.getBalance(newStakeAccount.publicKey)).toEqual(
expect(await connection.getBalance(newStakeAccount.publicKey)).to.eq(
minimumAmount + 42,
);
let delegation = StakeProgram.delegate({
const delegation = StakeProgram.delegate({
stakePubkey: newStakeAccount.publicKey,
authorizedPubkey: authorized.publicKey,
votePubkey,
@ -384,15 +396,15 @@ test('live staking actions', async () => {
// Create Stake account with seed
const seed = 'test string';
const newAccountPubkey = await PublicKey.createWithSeed(
from.publicKey,
payer.publicKey,
seed,
StakeProgram.programId,
);
let createAndInitializeWithSeed = StakeProgram.createAccountWithSeed({
fromPubkey: from.publicKey,
fromPubkey: payer.publicKey,
stakePubkey: newAccountPubkey,
basePubkey: from.publicKey,
basePubkey: payer.publicKey,
seed,
authorized: new Authorized(authorized.publicKey, authorized.publicKey),
lockup: new Lockup(0, 0, new PublicKey(0)),
@ -402,11 +414,11 @@ test('live staking actions', async () => {
await sendAndConfirmTransaction(
connection,
createAndInitializeWithSeed,
[from],
{commitment: 'singleGossip'},
[payer],
{preflightCommitment: 'singleGossip'},
);
let originalStakeBalance = await connection.getBalance(newAccountPubkey);
expect(originalStakeBalance).toEqual(3 * minimumAmount + 42);
expect(originalStakeBalance).to.eq(3 * minimumAmount + 42);
let delegation = StakeProgram.delegate({
stakePubkey: newAccountPubkey,
@ -414,7 +426,7 @@ test('live staking actions', async () => {
votePubkey,
});
await sendAndConfirmTransaction(connection, delegation, [authorized], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
// Test that withdraw fails before deactivation
@ -427,9 +439,9 @@ test('live staking actions', async () => {
});
await expect(
sendAndConfirmTransaction(connection, withdraw, [authorized], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
}),
).rejects.toThrow();
).to.be.rejected;
// Deactivate stake
let deactivate = StakeProgram.deactivate({
@ -437,7 +449,7 @@ test('live staking actions', async () => {
authorizedPubkey: authorized.publicKey,
});
await sendAndConfirmTransaction(connection, deactivate, [authorized], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
let stakeActivationState;
@ -456,10 +468,10 @@ test('live staking actions', async () => {
});
await sendAndConfirmTransaction(connection, withdraw, [authorized], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
const recipientBalance = await connection.getBalance(recipient.publicKey);
expect(recipientBalance).toEqual(minimumAmount + 20);
expect(recipientBalance).to.eq(minimumAmount + 20);
// Split stake
const newStake = new Account();
@ -469,15 +481,23 @@ test('live staking actions', async () => {
splitStakePubkey: newStake.publicKey,
lamports: minimumAmount + 20,
});
await sendAndConfirmTransaction(connection, split, [authorized, newStake], {
commitment: 'singleGossip',
});
await sendAndConfirmTransaction(
connection,
split,
[authorized, newStake],
{
preflightCommitment: 'singleGossip',
},
);
const balance = await connection.getBalance(newAccountPubkey);
expect(balance).toEqual(minimumAmount + 2);
expect(balance).to.eq(minimumAmount + 2);
// Authorize to new account
const newAuthorized = new Account();
await connection.requestAirdrop(newAuthorized.publicKey, LAMPORTS_PER_SOL);
await connection.requestAirdrop(
newAuthorized.publicKey,
LAMPORTS_PER_SOL,
);
let authorize = StakeProgram.authorize({
stakePubkey: newAccountPubkey,
@ -486,7 +506,7 @@ test('live staking actions', async () => {
stakeAuthorizationType: StakeAuthorizationLayout.Withdrawer,
});
await sendAndConfirmTransaction(connection, authorize, [authorized], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
authorize = StakeProgram.authorize({
stakePubkey: newAccountPubkey,
@ -495,7 +515,7 @@ test('live staking actions', async () => {
stakeAuthorizationType: StakeAuthorizationLayout.Staker,
});
await sendAndConfirmTransaction(connection, authorize, [authorized], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
// Test old authorized can't delegate
@ -505,10 +525,15 @@ test('live staking actions', async () => {
votePubkey,
});
await expect(
sendAndConfirmTransaction(connection, delegateNotAuthorized, [authorized], {
commitment: 'singleGossip',
}),
).rejects.toThrow();
sendAndConfirmTransaction(
connection,
delegateNotAuthorized,
[authorized],
{
preflightCommitment: 'singleGossip',
},
),
).to.be.rejected;
// Authorize a derived address
authorize = StakeProgram.authorize({
@ -518,19 +543,21 @@ test('live staking actions', async () => {
stakeAuthorizationType: StakeAuthorizationLayout.Withdrawer,
});
await sendAndConfirmTransaction(connection, authorize, [newAuthorized], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
// Restore the previous authority using a derived address
authorize = StakeProgram.authorizeWithSeed({
stakePubkey: newAccountPubkey,
authorityBase: from.publicKey,
authorityBase: payer.publicKey,
authoritySeed: seed,
authorityOwner: StakeProgram.programId,
newAuthorizedPubkey: newAuthorized.publicKey,
stakeAuthorizationType: StakeAuthorizationLayout.Withdrawer,
});
await sendAndConfirmTransaction(connection, authorize, [from], {
commitment: 'singleGossip',
await sendAndConfirmTransaction(connection, authorize, [payer], {
preflightCommitment: 'singleGossip',
});
}).timeout(10 * 1000);
}
});

View File

@ -1,5 +1,8 @@
// @flow
import {Buffer} from 'buffer';
import {expect} from 'chai';
import {
Account,
Connection,
@ -13,17 +16,12 @@ import {
LAMPORTS_PER_SOL,
} from '../src';
import {NONCE_ACCOUNT_LENGTH} from '../src/nonce-account';
import {mockRpcEnabled} from './__mocks__/node-fetch';
import {newAccountWithLamports} from './new-account-with-lamports';
import {sleep} from '../src/util/sleep';
import {helpers} from './mocks/rpc-http';
import {url} from './url';
if (!mockRpcEnabled) {
// Testing max commitment level takes around 20s to complete
jest.setTimeout(30000);
}
test('createAccount', () => {
describe('SystemProgram', () => {
it('createAccount', () => {
const params = {
fromPubkey: new Account().publicKey,
newAccountPubkey: new Account().publicKey,
@ -34,26 +32,26 @@ test('createAccount', () => {
const transaction = new Transaction().add(
SystemProgram.createAccount(params),
);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(
expect(params).to.eql(
SystemInstruction.decodeCreateAccount(systemInstruction),
);
});
});
test('transfer', () => {
it('transfer', () => {
const params = {
fromPubkey: new Account().publicKey,
toPubkey: new Account().publicKey,
lamports: 123,
};
const transaction = new Transaction().add(SystemProgram.transfer(params));
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(SystemInstruction.decodeTransfer(systemInstruction));
});
expect(params).to.eql(SystemInstruction.decodeTransfer(systemInstruction));
});
test('transferWithSeed', () => {
it('transferWithSeed', () => {
const params = {
fromPubkey: new Account().publicKey,
basePubkey: new Account().publicKey,
@ -63,25 +61,25 @@ test('transferWithSeed', () => {
programId: new Account().publicKey,
};
const transaction = new Transaction().add(SystemProgram.transfer(params));
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(
expect(params).to.eql(
SystemInstruction.decodeTransferWithSeed(systemInstruction),
);
});
});
test('allocate', () => {
it('allocate', () => {
const params = {
accountPubkey: new Account().publicKey,
space: 42,
};
const transaction = new Transaction().add(SystemProgram.allocate(params));
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(SystemInstruction.decodeAllocate(systemInstruction));
});
expect(params).to.eql(SystemInstruction.decodeAllocate(systemInstruction));
});
test('allocateWithSeed', () => {
it('allocateWithSeed', () => {
const params = {
accountPubkey: new Account().publicKey,
basePubkey: new Account().publicKey,
@ -90,25 +88,25 @@ test('allocateWithSeed', () => {
programId: new Account().publicKey,
};
const transaction = new Transaction().add(SystemProgram.allocate(params));
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(
expect(params).to.eql(
SystemInstruction.decodeAllocateWithSeed(systemInstruction),
);
});
});
test('assign', () => {
it('assign', () => {
const params = {
accountPubkey: new Account().publicKey,
programId: new Account().publicKey,
};
const transaction = new Transaction().add(SystemProgram.assign(params));
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(SystemInstruction.decodeAssign(systemInstruction));
});
expect(params).to.eql(SystemInstruction.decodeAssign(systemInstruction));
});
test('assignWithSeed', () => {
it('assignWithSeed', () => {
const params = {
accountPubkey: new Account().publicKey,
basePubkey: new Account().publicKey,
@ -116,14 +114,14 @@ test('assignWithSeed', () => {
programId: new Account().publicKey,
};
const transaction = new Transaction().add(SystemProgram.assign(params));
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(
expect(params).to.eql(
SystemInstruction.decodeAssignWithSeed(systemInstruction),
);
});
});
test('createAccountWithSeed', () => {
it('createAccountWithSeed', () => {
const fromPubkey = new Account().publicKey;
const params = {
fromPubkey,
@ -137,14 +135,14 @@ test('createAccountWithSeed', () => {
const transaction = new Transaction().add(
SystemProgram.createAccountWithSeed(params),
);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [systemInstruction] = transaction.instructions;
expect(params).toEqual(
expect(params).to.eql(
SystemInstruction.decodeCreateWithSeed(systemInstruction),
);
});
});
test('createNonceAccount', () => {
it('createNonceAccount', () => {
const fromPubkey = new Account().publicKey;
const params = {
fromPubkey,
@ -156,7 +154,7 @@ test('createNonceAccount', () => {
const transaction = new Transaction().add(
SystemProgram.createNonceAccount(params),
);
expect(transaction.instructions).toHaveLength(2);
expect(transaction.instructions).to.have.length(2);
const [createInstruction, initInstruction] = transaction.instructions;
const createParams = {
@ -166,7 +164,7 @@ test('createNonceAccount', () => {
space: NONCE_ACCOUNT_LENGTH,
programId: SystemProgram.programId,
};
expect(createParams).toEqual(
expect(createParams).to.eql(
SystemInstruction.decodeCreateAccount(createInstruction),
);
@ -174,12 +172,12 @@ test('createNonceAccount', () => {
noncePubkey: params.noncePubkey,
authorizedPubkey: fromPubkey,
};
expect(initParams).toEqual(
expect(initParams).to.eql(
SystemInstruction.decodeNonceInitialize(initInstruction),
);
});
});
test('createNonceAccount with seed', () => {
it('createNonceAccount with seed', () => {
const fromPubkey = new Account().publicKey;
const params = {
fromPubkey,
@ -193,7 +191,7 @@ test('createNonceAccount with seed', () => {
const transaction = new Transaction().add(
SystemProgram.createNonceAccount(params),
);
expect(transaction.instructions).toHaveLength(2);
expect(transaction.instructions).to.have.length(2);
const [createInstruction, initInstruction] = transaction.instructions;
const createParams = {
@ -205,7 +203,7 @@ test('createNonceAccount with seed', () => {
space: NONCE_ACCOUNT_LENGTH,
programId: SystemProgram.programId,
};
expect(createParams).toEqual(
expect(createParams).to.eql(
SystemInstruction.decodeCreateWithSeed(createInstruction),
);
@ -213,21 +211,21 @@ test('createNonceAccount with seed', () => {
noncePubkey: params.noncePubkey,
authorizedPubkey: fromPubkey,
};
expect(initParams).toEqual(
expect(initParams).to.eql(
SystemInstruction.decodeNonceInitialize(initInstruction),
);
});
});
test('nonceAdvance', () => {
it('nonceAdvance', () => {
const params = {
noncePubkey: new Account().publicKey,
authorizedPubkey: new Account().publicKey,
};
const instruction = SystemProgram.nonceAdvance(params);
expect(params).toEqual(SystemInstruction.decodeNonceAdvance(instruction));
});
expect(params).to.eql(SystemInstruction.decodeNonceAdvance(instruction));
});
test('nonceWithdraw', () => {
it('nonceWithdraw', () => {
const params = {
noncePubkey: new Account().publicKey,
authorizedPubkey: new Account().publicKey,
@ -237,12 +235,12 @@ test('nonceWithdraw', () => {
const transaction = new Transaction().add(
SystemProgram.nonceWithdraw(params),
);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [instruction] = transaction.instructions;
expect(params).toEqual(SystemInstruction.decodeNonceWithdraw(instruction));
});
expect(params).to.eql(SystemInstruction.decodeNonceWithdraw(instruction));
});
test('nonceAuthorize', () => {
it('nonceAuthorize', () => {
const params = {
noncePubkey: new Account().publicKey,
authorizedPubkey: new Account().publicKey,
@ -252,12 +250,12 @@ test('nonceAuthorize', () => {
const transaction = new Transaction().add(
SystemProgram.nonceAuthorize(params),
);
expect(transaction.instructions).toHaveLength(1);
expect(transaction.instructions).to.have.length(1);
const [instruction] = transaction.instructions;
expect(params).toEqual(SystemInstruction.decodeNonceAuthorize(instruction));
});
expect(params).to.eql(SystemInstruction.decodeNonceAuthorize(instruction));
});
test('non-SystemInstruction error', () => {
it('non-SystemInstruction error', () => {
const from = new Account();
const to = new Account();
@ -273,7 +271,7 @@ test('non-SystemInstruction error', () => {
SystemInstruction.decodeInstructionType(
new TransactionInstruction(badProgramId),
);
}).toThrow();
}).to.throw();
const stakePubkey = new Account().publicKey;
const authorizedPubkey = new Account().publicKey;
@ -282,28 +280,32 @@ test('non-SystemInstruction error', () => {
expect(() => {
SystemInstruction.decodeInstructionType(transaction.instructions[1]);
}).toThrow();
}).to.throw();
transaction.instructions[0].data[0] = 11;
expect(() => {
SystemInstruction.decodeInstructionType(transaction.instructions[0]);
}).toThrow();
});
test('live Nonce actions', async () => {
if (mockRpcEnabled) {
console.log('non-live test skipped');
return;
}
}).to.throw();
});
if (process.env.TEST_LIVE) {
it('live Nonce actions', async () => {
const connection = new Connection(url, 'singleGossip');
const nonceAccount = new Account();
const from = await newAccountWithLamports(connection, 2 * LAMPORTS_PER_SOL);
const to = new Account();
const newAuthority = await newAccountWithLamports(
const from = new Account();
await helpers.airdrop({
connection,
LAMPORTS_PER_SOL,
);
address: from.publicKey,
amount: 2 * LAMPORTS_PER_SOL,
});
const to = new Account();
const newAuthority = new Account();
await helpers.airdrop({
connection,
address: newAuthority.publicKey,
amount: LAMPORTS_PER_SOL,
});
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
NONCE_ACCOUNT_LENGTH,
@ -321,24 +323,24 @@ test('live Nonce actions', async () => {
connection,
createNonceAccount,
[from, nonceAccount],
{commitment: 'singleGossip', preflightCommitment: 'singleGossip'},
{preflightCommitment: 'singleGossip'},
);
const nonceBalance = await connection.getBalance(nonceAccount.publicKey);
expect(nonceBalance).toEqual(minimumAmount);
expect(nonceBalance).to.eq(minimumAmount);
const nonceQuery1 = await connection.getNonce(nonceAccount.publicKey);
if (nonceQuery1 === null) {
expect(nonceQuery1).not.toBeNull();
expect(nonceQuery1).not.to.be.null;
return;
}
const nonceQuery2 = await connection.getNonce(nonceAccount.publicKey);
if (nonceQuery2 === null) {
expect(nonceQuery2).not.toBeNull();
expect(nonceQuery2).not.to.be.null;
return;
}
expect(nonceQuery1.nonce).toEqual(nonceQuery2.nonce);
expect(nonceQuery1.nonce).to.eq(nonceQuery2.nonce);
// Wait for blockhash to advance
await sleep(500);
@ -350,15 +352,14 @@ test('live Nonce actions', async () => {
}),
);
await sendAndConfirmTransaction(connection, advanceNonce, [from], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
const nonceQuery3 = await connection.getNonce(nonceAccount.publicKey);
if (nonceQuery3 === null) {
expect(nonceQuery3).not.toBeNull();
expect(nonceQuery3).not.to.be.null;
return;
}
expect(nonceQuery1.nonce).not.toEqual(nonceQuery3.nonce);
expect(nonceQuery1.nonce).not.to.eq(nonceQuery3.nonce);
const nonce = nonceQuery3.nonce;
// Wait for blockhash to advance
@ -372,7 +373,6 @@ test('live Nonce actions', async () => {
}),
);
await sendAndConfirmTransaction(connection, authorizeNonce, [from], {
commitment: 'singleGossip',
preflightCommitment: 'singleGossip',
});
@ -391,12 +391,16 @@ test('live Nonce actions', async () => {
}),
};
await sendAndConfirmTransaction(connection, transfer, [from, newAuthority], {
commitment: 'singleGossip',
await sendAndConfirmTransaction(
connection,
transfer,
[from, newAuthority],
{
preflightCommitment: 'singleGossip',
});
},
);
const toBalance = await connection.getBalance(to.publicKey);
expect(toBalance).toEqual(minimumAmount);
expect(toBalance).to.eq(minimumAmount);
// Wait for blockhash to advance
await sleep(500);
@ -410,28 +414,29 @@ test('live Nonce actions', async () => {
toPubkey: withdrawAccount.publicKey,
}),
);
await sendAndConfirmTransaction(connection, withdrawNonce, [newAuthority], {
commitment: 'singleGossip',
await sendAndConfirmTransaction(
connection,
withdrawNonce,
[newAuthority],
{
preflightCommitment: 'singleGossip',
});
expect(await connection.getBalance(nonceAccount.publicKey)).toEqual(0);
},
);
expect(await connection.getBalance(nonceAccount.publicKey)).to.eq(0);
const withdrawBalance = await connection.getBalance(
withdrawAccount.publicKey,
);
expect(withdrawBalance).toEqual(minimumAmount);
});
test('live withSeed actions', async () => {
if (mockRpcEnabled) {
console.log('non-live test skipped');
return;
}
expect(withdrawBalance).to.eq(minimumAmount);
}).timeout(10 * 1000);
it('live withSeed actions', async () => {
const connection = new Connection(url, 'singleGossip');
const baseAccount = await newAccountWithLamports(
const baseAccount = new Account();
await helpers.airdrop({
connection,
2 * LAMPORTS_PER_SOL,
);
address: baseAccount.publicKey,
amount: 2 * LAMPORTS_PER_SOL,
});
const basePubkey = baseAccount.publicKey;
const seed = 'hi there';
const programId = new Account().publicKey;
@ -463,12 +468,12 @@ test('live withSeed actions', async () => {
connection,
createAccountWithSeedTransaction,
[baseAccount],
{commitment: 'singleGossip', preflightCommitment: 'singleGossip'},
{preflightCommitment: 'singleGossip'},
);
const createAccountWithSeedBalance = await connection.getBalance(
createAccountWithSeedAddress,
);
expect(createAccountWithSeedBalance).toEqual(minimumAmount);
expect(createAccountWithSeedBalance).to.eq(minimumAmount);
// Transfer to a derived address to prep for TransferWithSeed
const programId2 = new Account().publicKey;
@ -487,12 +492,12 @@ test('live withSeed actions', async () => {
}),
),
[baseAccount],
{commitment: 'singleGossip', preflightCommitment: 'singleGossip'},
{preflightCommitment: 'singleGossip'},
);
let transferWithSeedAddressBalance = await connection.getBalance(
transferWithSeedAddress,
);
expect(transferWithSeedAddressBalance).toEqual(3 * minimumAmount);
expect(transferWithSeedAddressBalance).to.eq(3 * minimumAmount);
// Test TransferWithSeed
const programId3 = new Account();
@ -516,14 +521,14 @@ test('live withSeed actions', async () => {
connection,
transferWithSeedTransaction,
[baseAccount],
{commitment: 'singleGossip', preflightCommitment: 'singleGossip'},
{preflightCommitment: 'singleGossip'},
);
const toBalance = await connection.getBalance(toPubkey);
expect(toBalance).toEqual(2 * minimumAmount);
expect(toBalance).to.eq(2 * minimumAmount);
transferWithSeedAddressBalance = await connection.getBalance(
createAccountWithSeedAddress,
);
expect(transferWithSeedAddressBalance).toEqual(minimumAmount);
expect(transferWithSeedAddressBalance).to.eq(minimumAmount);
// Test AllocateWithSeed
const allocateWithSeedParams = {
@ -540,14 +545,14 @@ test('live withSeed actions', async () => {
connection,
allocateWithSeedTransaction,
[baseAccount],
{commitment: 'singleGossip', preflightCommitment: 'singleGossip'},
{preflightCommitment: 'singleGossip'},
);
let account = await connection.getAccountInfo(toPubkey);
if (account === null) {
expect(account).not.toBeNull();
expect(account).not.to.be.null;
return;
}
expect(account.data).toHaveLength(10);
expect(account.data).to.have.length(10);
// Test AssignWithSeed
const assignWithSeedParams = {
@ -563,12 +568,14 @@ test('live withSeed actions', async () => {
connection,
assignWithSeedTransaction,
[baseAccount],
{commitment: 'singleGossip', preflightCommitment: 'singleGossip'},
{preflightCommitment: 'singleGossip'},
);
account = await connection.getAccountInfo(toPubkey);
if (account === null) {
expect(account).not.toBeNull();
expect(account).not.to.be.null;
return;
}
expect(account.owner).toEqual(programId3.publicKey);
expect(account.owner).to.eql(programId3.publicKey);
}).timeout(10 * 1000);
}
});

View File

@ -1,4 +1,7 @@
// @flow
import {expect} from 'chai';
import {
Account,
Connection,
@ -6,108 +9,72 @@ import {
SystemProgram,
LAMPORTS_PER_SOL,
} from '../src';
import {mockRpc, mockRpcEnabled} from './__mocks__/node-fetch';
import {mockGetRecentBlockhash} from './mockrpc/get-recent-blockhash';
import {url} from './url';
import {mockConfirmTransaction} from './mockrpc/confirm-transaction';
import {MOCK_PORT, url} from './url';
if (!mockRpcEnabled) {
// The default of 5 seconds is too slow for live testing sometimes
jest.setTimeout(30000);
}
import {
helpers,
mockErrorMessage,
mockErrorResponse,
uniqueSignature,
uniqueBlockhash,
mockRpcResponse,
mockServer,
} from './mocks/rpc-http';
import {
stubRpcWebSocket,
restoreRpcWebSocket,
mockRpcMessage,
} from './mocks/rpc-websockets';
import base58 from 'bs58';
test('transaction-payer', async () => {
describe('Transaction Payer', () => {
let connection: Connection;
beforeEach(() => {
connection = new Connection(url);
});
if (!process.env.TEST_LIVE) {
beforeEach(() => {
mockServer.start(MOCK_PORT);
stubRpcWebSocket(connection);
});
afterEach(() => {
mockServer.stop();
restoreRpcWebSocket(connection);
});
}
it('transaction-payer', async () => {
const accountPayer = new Account();
const accountFrom = new Account();
const accountTo = new Account();
const connection = new Connection(url, 'singleGossip');
mockRpc.push([
url,
{
await helpers.airdrop({
connection,
address: accountPayer.publicKey,
amount: LAMPORTS_PER_SOL,
});
await mockRpcResponse({
method: 'getMinimumBalanceForRentExemption',
params: [0, {commitment: 'singleGossip'}],
},
{
error: null,
result: 50,
},
]);
params: [0],
value: 50,
});
const minimumAmount = await connection.getMinimumBalanceForRentExemption(
0,
'singleGossip',
);
const minimumAmount = await connection.getMinimumBalanceForRentExemption(0);
mockRpc.push([
url,
{
method: 'requestAirdrop',
params: [accountPayer.publicKey.toBase58(), LAMPORTS_PER_SOL],
},
{
error: null,
result:
'8WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
let signature = await connection.requestAirdrop(
accountPayer.publicKey,
LAMPORTS_PER_SOL,
);
mockConfirmTransaction(signature);
await connection.confirmTransaction(signature, 'singleGossip');
await helpers.airdrop({
connection,
address: accountFrom.publicKey,
amount: minimumAmount + 12,
});
mockRpc.push([
url,
{
method: 'requestAirdrop',
params: [accountFrom.publicKey.toBase58(), minimumAmount + 12],
},
{
error: null,
result:
'8WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
signature = await connection.requestAirdrop(
accountFrom.publicKey,
minimumAmount + 12,
);
mockConfirmTransaction(signature);
await connection.confirmTransaction(signature, 'singleGossip');
mockRpc.push([
url,
{
method: 'requestAirdrop',
params: [accountTo.publicKey.toBase58(), minimumAmount + 21],
},
{
error: null,
result:
'8WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
signature = await connection.requestAirdrop(
accountTo.publicKey,
minimumAmount + 21,
);
mockConfirmTransaction(signature);
await connection.confirmTransaction(signature, 'singleGossip');
mockGetRecentBlockhash('max');
mockRpc.push([
url,
{
method: 'sendTransaction',
},
{
error: null,
result:
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
},
]);
await helpers.airdrop({
connection,
address: accountTo.publicKey,
amount: minimumAmount + 21,
});
const transaction = new Transaction().add(
SystemProgram.transfer({
@ -117,31 +84,18 @@ test('transaction-payer', async () => {
}),
);
signature = await connection.sendTransaction(
await helpers.processTransaction({
connection,
transaction,
[accountPayer, accountFrom],
{skipPreflight: true},
);
signers: [accountPayer, accountFrom],
commitment: 'singleGossip',
});
mockConfirmTransaction(signature);
await connection.confirmTransaction(signature, 'singleGossip');
const signature = base58.encode(transaction.signature);
mockRpc.push([
url,
{
await mockRpcResponse({
method: 'getSignatureStatuses',
params: [
[
'3WE5w4B7v59x6qjyC4FbG2FEKYKQfvsJwqSxNVmtMjT8TQ31hsZieDHcSgqzxiAoTL56n2w5TncjqEKjLhtF4Vk',
],
],
},
{
error: null,
result: {
context: {
slot: 11,
},
params: [[signature]],
value: [
{
slot: 0,
@ -150,58 +104,41 @@ test('transaction-payer', async () => {
err: null,
},
],
},
},
]);
withContext: true,
});
const {value} = await connection.getSignatureStatus(signature);
if (value !== null) {
expect(typeof value.slot).toEqual('number');
expect(value.err).toBeNull();
expect(typeof value.slot).to.eq('number');
expect(value.err).to.be.null;
} else {
expect(value).not.toBeNull();
expect(value).not.to.be.null;
}
mockRpc.push([
url,
{
await mockRpcResponse({
method: 'getBalance',
params: [accountPayer.publicKey.toBase58(), {commitment: 'singleGossip'}],
},
{
error: null,
result: {
context: {
slot: 11,
},
value: LAMPORTS_PER_SOL - 1,
},
},
]);
withContext: true,
});
// accountPayer should be less than LAMPORTS_PER_SOL as it paid for the transaction
// (exact amount less depends on the current cluster fees)
const balance = await connection.getBalance(accountPayer.publicKey);
expect(balance).toBeGreaterThan(0);
expect(balance).toBeLessThanOrEqual(LAMPORTS_PER_SOL);
const balance = await connection.getBalance(
accountPayer.publicKey,
'singleGossip',
);
expect(balance).to.be.greaterThan(0);
expect(balance).to.be.at.most(LAMPORTS_PER_SOL);
// accountFrom should have exactly 2, since it didn't pay for the transaction
mockRpc.push([
url,
{
await mockRpcResponse({
method: 'getBalance',
params: [accountFrom.publicKey.toBase58(), {commitment: 'singleGossip'}],
},
{
error: null,
result: {
context: {
slot: 11,
},
value: minimumAmount + 2,
},
},
]);
expect(await connection.getBalance(accountFrom.publicKey)).toBe(
minimumAmount + 2,
);
withContext: true,
});
expect(
await connection.getBalance(accountFrom.publicKey, 'singleGossip'),
).to.eq(minimumAmount + 2);
});
});

View File

@ -1,6 +1,8 @@
// @flow
import bs58 from 'bs58';
import {Buffer} from 'buffer';
import nacl from 'tweetnacl';
import {expect, use} from 'chai';
import {Account} from '../src/account';
import {PublicKey} from '../src/publickey';
@ -9,8 +11,9 @@ import {StakeProgram} from '../src/stake-program';
import {SystemProgram} from '../src/system-program';
import {Message} from '../src/message';
describe('compileMessage', () => {
test('accountKeys are ordered', () => {
describe('Transaction', () => {
describe('compileMessage', () => {
it('accountKeys are ordered', () => {
const payer = new Account();
const account2 = new Account();
const account3 = new Account();
@ -32,12 +35,12 @@ describe('compileMessage', () => {
);
const message = transaction.compileMessage();
expect(message.accountKeys[0].equals(payer.publicKey)).toBe(true);
expect(message.accountKeys[1].equals(account2.publicKey)).toBe(true);
expect(message.accountKeys[2].equals(account3.publicKey)).toBe(true);
expect(message.accountKeys[0]).to.eql(payer.publicKey);
expect(message.accountKeys[1]).to.eql(account2.publicKey);
expect(message.accountKeys[2]).to.eql(account3.publicKey);
});
test('payer is first account meta', () => {
it('payer is first account meta', () => {
const payer = new Account();
const other = new Account();
const recentBlockhash = new Account().publicKey.toBase58();
@ -52,14 +55,14 @@ describe('compileMessage', () => {
transaction.sign(payer, other);
const message = transaction.compileMessage();
expect(message.accountKeys[0].equals(payer.publicKey)).toBe(true);
expect(message.accountKeys[1].equals(other.publicKey)).toBe(true);
expect(message.header.numRequiredSignatures).toEqual(2);
expect(message.header.numReadonlySignedAccounts).toEqual(0);
expect(message.header.numReadonlyUnsignedAccounts).toEqual(1);
expect(message.accountKeys[0]).to.eql(payer.publicKey);
expect(message.accountKeys[1]).to.eql(other.publicKey);
expect(message.header.numRequiredSignatures).to.eq(2);
expect(message.header.numReadonlySignedAccounts).to.eq(0);
expect(message.header.numReadonlyUnsignedAccounts).to.eq(1);
});
test('validation', () => {
it('validation', () => {
const payer = new Account();
const other = new Account();
const recentBlockhash = new Account().publicKey.toBase58();
@ -68,13 +71,13 @@ describe('compileMessage', () => {
const transaction = new Transaction();
expect(() => {
transaction.compileMessage();
}).toThrow('Transaction recentBlockhash required');
}).to.throw('Transaction recentBlockhash required');
transaction.recentBlockhash = recentBlockhash;
expect(() => {
transaction.compileMessage();
}).toThrow('No instructions provided');
}).to.throw('No instructions provided');
transaction.add({
keys: [
@ -86,13 +89,13 @@ describe('compileMessage', () => {
expect(() => {
transaction.compileMessage();
}).toThrow('Transaction fee payer required');
}).to.throw('Transaction fee payer required');
transaction.setSigners(payer.publicKey, new Account().publicKey);
expect(() => {
transaction.compileMessage();
}).toThrow('unknown signer');
}).to.throw('unknown signer');
// Expect compile to succeed with implicit fee payer from signers
transaction.setSigners(payer.publicKey);
@ -104,7 +107,7 @@ describe('compileMessage', () => {
transaction.compileMessage();
});
test('payer is writable', () => {
it('payer is writable', () => {
const payer = new Account();
const recentBlockhash = new Account().publicKey.toBase58();
const programId = new Account().publicKey;
@ -115,14 +118,14 @@ describe('compileMessage', () => {
transaction.sign(payer);
const message = transaction.compileMessage();
expect(message.accountKeys[0].equals(payer.publicKey)).toBe(true);
expect(message.header.numRequiredSignatures).toEqual(1);
expect(message.header.numReadonlySignedAccounts).toEqual(0);
expect(message.header.numReadonlyUnsignedAccounts).toEqual(1);
expect(message.accountKeys[0]).to.eql(payer.publicKey);
expect(message.header.numRequiredSignatures).to.eq(1);
expect(message.header.numReadonlySignedAccounts).to.eq(0);
expect(message.header.numReadonlyUnsignedAccounts).to.eq(1);
});
});
});
test('partialSign', () => {
it('partialSign', () => {
const account1 = new Account();
const account2 = new Account();
const recentBlockhash = account1.publicKey.toBase58(); // Fake recentBlockhash
@ -137,26 +140,26 @@ test('partialSign', () => {
const partialTransaction = new Transaction({recentBlockhash}).add(transfer);
partialTransaction.setSigners(account1.publicKey, account2.publicKey);
expect(partialTransaction.signatures[0].signature).toBeNull();
expect(partialTransaction.signatures[1].signature).toBeNull();
expect(partialTransaction.signatures[0].signature).to.be.null;
expect(partialTransaction.signatures[1].signature).to.be.null;
partialTransaction.partialSign(account1);
expect(partialTransaction.signatures[0].signature).not.toBeNull();
expect(partialTransaction.signatures[1].signature).toBeNull();
expect(partialTransaction.signatures[0].signature).not.to.be.null;
expect(partialTransaction.signatures[1].signature).to.be.null;
expect(() => partialTransaction.serialize()).toThrow();
expect(() => partialTransaction.serialize()).to.throw();
expect(() =>
partialTransaction.serialize({requireAllSignatures: false}),
).not.toThrow();
).not.to.throw();
partialTransaction.partialSign(account2);
expect(partialTransaction.signatures[0].signature).not.toBeNull();
expect(partialTransaction.signatures[1].signature).not.toBeNull();
expect(partialTransaction.signatures[0].signature).not.to.be.null;
expect(partialTransaction.signatures[1].signature).not.to.be.null;
expect(() => partialTransaction.serialize()).not.toThrow();
expect(() => partialTransaction.serialize()).not.to.throw();
expect(partialTransaction).toEqual(transaction);
expect(partialTransaction).to.eql(transaction);
if (
partialTransaction.signatures[0].signature != null /* <-- pacify flow */
@ -164,26 +167,26 @@ test('partialSign', () => {
partialTransaction.signatures[0].signature[0] = 0;
expect(() =>
partialTransaction.serialize({requireAllSignatures: false}),
).toThrow();
).to.throw();
expect(() =>
partialTransaction.serialize({
verifySignatures: false,
requireAllSignatures: false,
}),
).not.toThrow();
).not.to.throw();
} else {
throw new Error('unreachable');
}
});
});
describe('dedupe', () => {
describe('dedupe', () => {
const payer = new Account();
const duplicate1 = payer;
const duplicate2 = payer;
const recentBlockhash = new Account().publicKey.toBase58();
const programId = new Account().publicKey;
test('setSigners', () => {
it('setSigners', () => {
const transaction = new Transaction({recentBlockhash}).add({
keys: [
{pubkey: duplicate1.publicKey, isSigner: true, isWritable: true},
@ -199,21 +202,19 @@ describe('dedupe', () => {
duplicate2.publicKey,
);
expect(transaction.signatures.length).toEqual(1);
expect(transaction.signatures[0].publicKey.equals(payer.publicKey)).toBe(
true,
);
expect(transaction.signatures).to.have.length(1);
expect(transaction.signatures[0].publicKey).to.eql(payer.publicKey);
const message = transaction.compileMessage();
expect(message.accountKeys[0].equals(payer.publicKey)).toBe(true);
expect(message.header.numRequiredSignatures).toEqual(1);
expect(message.header.numReadonlySignedAccounts).toEqual(0);
expect(message.header.numReadonlyUnsignedAccounts).toEqual(1);
expect(message.accountKeys[0]).to.eql(payer.publicKey);
expect(message.header.numRequiredSignatures).to.eq(1);
expect(message.header.numReadonlySignedAccounts).to.eq(0);
expect(message.header.numReadonlyUnsignedAccounts).to.eq(1);
transaction.signatures;
});
test('sign', () => {
it('sign', () => {
const transaction = new Transaction({recentBlockhash}).add({
keys: [
{pubkey: duplicate1.publicKey, isSigner: true, isWritable: true},
@ -225,22 +226,20 @@ describe('dedupe', () => {
transaction.sign(payer, duplicate1, duplicate2);
expect(transaction.signatures.length).toEqual(1);
expect(transaction.signatures[0].publicKey.equals(payer.publicKey)).toBe(
true,
);
expect(transaction.signatures).to.have.length(1);
expect(transaction.signatures[0].publicKey).to.eql(payer.publicKey);
const message = transaction.compileMessage();
expect(message.accountKeys[0].equals(payer.publicKey)).toBe(true);
expect(message.header.numRequiredSignatures).toEqual(1);
expect(message.header.numReadonlySignedAccounts).toEqual(0);
expect(message.header.numReadonlyUnsignedAccounts).toEqual(1);
expect(message.accountKeys[0]).to.eql(payer.publicKey);
expect(message.header.numRequiredSignatures).to.eq(1);
expect(message.header.numReadonlySignedAccounts).to.eq(0);
expect(message.header.numReadonlyUnsignedAccounts).to.eq(1);
transaction.signatures;
});
});
});
test('transfer signatures', () => {
it('transfer signatures', () => {
const account1 = new Account();
const account2 = new Account();
const recentBlockhash = account1.publicKey.toBase58(); // Fake recentBlockhash
@ -267,10 +266,10 @@ test('transfer signatures', () => {
signatures: orgTransaction.signatures,
}).add(transfer1, transfer2);
expect(newTransaction).toEqual(orgTransaction);
});
expect(newTransaction).to.eql(orgTransaction);
});
test('dedup signatures', () => {
it('dedup signatures', () => {
const account1 = new Account();
const account2 = new Account();
const recentBlockhash = account1.publicKey.toBase58(); // Fake recentBlockhash
@ -290,9 +289,9 @@ test('dedup signatures', () => {
transfer2,
);
orgTransaction.sign(account1);
});
});
test('use nonce', () => {
it('use nonce', () => {
const account1 = new Account();
const account2 = new Account();
const nonceAccount = new Account();
@ -318,12 +317,12 @@ test('use nonce', () => {
let expectedData = Buffer.alloc(4);
expectedData.writeInt32LE(4, 0);
expect(transferTransaction.instructions).toHaveLength(2);
expect(transferTransaction.instructions[0].programId).toEqual(
expect(transferTransaction.instructions).to.have.length(2);
expect(transferTransaction.instructions[0].programId).to.eql(
SystemProgram.programId,
);
expect(transferTransaction.instructions[0].data).toEqual(expectedData);
expect(transferTransaction.recentBlockhash).toEqual(nonce);
expect(transferTransaction.instructions[0].data).to.eql(expectedData);
expect(transferTransaction.recentBlockhash).to.eq(nonce);
const stakeAccount = new Account();
const voteAccount = new Account();
@ -336,15 +335,15 @@ test('use nonce', () => {
);
stakeTransaction.sign(account1);
expect(stakeTransaction.instructions).toHaveLength(2);
expect(stakeTransaction.instructions[0].programId).toEqual(
expect(stakeTransaction.instructions).to.have.length(2);
expect(stakeTransaction.instructions[0].programId).to.eql(
SystemProgram.programId,
);
expect(stakeTransaction.instructions[0].data).toEqual(expectedData);
expect(stakeTransaction.recentBlockhash).toEqual(nonce);
});
expect(stakeTransaction.instructions[0].data).to.eql(expectedData);
expect(stakeTransaction.recentBlockhash).to.eq(nonce);
});
test('parse wire format and serialize', () => {
it('parse wire format and serialize', () => {
const keypair = nacl.sign.keyPair.fromSeed(
Uint8Array.from(Array(32).fill(8)),
);
@ -370,11 +369,11 @@ test('parse wire format and serialize', () => {
);
const tx = Transaction.from(wireTransaction);
expect(tx).toEqual(expectedTransaction);
expect(wireTransaction).toEqual(expectedTransaction.serialize());
});
expect(tx).to.eql(expectedTransaction);
expect(wireTransaction).to.eql(expectedTransaction.serialize());
});
test('populate transaction', () => {
it('populate transaction', () => {
const recentBlockhash = new PublicKey(1).toString();
const message = {
accountKeys: [
@ -405,12 +404,12 @@ test('populate transaction', () => {
];
const transaction = Transaction.populate(new Message(message), signatures);
expect(transaction.instructions.length).toEqual(1);
expect(transaction.signatures.length).toEqual(2);
expect(transaction.recentBlockhash).toEqual(recentBlockhash);
});
expect(transaction.instructions).to.have.length(1);
expect(transaction.signatures).to.have.length(2);
expect(transaction.recentBlockhash).to.eq(recentBlockhash);
});
test('serialize unsigned transaction', () => {
it('serialize unsigned transaction', () => {
const keypair = nacl.sign.keyPair.fromSeed(
Uint8Array.from(Array(32).fill(8)),
);
@ -424,26 +423,28 @@ test('serialize unsigned transaction', () => {
toPubkey: recipient,
lamports: 49,
});
const expectedTransaction = new Transaction({recentBlockhash}).add(transfer);
const expectedTransaction = new Transaction({recentBlockhash}).add(
transfer,
);
// Empty signature array fails.
expect(expectedTransaction.signatures.length).toBe(0);
expect(expectedTransaction.signatures).to.have.length(0);
expect(() => {
expectedTransaction.serialize();
}).toThrow('Transaction fee payer required');
}).to.throw('Transaction fee payer required');
expect(() => {
expectedTransaction.serialize({verifySignatures: false});
}).toThrow('Transaction fee payer required');
}).to.throw('Transaction fee payer required');
expect(() => {
expectedTransaction.serializeMessage();
}).toThrow('Transaction fee payer required');
}).to.throw('Transaction fee payer required');
expectedTransaction.feePayer = sender.publicKey;
// Transactions with missing signatures will fail sigverify.
expect(() => {
expectedTransaction.serialize();
}).toThrow('Signature verification failed');
}).to.throw('Signature verification failed');
// Serializing without signatures is allowed if sigverify disabled.
expectedTransaction.serialize({verifySignatures: false});
@ -453,12 +454,12 @@ test('serialize unsigned transaction', () => {
expectedTransaction.feePayer = null;
expectedTransaction.setSigners(sender.publicKey);
expect(expectedTransaction.signatures.length).toBe(1);
expect(expectedTransaction.signatures).to.have.length(1);
// Transactions with missing signatures will fail sigverify.
expect(() => {
expectedTransaction.serialize();
}).toThrow('Signature verification failed');
}).to.throw('Signature verification failed');
// Serializing without signatures is allowed if sigverify disabled.
expectedTransaction.serialize({verifySignatures: false});
@ -474,13 +475,13 @@ test('serialize unsigned transaction', () => {
'AAAAMQAAAAAAAAA=',
'base64',
);
expect(
expectedTransaction.serialize({requireAllSignatures: false}),
).toStrictEqual(expectedSerializationWithNoSignatures);
expect(expectedTransaction.serialize({requireAllSignatures: false})).to.eql(
expectedSerializationWithNoSignatures,
);
// Properly signed transaction succeeds
expectedTransaction.partialSign(sender);
expect(expectedTransaction.signatures.length).toBe(1);
expect(expectedTransaction.signatures).to.have.length(1);
const expectedSerialization = Buffer.from(
'AVuErQHaXv0SG0/PchunfxHKt8wMRfMZzqV0tkC5qO6owYxWU2v871AoWywGoFQr4z+q/7mE8lIufNl/' +
'kxj+nQ0BAAEDE5j2LG0aRXxRumpLXz29L2n8qTIWIY3ImX5Ba9F9k8r9Q5/Mtmcn8onFxt47xKj+XdXX' +
@ -488,11 +489,11 @@ test('serialize unsigned transaction', () => {
'ROug7bEsbx0xxuDkqEvwUusBAgIAAQwCAAAAMQAAAAAAAAA=',
'base64',
);
expect(expectedTransaction.serialize()).toStrictEqual(expectedSerialization);
expect(expectedTransaction.signatures.length).toBe(1);
});
expect(expectedTransaction.serialize()).to.eql(expectedSerialization);
expect(expectedTransaction.signatures).to.have.length(1);
});
test('deprecated - externally signed stake delegate', () => {
it('deprecated - externally signed stake delegate', () => {
const from_keypair = nacl.sign.keyPair.fromSeed(
Uint8Array.from(Array(32).fill(1)),
);
@ -511,10 +512,10 @@ test('deprecated - externally signed stake delegate', () => {
const tx_bytes = tx.serializeMessage();
const signature = nacl.sign.detached(tx_bytes, from.secretKey);
tx.addSignature(from.publicKey, signature);
expect(tx.verifySignatures()).toBe(true);
});
expect(tx.verifySignatures()).to.be.true;
});
test('externally signed stake delegate', () => {
it('externally signed stake delegate', () => {
const from_keypair = nacl.sign.keyPair.fromSeed(
Uint8Array.from(Array(32).fill(1)),
);
@ -533,5 +534,6 @@ test('externally signed stake delegate', () => {
const tx_bytes = tx.serializeMessage();
const signature = nacl.sign.detached(tx_bytes, from.secretKey);
tx.addSignature(from.publicKey, signature);
expect(tx.verifySignatures()).toBe(true);
expect(tx.verifySignatures()).to.be.true;
});
});

View File

@ -4,7 +4,10 @@
* The connection url to use when running unit tests against a live cluster
*/
export const url = 'http://localhost:8899/';
export const MOCK_PORT = 9999;
export const url = process.env.TEST_LIVE
? 'http://localhost:8899/'
: 'http://localhost:9999/';
//export const url = 'https://devnet.solana.com/';
//export const url = 'http://devnet.solana.com/';

View File

@ -1,10 +1,13 @@
// @flow
import {Buffer} from 'buffer';
import {expect} from 'chai';
import nacl from 'tweetnacl';
import {PublicKey} from '../src/publickey';
import {ValidatorInfo} from '../src/validator-info';
test('from config account data', () => {
describe('ValidatorInfo', () => {
it('from config account data', () => {
const keypair = nacl.sign.keyPair.fromSeed(
Uint8Array.from(Array(32).fill(8)),
);
@ -32,5 +35,6 @@ test('from config account data', () => {
);
const info = ValidatorInfo.fromConfigData(configData);
expect(info).toEqual(expectedValidatorInfo);
expect(info).to.eql(expectedValidatorInfo);
});
});

View File

@ -1,68 +1,68 @@
// @flow
import bs58 from 'bs58';
import {Buffer} from 'buffer';
import {expect, use} from 'chai';
import chaiAsPromised from 'chai-as-promised';
import {Connection} from '../src';
import {url} from './url';
import {mockRpcEnabled} from './__mocks__/node-fetch';
import {sleep} from '../src/util/sleep';
describe('websocket', () => {
if (mockRpcEnabled) {
test('no-op', () => {});
console.log('non-live test skipped');
return;
}
use(chaiAsPromised);
if (process.env.TEST_LIVE) {
describe('websocket', () => {
const connection = new Connection(url);
test('connect and disconnect', async () => {
it('connect and disconnect', async () => {
const testSignature = bs58.encode(Buffer.alloc(64));
const id = connection.onSignature(testSignature, () => {});
// wait for websocket to connect
await sleep(100);
expect(connection._rpcWebSocketConnected).toBe(true);
expect(connection._rpcWebSocketHeartbeat).not.toBe(null);
expect(connection._rpcWebSocketConnected).to.be.true;
expect(connection._rpcWebSocketHeartbeat).not.to.eq(null);
// test if socket is open
await connection._rpcWebSocket.notify('ping');
await connection.removeSignatureListener(id);
expect(connection._rpcWebSocketConnected).toBe(false);
expect(connection._rpcWebSocketHeartbeat).not.toBe(null);
expect(connection._rpcWebSocketIdleTimeout).not.toBe(null);
expect(connection._rpcWebSocketConnected).to.eq(false);
expect(connection._rpcWebSocketHeartbeat).not.to.eq(null);
expect(connection._rpcWebSocketIdleTimeout).not.to.eq(null);
// wait for websocket to disconnect
await sleep(1100);
expect(connection._rpcWebSocketConnected).toBe(false);
expect(connection._rpcWebSocketHeartbeat).toBe(null);
expect(connection._rpcWebSocketIdleTimeout).toBe(null);
expect(connection._rpcWebSocketConnected).to.eq(false);
expect(connection._rpcWebSocketHeartbeat).to.eq(null);
expect(connection._rpcWebSocketIdleTimeout).to.eq(null);
// test if socket is closed
await expect(connection._rpcWebSocket.notify('ping')).rejects.toThrow(
await expect(connection._rpcWebSocket.notify('ping')).to.be.rejectedWith(
'socket not ready',
);
});
test('idle timeout', async () => {
it('idle timeout', async () => {
const testSignature = bs58.encode(Buffer.alloc(64));
const id = connection.onSignature(testSignature, () => {});
// wait for websocket to connect
await sleep(100);
expect(connection._rpcWebSocketIdleTimeout).toBe(null);
expect(connection._rpcWebSocketIdleTimeout).to.eq(null);
await connection.removeSignatureListener(id);
expect(connection._rpcWebSocketIdleTimeout).not.toBe(null);
expect(connection._rpcWebSocketIdleTimeout).not.to.eq(null);
const nextId = connection.onSignature(testSignature, () => {});
expect(connection._rpcWebSocketIdleTimeout).toBe(null);
expect(connection._rpcWebSocketIdleTimeout).to.eq(null);
await connection.removeSignatureListener(nextId);
expect(connection._rpcWebSocketIdleTimeout).not.toBe(null);
expect(connection._rpcWebSocketIdleTimeout).not.to.eq(null);
// wait for websocket to disconnect
await sleep(1100);
expect(connection._rpcWebSocketIdleTimeout).toBe(null);
expect(connection._rpcWebSocketIdleTimeout).to.eq(null);
});
});
});
}