itns-sidechain/lib/node/fullnode.js

700 lines
17 KiB
JavaScript
Raw Permalink Normal View History

/*!
2018-08-01 20:00:09 -07:00
* fullnode.js - full node for hsd
2018-02-01 13:40:45 -08:00
* Copyright (c) 2017-2018, Christopher Jeffrey (MIT License).
2018-08-01 20:00:09 -07:00
* https://github.com/handshake-org/hsd
2016-03-10 02:40:33 -08:00
*/
2016-06-13 01:06:01 -07:00
'use strict';
2018-07-19 05:40:48 -07:00
const assert = require('bsert');
2017-06-29 20:54:07 -07:00
const Chain = require('../blockchain/chain');
const Fees = require('../mempool/fees');
const Mempool = require('../mempool/mempool');
const Pool = require('../net/pool');
const Miner = require('../mining/miner');
const Node = require('./node');
2017-10-26 12:31:08 -07:00
const HTTP = require('./http');
const RPC = require('./rpc');
2020-06-11 09:25:17 -07:00
const blockstore = require('../blockstore');
2018-01-02 20:24:56 -08:00
const pkg = require('../pkg');
const {RootServer, RecursiveServer} = require('../dns/server');
2016-03-10 02:40:33 -08:00
/**
2017-11-16 19:43:07 -08:00
* Full Node
2017-02-07 14:17:41 -08:00
* Respresents a fullnode complete with a
* chain, mempool, miner, etc.
2017-02-03 22:47:26 -08:00
* @alias module:node.FullNode
* @extends Node
2016-03-10 02:40:33 -08:00
*/
2017-11-16 19:43:07 -08:00
class FullNode extends Node {
/**
* Create a full node.
* @constructor
* @param {Object?} options
*/
constructor(options) {
2018-03-11 20:29:05 -07:00
super(pkg.core, pkg.cfg, 'debug.log', options);
2017-11-16 19:43:07 -08:00
this.opened = false;
// SPV flag.
this.spv = false;
2020-06-11 09:25:17 -07:00
// Instantiate block storage.
this.blocks = blockstore.create({
network: this.network,
logger: this.logger,
prefix: this.config.prefix,
cacheSize: this.config.mb('block-cache-size'),
memory: this.memory
});
2017-11-16 19:43:07 -08:00
// Instantiate blockchain.
this.chain = new Chain({
network: this.network,
logger: this.logger,
2020-06-11 09:25:17 -07:00
blocks: this.blocks,
2017-11-16 19:43:07 -08:00
workers: this.workers,
2017-12-06 17:05:00 -08:00
memory: this.config.bool('memory'),
2017-11-16 19:43:07 -08:00
prefix: this.config.prefix,
maxFiles: this.config.uint('max-files'),
cacheSize: this.config.mb('cache-size'),
prune: this.config.bool('prune'),
checkpoints: this.config.bool('checkpoints'),
entryCache: this.config.uint('entry-cache'),
chainMigrate: this.config.uint('chain-migrate'),
2017-11-16 19:43:07 -08:00
indexTX: this.config.bool('index-tx'),
indexAddress: this.config.bool('index-address'),
compactTreeOnInit: this.config.bool('compact-tree-on-init'),
compactTreeInitInterval: this.config.uint('compact-tree-init-interval')
2017-11-16 19:43:07 -08:00
});
// Fee estimation.
this.fees = new Fees(this.logger);
this.fees.init();
// Mempool needs access to the chain.
this.mempool = new Mempool({
network: this.network,
logger: this.logger,
workers: this.workers,
chain: this.chain,
fees: this.fees,
2017-12-06 17:05:00 -08:00
memory: this.config.bool('memory'),
2017-11-16 19:43:07 -08:00
prefix: this.config.prefix,
persistent: this.config.bool('persistent-mempool'),
maxSize: this.config.mb('mempool-size'),
limitFree: this.config.bool('limit-free'),
limitFreeRelay: this.config.uint('limit-free-relay'),
requireStandard: this.config.bool('require-standard'),
rejectAbsurdFees: this.config.bool('reject-absurd-fees'),
2021-01-12 10:37:08 -05:00
indexAddress: this.config.bool('index-address'),
expiryTime: this.config.uint('mempool-expiry-time')
2017-11-16 19:43:07 -08:00
});
// Pool needs access to the chain and mempool.
this.pool = new Pool({
network: this.network,
logger: this.logger,
chain: this.chain,
mempool: this.mempool,
prefix: this.config.prefix,
compact: this.config.bool('compact'),
bip37: this.config.bool('bip37'),
identityKey: this.identityKey,
2017-11-16 19:43:07 -08:00
maxOutbound: this.config.uint('max-outbound'),
maxInbound: this.config.uint('max-inbound'),
maxProofRPS: this.config.uint('max-proof-rps'),
2017-11-16 19:43:07 -08:00
createSocket: this.config.func('create-socket'),
proxy: this.config.str('proxy'),
onion: this.config.bool('onion'),
2020-03-02 18:08:38 -08:00
brontideOnly: this.config.bool('brontide-only'),
2017-11-16 19:43:07 -08:00
upnp: this.config.bool('upnp'),
seeds: this.config.array('seeds'),
nodes: this.config.array('nodes'),
only: this.config.array('only'),
publicHost: this.config.str('public-host'),
publicPort: this.config.uint('public-port'),
publicBrontidePort: this.config.uint('public-brontide-port'),
2017-11-16 19:43:07 -08:00
host: this.config.str('host'),
port: this.config.uint('port'),
brontidePort: this.config.uint('brontide-port'),
2017-11-16 19:43:07 -08:00
listen: this.config.bool('listen'),
memory: this.config.bool('memory'),
agent: this.config.str('agent')
2017-11-16 19:43:07 -08:00
});
// Miner needs access to the chain and mempool.
this.miner = new Miner({
network: this.network,
logger: this.logger,
workers: this.workers,
chain: this.chain,
mempool: this.mempool,
address: this.config.array('coinbase-address'),
coinbaseFlags: this.config.str('coinbase-flags'),
preverify: this.config.bool('preverify'),
minWeight: this.config.uint('min-weight'),
2017-11-16 19:43:07 -08:00
maxWeight: this.config.uint('max-weight'),
reservedWeight: this.config.uint('reserved-weight'),
reservedSigops: this.config.uint('reserved-sigops')
});
// RPC needs access to the node.
this.rpc = new RPC(this);
// HTTP needs access to the node.
this.http = new HTTP({
network: this.network,
logger: this.logger,
node: this,
prefix: this.config.prefix,
ssl: this.config.bool('ssl'),
keyFile: this.config.path('ssl-key'),
certFile: this.config.path('ssl-cert'),
host: this.config.str('http-host'),
port: this.config.uint('http-port'),
apiKey: this.config.str('api-key'),
2018-07-10 19:16:08 -07:00
noAuth: this.config.bool('no-auth'),
cors: this.config.bool('cors')
2017-11-16 19:43:07 -08:00
});
if (!this.config.bool('no-dns')) {
const publicHost = this.config.str('public-host');
this.ns = new RootServer({
logger: this.logger,
key: this.identityKey,
host: this.config.str('ns-host'),
port: this.config.uint('ns-port', this.network.nsPort),
lookup: key => this.chain.db.tree.get(key),
publicHost: this.config.str('ns-public-host', publicHost),
noSig0: this.config.bool('no-sig0')
});
if (!this.config.bool('no-rs')) {
this.rs = new RecursiveServer({
logger: this.logger,
key: this.identityKey,
host: this.config.str('rs-host'),
port: this.config.uint('rs-port', this.network.rsPort),
stubHost: this.ns.host,
stubPort: this.ns.port,
2019-11-11 16:18:21 -08:00
noUnbound: this.config.bool('rs-no-unbound'),
noSig0: this.config.bool('no-sig0')
});
}
}
2018-02-03 20:48:36 -08:00
2017-11-16 19:43:07 -08:00
this.init();
}
2017-11-16 19:43:07 -08:00
/**
* Initialize the node.
* @private
*/
init() {
// Bind to errors
this.chain.on('error', err => this.error(err));
this.chain.on('abort', err => this.abort(err));
2017-11-16 19:43:07 -08:00
this.mempool.on('error', err => this.error(err));
this.pool.on('error', err => this.error(err));
this.miner.on('error', err => this.error(err));
if (this.http)
this.http.on('error', err => this.error(err));
this.mempool.on('tx', (tx) => {
this.miner.cpu.notifyEntry();
this.emit('tx', tx);
});
this.mempool.on('claim', (claim) => {
this.miner.cpu.notifyEntry();
this.emit('claim', claim);
});
this.mempool.on('airdrop', (proof) => {
this.miner.cpu.notifyEntry();
this.emit('airdrop', proof);
});
this.chain.on('connect', async (entry, block, view) => {
2017-11-16 19:43:07 -08:00
try {
await this.mempool._addBlock(entry, block.txs, view);
2017-11-16 19:43:07 -08:00
} catch (e) {
this.error(e);
}
this.emit('block', block);
this.emit('connect', entry, block);
});
this.chain.on('disconnect', async (entry, block) => {
try {
await this.mempool._removeBlock(entry, block.txs);
} catch (e) {
this.error(e);
}
this.emit('disconnect', entry, block);
});
this.chain.on('reorganize', async (tip, competitor, fork) => {
2017-11-16 19:43:07 -08:00
try {
await this.mempool._handleReorg();
} catch (e) {
this.error(e);
}
this.emit('reorganize', tip, competitor, fork);
2017-11-16 19:43:07 -08:00
});
this.chain.on('reset', async (tip) => {
try {
await this.mempool._reset();
} catch (e) {
this.error(e);
}
this.emit('reset', tip);
});
this.chain.on('tree compact start', (treeRoot, entry) => {
this.emit('tree compact start', treeRoot, entry);
});
this.chain.on('tree compact end', (treeRoot, entry) => {
this.emit('tree compact end', treeRoot, entry);
});
this.chain.on('tree reconstruct start', () => {
this.emit('tree reconstruct start');
});
this.chain.on('tree reconstruct end', () => {
this.emit('tree reconstruct end');
});
2017-11-16 19:43:07 -08:00
this.loadPlugins();
}
2016-04-04 18:45:02 -07:00
2017-11-16 19:43:07 -08:00
/**
* Open the node and all its child objects,
* wait for the database to load.
* @alias FullNode#open
* @returns {Promise}
*/
2017-11-16 19:43:07 -08:00
async open() {
assert(!this.opened, 'FullNode is already open.');
this.opened = true;
2017-11-01 12:57:11 -07:00
2017-11-16 19:43:07 -08:00
await this.handlePreopen();
2020-06-11 09:25:17 -07:00
await this.blocks.open();
await this.chain.open();
2017-11-16 19:43:07 -08:00
await this.mempool.open();
await this.miner.open();
await this.pool.open();
2017-11-16 19:43:07 -08:00
await this.openPlugins();
2016-09-20 14:56:54 -07:00
2017-11-16 19:43:07 -08:00
await this.http.open();
if (this.ns)
await this.ns.open();
if (this.rs)
await this.rs.open();
2017-11-16 19:43:07 -08:00
await this.handleOpen();
2017-02-28 14:10:45 -08:00
if (this.has('walletdb')) {
const {wdb} = this.require('walletdb');
if (this.miner.addresses.length === 0) {
const addr = await wdb.primary.receiveAddress();
this.miner.addresses.push(addr);
}
}
2017-11-16 19:43:07 -08:00
this.logger.info('Node is loaded.');
this.emit('open');
2017-11-16 19:43:07 -08:00
}
2017-11-16 19:43:07 -08:00
/**
* Close the node, wait for the database to close.
* @alias FullNode#close
* @returns {Promise}
*/
2017-11-16 19:43:07 -08:00
async close() {
assert(this.opened, 'FullNode is not open.');
this.opened = false;
2017-11-01 12:57:11 -07:00
2017-11-16 19:43:07 -08:00
await this.handlePreclose();
await this.http.close();
if (this.rs)
await this.rs.close();
if (this.ns)
await this.ns.close();
2016-09-20 14:56:54 -07:00
2017-11-16 19:43:07 -08:00
await this.closePlugins();
2016-09-23 18:32:49 -07:00
2017-11-16 19:43:07 -08:00
await this.pool.close();
await this.miner.close();
await this.mempool.close();
await this.chain.close();
2020-06-11 09:25:17 -07:00
await this.blocks.close();
2017-11-16 19:43:07 -08:00
await this.handleClose();
2017-11-16 19:43:07 -08:00
this.logger.info('Node is closed.');
this.emit('closed');
this.emit('close');
2017-11-16 19:43:07 -08:00
}
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
/**
* Rescan for any missed transactions.
* @param {Number|Hash} start - Start block.
2023-10-11 20:02:11 +04:00
* @param {BloomFilter} filter
2017-11-16 19:43:07 -08:00
* @param {Function} iter - Iterator.
* @returns {Promise}
*/
2017-11-16 19:43:07 -08:00
scan(start, filter, iter) {
return this.chain.scan(start, filter, iter);
}
2023-10-11 19:54:11 +04:00
/**
* Interactive rescan for any missed transactions.
* @param {Number|Hash} start - Start block.
2023-10-11 20:02:11 +04:00
* @param {BloomFilter} filter
2023-10-11 19:54:11 +04:00
* @param {Function} iter - Iterator.
* @param {Boolean} [fullLock=false] - lock the whole chain instead of per
* scan.
2023-10-11 19:54:11 +04:00
* @returns {Promise}
*/
scanInteractive(start, filter, iter, fullLock = false) {
return this.chain.scanInteractive(start, filter, iter, fullLock);
2023-10-11 19:54:11 +04:00
}
2017-11-16 19:43:07 -08:00
/**
2018-08-01 20:00:09 -07:00
* Broadcast a transaction.
* @param {TX|Block|Claim|AirdropProof} item
2017-11-16 19:43:07 -08:00
* @returns {Promise}
*/
2017-11-16 19:43:07 -08:00
async broadcast(item) {
try {
await this.pool.broadcast(item);
} catch (e) {
this.emit('error', e);
}
}
2016-03-29 16:14:39 -07:00
2017-11-16 19:43:07 -08:00
/**
* Add transaction to mempool, broadcast.
* @param {TX} tx
*/
async sendTX(tx) {
let missing;
2017-11-16 19:43:07 -08:00
try {
missing = await this.mempool.addTX(tx);
} catch (err) {
if (err.type === 'VerifyError' && err.score === 0) {
this.error(err);
2018-07-15 10:47:14 -07:00
this.logger.warning('Verification failed for tx: %x.', tx.hash());
2017-11-16 19:43:07 -08:00
this.logger.warning('Attempting to broadcast anyway...');
this.broadcast(tx);
return;
}
throw err;
}
2017-11-16 19:43:07 -08:00
if (missing) {
2018-07-15 10:47:14 -07:00
this.logger.warning('TX was orphaned in mempool: %x.', tx.hash());
2016-09-21 22:58:27 -07:00
this.logger.warning('Attempting to broadcast anyway...');
this.broadcast(tx);
2016-08-26 05:02:08 -07:00
}
}
2017-11-16 19:43:07 -08:00
/**
* Add transaction to mempool, broadcast. Silence errors.
* @param {TX} tx
* @returns {Promise}
*/
2016-03-29 16:14:39 -07:00
2017-11-16 19:43:07 -08:00
async relay(tx) {
try {
await this.sendTX(tx);
} catch (e) {
this.error(e);
}
2017-01-14 19:21:46 -08:00
}
/**
* Add claim to mempool, broadcast.
* @param {Claim} claim
*/
async sendClaim(claim) {
try {
await this.mempool.addClaim(claim);
} catch (err) {
if (err.type === 'VerifyError' && err.score === 0) {
this.error(err);
2018-07-15 10:47:14 -07:00
this.logger.warning('Verification failed for claim: %x.', claim.hash());
this.logger.warning('Attempting to broadcast anyway...');
this.broadcast(claim);
return;
}
throw err;
}
}
/**
* Add claim to mempool, broadcast. Silence errors.
* @param {Claim} claim
* @returns {Promise}
*/
async relayClaim(claim) {
try {
await this.sendClaim(claim);
} catch (e) {
this.error(e);
}
}
/**
* Add airdrop proof to mempool, broadcast.
* @param {AirdropProof} proof
*/
async sendAirdrop(proof) {
try {
await this.mempool.addAirdrop(proof);
} catch (err) {
if (err.type === 'VerifyError' && err.score === 0) {
this.error(err);
this.logger.warning('Verification failed for proof: %x.', proof.hash());
this.logger.warning('Attempting to broadcast anyway...');
this.broadcast(proof);
return;
}
throw err;
}
}
/**
* Add airdrop proof to mempool, broadcast. Silence errors.
* @param {AirdropProof} proof
* @returns {Promise}
*/
async relayAirdrop(proof) {
try {
await this.sendAirdrop(proof);
} catch (e) {
this.error(e);
}
}
2017-11-16 19:43:07 -08:00
/**
* Connect to the network.
* @returns {Promise}
*/
2016-05-19 11:56:11 -07:00
2017-11-16 19:43:07 -08:00
connect() {
return this.pool.connect();
}
2017-11-16 19:43:07 -08:00
/**
* Disconnect from the network.
* @returns {Promise}
*/
2016-04-03 06:11:30 -07:00
2017-11-16 19:43:07 -08:00
disconnect() {
return this.pool.disconnect();
}
2017-11-16 19:43:07 -08:00
/**
* Start the blockchain sync.
*/
2016-03-22 17:36:58 -07:00
2017-11-16 19:43:07 -08:00
startSync() {
return this.pool.startSync();
}
2017-11-16 19:43:07 -08:00
/**
* Stop syncing the blockchain.
*/
2016-03-22 17:36:58 -07:00
2017-11-16 19:43:07 -08:00
stopSync() {
return this.pool.stopSync();
}
2017-11-16 19:43:07 -08:00
/**
* Retrieve a block from the chain database.
* @param {Hash} hash
* @returns {Promise} - Returns {@link Block}.
*/
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
getBlock(hash) {
return this.chain.getBlock(hash);
}
2017-11-16 19:43:07 -08:00
/**
* Retrieve a coin from the mempool or chain database.
* Takes into account spent coins in the mempool.
* @param {Hash} hash
* @param {Number} index
* @returns {Promise} - Returns {@link Coin}.
*/
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
async getCoin(hash, index) {
const coin = this.mempool.getCoin(hash, index);
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
if (coin)
return coin;
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
if (this.mempool.isSpent(hash, index))
return null;
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
return this.chain.getCoin(hash, index);
}
2017-11-16 19:43:07 -08:00
/**
* Get coins that pertain to an address from the mempool or chain database.
* Takes into account spent coins in the mempool.
* @param {Address} addrs
* @returns {Promise} - Returns {@link Coin}[].
*/
2016-08-15 15:46:37 -07:00
2017-11-16 19:43:07 -08:00
async getCoinsByAddress(addrs) {
const mempool = this.mempool.getCoinsByAddress(addrs);
const chain = await this.chain.getCoinsByAddress(addrs);
const out = [];
2016-03-21 16:29:02 -07:00
2017-11-16 19:43:07 -08:00
for (const coin of chain) {
const spent = this.mempool.isSpent(coin.hash, coin.index);
2017-11-16 19:43:07 -08:00
if (spent)
continue;
2017-11-16 19:43:07 -08:00
out.push(coin);
}
2016-03-21 16:29:02 -07:00
2017-11-16 19:43:07 -08:00
for (const coin of mempool)
out.push(coin);
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
return out;
}
2016-08-26 05:02:08 -07:00
2017-11-16 19:43:07 -08:00
/**
* Retrieve transactions pertaining to an
* address from the mempool or chain database.
* @param {Address} addrs
* @returns {Promise} - Returns {@link TXMeta}[].
*/
async getMetaByAddress(addrs) {
const mempool = this.mempool.getMetaByAddress(addrs);
const chain = await this.chain.getMetaByAddress(addrs);
return chain.concat(mempool);
}
2016-08-26 05:02:08 -07:00
2017-11-16 19:43:07 -08:00
/**
* Retrieve a transaction from the mempool or chain database.
* @param {Hash} hash
* @returns {Promise} - Returns {@link TXMeta}.
*/
2017-11-16 19:43:07 -08:00
async getMeta(hash) {
const meta = this.mempool.getMeta(hash);
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
if (meta)
return meta;
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
return this.chain.getMeta(hash);
}
2016-03-21 16:29:02 -07:00
2017-11-16 19:43:07 -08:00
/**
* Retrieve a spent coin viewpoint from mempool or chain database.
* @param {TXMeta} meta
* @returns {Promise} - Returns {@link CoinView}.
*/
2017-11-16 19:43:07 -08:00
async getMetaView(meta) {
if (meta.height === -1)
return this.mempool.getSpentView(meta.tx);
return this.chain.getSpentView(meta.tx);
}
2017-11-16 19:43:07 -08:00
/**
* Retrieve transactions pertaining to an
* address from the mempool or chain database.
* @param {Address} addrs
* @returns {Promise} - Returns {@link TX}[].
*/
2017-11-16 19:43:07 -08:00
async getTXByAddress(addrs) {
const mtxs = await this.getMetaByAddress(addrs);
const out = [];
2017-11-16 19:43:07 -08:00
for (const mtx of mtxs)
out.push(mtx.tx);
2017-11-16 19:43:07 -08:00
return out;
}
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
/**
* Retrieve a transaction from the mempool or chain database.
* @param {Hash} hash
* @returns {Promise} - Returns {@link TX}.
*/
2017-11-16 19:43:07 -08:00
async getTX(hash) {
const mtx = await this.getMeta(hash);
2017-11-16 19:43:07 -08:00
if (!mtx)
return null;
2017-11-16 19:43:07 -08:00
return mtx.tx;
}
2017-11-16 19:43:07 -08:00
/**
* Test whether the mempool or chain contains a transaction.
* @param {Hash} hash
* @returns {Promise} - Returns Boolean.
*/
2017-11-16 19:43:07 -08:00
async hasTX(hash) {
if (this.mempool.hasEntry(hash))
return true;
2016-03-10 02:40:33 -08:00
2017-11-16 19:43:07 -08:00
return this.chain.hasTX(hash);
}
/**
* Get current name state.
* @param {Buffer} nameHash
* @returns {NameState}
*/
async getNameStatus(nameHash) {
const height = this.chain.height + 1;
return this.chain.db.getNameStatus(nameHash, height);
}
2017-11-16 19:43:07 -08:00
}
2016-03-10 02:40:33 -08:00
2016-05-15 18:07:06 -07:00
/*
* Expose
*/
module.exports = FullNode;