443 lines
9.1 KiB
JavaScript
443 lines
9.1 KiB
JavaScript
/*!
|
|
* network.js - network 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 util = require('../utils/util');
|
|
const networks = require('./networks');
|
|
const consensus = require('./consensus');
|
|
const TimeData = require('./timedata');
|
|
|
|
/**
|
|
* Represents a network.
|
|
* @alias module:protocol.Network
|
|
* @constructor
|
|
* @param {Object|NetworkType} options - See {@link module:network}.
|
|
*/
|
|
|
|
function Network(options) {
|
|
if (!(this instanceof Network))
|
|
return new Network(options);
|
|
|
|
assert(!Network[options.type], 'Cannot create two networks.');
|
|
|
|
this.type = options.type;
|
|
this.seeds = options.seeds;
|
|
this.magic = options.magic;
|
|
this.port = options.port;
|
|
this.checkpointMap = options.checkpointMap;
|
|
this.lastCheckpoint = options.lastCheckpoint;
|
|
this.checkpoints = [];
|
|
this.halvingInterval = options.halvingInterval;
|
|
this.genesis = options.genesis;
|
|
this.genesisBlock = options.genesisBlock;
|
|
this.pow = options.pow;
|
|
this.block = options.block;
|
|
this.bip30 = options.bip30;
|
|
this.activationThreshold = options.activationThreshold;
|
|
this.minerWindow = options.minerWindow;
|
|
this.deployments = options.deployments;
|
|
this.deploys = options.deploys;
|
|
this.unknownBits = ~consensus.VERSION_TOP_MASK;
|
|
this.keyPrefix = options.keyPrefix;
|
|
this.addressPrefix = options.addressPrefix;
|
|
this.requireStandard = options.requireStandard;
|
|
this.rpcPort = options.rpcPort;
|
|
this.minRelay = options.minRelay;
|
|
this.feeRate = options.feeRate;
|
|
this.maxFeeRate = options.maxFeeRate;
|
|
this.selfConnect = options.selfConnect;
|
|
this.requestMempool = options.requestMempool;
|
|
this.time = new TimeData();
|
|
|
|
this._init();
|
|
}
|
|
|
|
/**
|
|
* Default network.
|
|
* @type {Network}
|
|
*/
|
|
|
|
Network.primary = null;
|
|
|
|
/**
|
|
* Default network type.
|
|
* @type {String}
|
|
*/
|
|
|
|
Network.type = null;
|
|
|
|
/*
|
|
* Networks (to avoid hash table mode).
|
|
*/
|
|
|
|
Network.main = null;
|
|
Network.testnet = null;
|
|
Network.regtest = null;
|
|
Network.segnet4 = null;
|
|
Network.simnet = null;
|
|
|
|
/**
|
|
* Get a deployment by bit index.
|
|
* @param {Number} bit
|
|
* @returns {Object}
|
|
*/
|
|
|
|
Network.prototype._init = function _init() {
|
|
let bits = 0;
|
|
let keys;
|
|
|
|
for (let deployment of this.deploys)
|
|
bits |= 1 << deployment.bit;
|
|
|
|
bits |= consensus.VERSION_TOP_MASK;
|
|
|
|
this.unknownBits = ~bits;
|
|
|
|
keys = Object.keys(this.checkpointMap);
|
|
|
|
for (let key of keys) {
|
|
let hash = this.checkpointMap[key];
|
|
let height = +key;
|
|
|
|
this.checkpoints.push({ hash: hash, height: height });
|
|
}
|
|
|
|
this.checkpoints.sort(cmpNode);
|
|
};
|
|
|
|
/**
|
|
* Get a deployment by bit index.
|
|
* @param {Number} bit
|
|
* @returns {Object}
|
|
*/
|
|
|
|
Network.prototype.byBit = function byBit(bit) {
|
|
let index = util.binarySearch(this.deploys, bit, cmpBit);
|
|
if (index === -1)
|
|
return null;
|
|
return this.deploys[index];
|
|
};
|
|
|
|
/**
|
|
* Get network adjusted time.
|
|
* @returns {Number}
|
|
*/
|
|
|
|
Network.prototype.now = function now() {
|
|
return this.time.now();
|
|
};
|
|
|
|
/**
|
|
* Get network adjusted time in milliseconds.
|
|
* @returns {Number}
|
|
*/
|
|
|
|
Network.prototype.ms = function ms() {
|
|
return this.time.ms();
|
|
};
|
|
|
|
/**
|
|
* Create a network. Get existing network if possible.
|
|
* @param {NetworkType|Object} options
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.create = function create(options) {
|
|
let network;
|
|
|
|
if (typeof options === 'string')
|
|
options = networks[options];
|
|
|
|
assert(options, 'Unknown network.');
|
|
|
|
if (Network[options.type])
|
|
return Network[options.type];
|
|
|
|
network = new Network(options);
|
|
|
|
Network[network.type] = network;
|
|
|
|
if (!Network.primary)
|
|
Network.primary = network;
|
|
|
|
return network;
|
|
};
|
|
|
|
/**
|
|
* Set the default network. This network will be used
|
|
* if nothing is passed as the `network` option for
|
|
* certain objects.
|
|
* @param {NetworkType} type - Network type.
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.set = function set(type) {
|
|
assert(typeof type === 'string', 'Bad network.');
|
|
Network.primary = Network.get(type);
|
|
Network.type = type;
|
|
return Network.primary;
|
|
};
|
|
|
|
/**
|
|
* Get a network with a string or a Network object.
|
|
* @param {NetworkType|Network} type - Network type.
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.get = function get(type) {
|
|
if (!type) {
|
|
assert(Network.primary, 'No default network.');
|
|
return Network.primary;
|
|
}
|
|
|
|
if (type instanceof Network)
|
|
return type;
|
|
|
|
if (typeof type === 'string')
|
|
return Network.create(type);
|
|
|
|
assert(false, 'Unknown network.');
|
|
};
|
|
|
|
/**
|
|
* Get a network with a string or a Network object.
|
|
* @param {NetworkType|Network} type - Network type.
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.ensure = function ensure(type) {
|
|
if (!type) {
|
|
assert(Network.primary, 'No default network.');
|
|
return Network.primary;
|
|
}
|
|
|
|
if (type instanceof Network)
|
|
return type;
|
|
|
|
if (typeof type === 'string') {
|
|
if (networks[type])
|
|
return Network.create(type);
|
|
}
|
|
|
|
assert(Network.primary, 'No default network.');
|
|
|
|
return Network.primary;
|
|
};
|
|
|
|
/**
|
|
* Get a network by an associated comparator.
|
|
* @private
|
|
* @param {Object} value
|
|
* @param {Function} compare
|
|
* @param {Network|null} network
|
|
* @param {String} name
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.by = function by(value, compare, network, name) {
|
|
if (network) {
|
|
network = Network.get(network);
|
|
if (compare(network, value))
|
|
return network;
|
|
throw new Error(`Network mismatch for ${name}.`);
|
|
}
|
|
|
|
for (let type of networks.types) {
|
|
network = networks[type];
|
|
if (compare(network, value))
|
|
return Network.get(type);
|
|
}
|
|
|
|
throw new Error(`Network not found for ${name}.`);
|
|
};
|
|
|
|
/**
|
|
* Get a network by its magic number.
|
|
* @param {Number} value
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromMagic = function fromMagic(value, network) {
|
|
return Network.by(value, cmpMagic, network, 'magic number');
|
|
};
|
|
|
|
/**
|
|
* Get a network by its WIF prefix.
|
|
* @param {Number} value
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromWIF = function fromWIF(prefix, network) {
|
|
return Network.by(prefix, cmpWIF, network, 'WIF');
|
|
};
|
|
|
|
/**
|
|
* Get a network by its xpubkey prefix.
|
|
* @param {Number} value
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromPublic = function fromPublic(prefix, network) {
|
|
return Network.by(prefix, cmpPub, network, 'xpubkey');
|
|
};
|
|
|
|
/**
|
|
* Get a network by its xprivkey prefix.
|
|
* @param {Number} value
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromPrivate = function fromPrivate(prefix, network) {
|
|
return Network.by(prefix, cmpPriv, network, 'xprivkey');
|
|
};
|
|
|
|
/**
|
|
* Get a network by its xpubkey base58 prefix.
|
|
* @param {String} prefix
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromPublic58 = function fromPublic58(prefix, network) {
|
|
return Network.by(prefix, cmpPub58, network, 'xpubkey');
|
|
};
|
|
|
|
/**
|
|
* Get a network by its xprivkey base58 prefix.
|
|
* @param {String} prefix
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromPrivate58 = function fromPrivate58(prefix, network) {
|
|
return Network.by(prefix, cmpPriv58, network, 'xprivkey');
|
|
};
|
|
|
|
/**
|
|
* Get a network by its base58 address prefix.
|
|
* @param {Number} value
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromAddress = function fromAddress(prefix, network) {
|
|
return Network.by(prefix, cmpAddress, network, 'base58 address');
|
|
};
|
|
|
|
/**
|
|
* Get a network by its bech32 address prefix.
|
|
* @param {String} hrp
|
|
* @param {Network?} network
|
|
* @returns {Network}
|
|
*/
|
|
|
|
Network.fromBech32 = function fromBech32(hrp, network) {
|
|
return Network.by(hrp, cmpBech32, network, 'bech32 address');
|
|
};
|
|
|
|
/**
|
|
* Convert the network to a string.
|
|
* @returns {String}
|
|
*/
|
|
|
|
Network.prototype.toString = function toString() {
|
|
return this.type;
|
|
};
|
|
|
|
/**
|
|
* Inspect the network.
|
|
* @returns {String}
|
|
*/
|
|
|
|
Network.prototype.inspect = function inspect() {
|
|
return `<Network: ${this.type}>`;
|
|
};
|
|
|
|
/**
|
|
* Test an object to see if it is a Network.
|
|
* @param {Object} obj
|
|
* @returns {Boolean}
|
|
*/
|
|
|
|
Network.isNetwork = function isNetwork(obj) {
|
|
return obj
|
|
&& typeof obj.genesisBlock === 'string'
|
|
&& typeof obj.pow === 'object';
|
|
};
|
|
|
|
/*
|
|
* Set initial network.
|
|
*/
|
|
|
|
Network.set(process.env.BCOIN_NETWORK || 'main');
|
|
|
|
/*
|
|
* Helpers
|
|
*/
|
|
|
|
function cmpBit(a, b) {
|
|
return a.bit - b;
|
|
}
|
|
|
|
function cmpNode(a, b) {
|
|
return a.height - b.height;
|
|
}
|
|
|
|
function cmpMagic(network, magic) {
|
|
return network.magic === magic;
|
|
}
|
|
|
|
function cmpWIF(network, prefix) {
|
|
return network.keyPrefix.privkey === prefix;
|
|
}
|
|
|
|
function cmpPub(network, prefix) {
|
|
return network.keyPrefix.xpubkey === prefix;
|
|
}
|
|
|
|
function cmpPriv(network, prefix) {
|
|
return network.keyPrefix.xprivkey === prefix;
|
|
}
|
|
|
|
function cmpPub58(network, prefix) {
|
|
return network.keyPrefix.xpubkey58 === prefix;
|
|
}
|
|
|
|
function cmpPriv58(network, prefix) {
|
|
return network.keyPrefix.xprivkey58 === prefix;
|
|
}
|
|
|
|
function cmpAddress(network, prefix) {
|
|
let prefixes = network.addressPrefix;
|
|
|
|
switch (prefix) {
|
|
case prefixes.pubkeyhash:
|
|
case prefixes.scripthash:
|
|
case prefixes.witnesspubkeyhash:
|
|
case prefixes.witnessscripthash:
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function cmpBech32(network, hrp) {
|
|
return network.addressPrefix.bech32 === hrp;
|
|
}
|
|
|
|
/*
|
|
* Expose
|
|
*/
|
|
|
|
module.exports = Network;
|