fcoin/lib/node/node.js
2018-03-29 21:56:46 -07:00

413 lines
8.7 KiB
JavaScript

/*!
* node.js - node object for bcoin
* Copyright (c) 2014-2015, Fedor Indutny (MIT License)
* Copyright (c) 2014-2017, Christopher Jeffrey (MIT License).
* https://github.com/bcoin-org/bcoin
*/
'use strict';
const assert = require('assert');
const AsyncObject = require('../utils/asyncobject');
const util = require('../utils/util');
const Network = require('../protocol/network');
const Logger = require('./logger');
const WorkerPool = require('../workers/workerpool');
const secp256k1 = require('bcrypto/lib/secp256k1');
const Config = require('./config');
/**
* Base class from which every other
* Node-like object inherits.
* @alias module:node.Node
* @constructor
* @abstract
* @param {Object} options
*/
function Node(module, config, file, options) {
if (!(this instanceof Node))
return new Node(module, config, file, options);
AsyncObject.call(this);
this.config = new Config(module, {
suffix: 'network',
fallback: 'main',
alias: { 'n': 'network' }
});
this.config.inject(options);
this.config.load(options);
if (options.config)
this.config.open(config);
this.network = Network.get(this.config.getSuffix());
this.startTime = -1;
this.bound = [];
this.plugins = Object.create(null);
this.stack = [];
this.logger = null;
this.workers = null;
this.spv = false;
this.chain = null;
this.fees = null;
this.mempool = null;
this.pool = null;
this.miner = null;
this.http = null;
this.init(file);
}
Object.setPrototypeOf(Node.prototype, AsyncObject.prototype);
/**
* Initialize options.
* @private
* @param {Object} options
*/
Node.prototype.initOptions = function initOptions(file) {
const config = this.config;
let logger = new Logger();
if (config.has('logger'))
logger = config.obj('logger');
logger.set({
filename: config.bool('log-file')
? config.location(file)
: null,
level: config.str('log-level'),
console: config.bool('log-console'),
shrink: config.bool('log-shrink')
});
this.logger = logger.context('node');
this.workers = new WorkerPool({
enabled: config.bool('workers'),
size: config.uint('workers-size'),
timeout: config.uint('workers-timeout'),
file: config.str('worker-file')
});
};
/**
* Initialize node.
* @private
* @param {Object} options
*/
Node.prototype.init = function init(file) {
this.initOptions(file);
this.on('error', () => {});
this.workers.on('spawn', (child) => {
this.logger.info('Spawning worker process: %d.', child.id);
});
this.workers.on('exit', (code, child) => {
this.logger.warning('Worker %d exited: %s.', child.id, code);
});
this.workers.on('log', (text, child) => {
this.logger.debug('Worker %d says:', child.id);
this.logger.debug(text);
});
this.workers.on('error', (err, child) => {
if (child) {
this.logger.error('Worker %d error: %s', child.id, err.message);
return;
}
this.emit('error', err);
});
this.hook('preopen', () => this.handlePreopen());
this.hook('preclose', () => this.handlePreclose());
this.hook('open', () => this.handleOpen());
this.hook('close', () => this.handleClose());
};
/**
* Ensure prefix directory.
* @returns {Promise}
*/
Node.prototype.ensure = function ensure() {
return this.config.ensure();
};
/**
* Create a file path using `prefix`.
* @param {String} file
* @returns {String}
*/
Node.prototype.location = function location(name) {
return this.config.location(name);
};
/**
* Open node. Bind all events.
* @private
*/
Node.prototype.handlePreopen = async function handlePreopen() {
await this.logger.open();
await this.workers.open();
this.bind(this.network.time, 'offset', (offset) => {
this.logger.info('Time offset: %d (%d minutes).', offset, offset / 60 | 0);
});
this.bind(this.network.time, 'sample', (sample, total) => {
this.logger.debug(
'Added time data: samples=%d, offset=%d (%d minutes).',
total, sample, sample / 60 | 0);
});
this.bind(this.network.time, 'mismatch', () => {
this.logger.warning('Adjusted time mismatch!');
this.logger.warning('Please make sure your system clock is correct!');
});
};
/**
* Open node.
* @private
*/
Node.prototype.handleOpen = async function handleOpen() {
this.startTime = util.now();
if (!this.workers.enabled) {
this.logger.warning('Warning: worker pool is disabled.');
this.logger.warning('Verification will be slow.');
}
};
/**
* Open node. Bind all events.
* @private
*/
Node.prototype.handlePreclose = async function handlePreclose() {
;
};
/**
* Close node. Unbind all events.
* @private
*/
Node.prototype.handleClose = async function handleClose() {
for (const [obj, event, listener] of this.bound)
obj.removeListener(event, listener);
this.bound.length = 0;
this.startTime = -1;
await this.workers.close();
await this.logger.close();
};
/**
* Bind to an event on `obj`, save listener for removal.
* @private
* @param {EventEmitter} obj
* @param {String} event
* @param {Function} listener
*/
Node.prototype.bind = function bind(obj, event, listener) {
this.bound.push([obj, event, listener]);
obj.on(event, listener);
};
/**
* Emit and log an error.
* @private
* @param {Error} err
*/
Node.prototype.error = function error(err) {
this.logger.error(err);
this.emit('error', err);
};
/**
* Get node uptime in seconds.
* @returns {Number}
*/
Node.prototype.uptime = function uptime() {
if (this.startTime === -1)
return 0;
return util.now() - this.startTime;
};
/**
* Attach a plugin.
* @param {Object} plugin
* @returns {Object} Plugin instance.
*/
Node.prototype.use = function use(plugin) {
assert(plugin, 'Plugin must be an object.');
assert(typeof plugin.init === 'function', '`init` must be a function.');
assert(!this.loaded, 'Cannot add plugin after node is loaded.');
const instance = plugin.init(this);
assert(!instance.open || typeof instance.open === 'function',
'`open` must be a function.');
assert(!instance.close || typeof instance.close === 'function',
'`close` must be a function.');
if (plugin.id) {
assert(typeof plugin.id === 'string', '`id` must be a string.');
// Reserved names
switch (plugin.id) {
case 'chain':
case 'fees':
case 'mempool':
case 'miner':
case 'pool':
case 'rpc':
case 'http':
assert(false, `${plugin.id} is already added.`);
break;
}
assert(!this.plugins[plugin.id], `${plugin.id} is already added.`);
this.plugins[plugin.id] = instance;
}
this.stack.push(instance);
if (typeof instance.on === 'function')
instance.on('error', err => this.error(err));
return instance;
};
/**
* Test whether a plugin is available.
* @param {String} name
* @returns {Boolean}
*/
Node.prototype.has = function has(name) {
return this.plugins[name] != null;
};
/**
* Get a plugin.
* @param {String} name
* @returns {Object|null}
*/
Node.prototype.get = function get(name) {
assert(typeof name === 'string', 'Plugin name must be a string.');
// Reserved names.
switch (name) {
case 'chain':
assert(this.chain, 'chain is not loaded.');
return this.chain;
case 'fees':
assert(this.fees, 'fees is not loaded.');
return this.fees;
case 'mempool':
assert(this.mempool, 'mempool is not loaded.');
return this.mempool;
case 'miner':
assert(this.miner, 'miner is not loaded.');
return this.miner;
case 'pool':
assert(this.pool, 'pool is not loaded.');
return this.pool;
case 'rpc':
assert(this.rpc, 'rpc is not loaded.');
return this.rpc;
case 'http':
assert(this.http, 'http is not loaded.');
return this.http;
}
return this.plugins[name] || null;
};
/**
* Require a plugin.
* @param {String} name
* @returns {Object}
* @throws {Error} on onloaded plugin
*/
Node.prototype.require = function require(name) {
const plugin = this.get(name);
assert(plugin, `${name} is not loaded.`);
return plugin;
};
/**
* Load plugins.
* @private
*/
Node.prototype.loadPlugins = function loadPlugins() {
const plugins = this.config.array('plugins', []);
const loader = this.config.func('loader');
for (let plugin of plugins) {
if (typeof plugin === 'string') {
assert(loader, 'Must pass a loader function.');
plugin = loader(plugin);
}
this.use(plugin);
}
};
/**
* Open plugins.
* @private
*/
Node.prototype.openPlugins = async function openPlugins() {
for (const plugin of this.stack) {
if (plugin.open)
await plugin.open();
}
};
/**
* Close plugins.
* @private
*/
Node.prototype.closePlugins = async function closePlugins() {
for (const plugin of this.stack) {
if (plugin.close)
await plugin.close();
}
};
/*
* Expose
*/
module.exports = Node;