solana/web3.js/src/loader.js

129 lines
3.3 KiB
JavaScript
Raw Normal View History

// @flow
import * as BufferLayout from 'buffer-layout';
2018-10-26 13:19:47 -07:00
import {Account} from './account';
import {PublicKey} from './publickey';
import {NUM_TICKS_PER_SECOND} from './timing';
2018-10-26 13:19:47 -07:00
import {Transaction} from './transaction';
import {sendAndConfirmTransaction} from './util/send-and-confirm-transaction';
import {sleep} from './util/sleep';
2018-10-26 13:19:47 -07:00
import type {Connection} from './connection';
/**
* Program loader interface
*/
export class Loader {
/**
* @private
*/
connection: Connection;
/**
* @private
*/
programId: PublicKey;
/**
* Amount of program data placed in each load Transaction
*/
static get chunkSize(): number {
return 256;
}
/**
* @param connection The connection to use
* @param programId Public key that identifies the loader
*/
constructor(connection: Connection, programId: PublicKey) {
Object.assign(this, {connection, programId});
}
/**
* Load program data
*
* @param program Account to load the program info
* @param data Program data
*/
async load(program: Account, data: Array<number>) {
2019-03-14 13:27:47 -07:00
const dataLayout = BufferLayout.struct([
BufferLayout.u32('instruction'),
BufferLayout.u32('offset'),
BufferLayout.u32('bytesLength'),
BufferLayout.u32('bytesLengthPadding'),
BufferLayout.seq(
BufferLayout.u8('byte'),
2018-11-04 11:41:21 -08:00
BufferLayout.offset(BufferLayout.u32(), -8),
'bytes',
),
]);
const chunkSize = Loader.chunkSize;
let offset = 0;
let array = data;
let transactions = [];
while (array.length > 0) {
const bytes = array.slice(0, chunkSize);
2019-03-14 13:27:47 -07:00
const data = Buffer.alloc(chunkSize + 16);
dataLayout.encode(
{
instruction: 0, // Load instruction
offset,
bytes,
},
2019-03-14 13:27:47 -07:00
data,
);
const transaction = new Transaction().add({
keys: [{pubkey: program.publicKey, isSigner: true}],
programId: this.programId,
2019-03-14 13:27:47 -07:00
data,
});
2018-11-04 11:41:21 -08:00
transactions.push(
sendAndConfirmTransaction(this.connection, transaction, program),
2018-11-04 11:41:21 -08:00
);
// Delay ~1 tick between write transactions in an attempt to reduce AccountInUse errors
// since all the write transactions modify the same program account
await sleep(1000 / NUM_TICKS_PER_SECOND);
// Run up to 8 Loads in parallel to prevent too many parallel transactions from
// getting rejected with AccountInUse.
//
// TODO: 8 was selected empirically and should probably be revisited
if (transactions.length === 8) {
await Promise.all(transactions);
transactions = [];
}
offset += chunkSize;
array = array.slice(chunkSize);
}
await Promise.all(transactions);
}
/**
* Finalize an account loaded with program data for execution
*
* @param program `load()`ed Account
*/
async finalize(program: Account) {
2019-03-19 12:54:47 -07:00
const dataLayout = BufferLayout.struct([BufferLayout.u32('instruction')]);
2019-03-14 13:27:47 -07:00
const data = Buffer.alloc(dataLayout.span);
dataLayout.encode(
{
instruction: 1, // Finalize instruction
},
2019-03-14 13:27:47 -07:00
data,
);
2019-02-14 22:04:13 -08:00
const transaction = new Transaction().add({
keys: [{pubkey: program.publicKey, isSigner: true}],
programId: this.programId,
2019-03-14 13:27:47 -07:00
data,
});
await sendAndConfirmTransaction(this.connection, transaction, program);
}
}