/*! * chain.js - blockchain management 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 path = require('path'); const AsyncObject = require('../utils/asyncobject'); const Network = require('../protocol/network'); const Logger = require('../node/logger'); const ChainDB = require('./chaindb'); const common = require('./common'); const consensus = require('../protocol/consensus'); const util = require('../utils/util'); const Lock = require('../utils/lock'); const LRU = require('../utils/lru'); const ChainEntry = require('./chainentry'); const CoinView = require('../coins/coinview'); const Script = require('../script/script'); const {VerifyError} = require('../protocol/errors'); const co = require('../utils/co'); const thresholdStates = common.thresholdStates; /** * Represents a blockchain. * @alias module:blockchain.Chain * @constructor * @param {Object} options * @param {String?} options.name - Database name. * @param {String?} options.location - Database file location. * @param {String?} options.db - Database backend (`"leveldb"` by default). * @param {Number?} options.maxOrphans * @param {Boolean?} options.spv * @property {Boolean} loaded * @property {ChainDB} db - Note that Chain `options` will be passed * to the instantiated ChainDB. * @property {Lock} locker * @property {Object} invalid * @property {ChainEntry?} tip * @property {Number} height * @property {DeploymentState} state * @property {Object} orphan - Orphan map. * @emits Chain#open * @emits Chain#error * @emits Chain#block * @emits Chain#competitor * @emits Chain#resolved * @emits Chain#checkpoint * @emits Chain#fork * @emits Chain#reorganize * @emits Chain#invalid * @emits Chain#exists * @emits Chain#purge * @emits Chain#connect * @emits Chain#reconnect * @emits Chain#disconnect */ function Chain(options) { if (!(this instanceof Chain)) return new Chain(options); AsyncObject.call(this); this.options = new ChainOptions(options); this.network = this.options.network; this.logger = this.options.logger.context('chain'); this.workers = this.options.workers; this.locker = new Lock(true); this.invalid = new LRU(100); this.state = new DeploymentState(); this.tip = new ChainEntry(this); this.height = -1; this.synced = false; this.orphanMap = new Map(); this.orphanPrev = new Map(); this.db = new ChainDB(this); } Object.setPrototypeOf(Chain.prototype, AsyncObject.prototype); /** * Open the chain, wait for the database to load. * @method * @alias Chain#open * @returns {Promise} */ Chain.prototype._open = async function _open() { this.logger.info('Chain is loading.'); if (this.options.checkpoints) this.logger.info('Checkpoints are enabled.'); if (this.options.coinCache) this.logger.info('Coin cache is enabled.'); if (this.options.bip91) this.logger.warning('BIP91 enabled. Segsignal will be enforced.'); if (this.options.bip148) this.logger.warning('BIP148 enabled. UASF will be enforced.'); await this.db.open(); const tip = await this.db.getTip(); assert(tip); this.tip = tip; this.height = tip.height; this.logger.info('Chain Height: %d', tip.height); this.logger.memory(); const state = await this.getDeploymentState(); this.setDeploymentState(state); this.logger.memory(); this.emit('tip', tip); this.maybeSync(); }; /** * Close the chain, wait for the database to close. * @alias Chain#close * @returns {Promise} */ Chain.prototype._close = function _close() { return this.db.close(); }; /** * Perform all necessary contextual verification on a block. * @method * @private * @param {Block} block * @param {ChainEntry} prev * @param {Number} flags * @returns {Promise} - Returns {@link ContextResult}. */ Chain.prototype.verifyContext = async function verifyContext(block, prev, flags) { // Initial non-contextual verification. const state = await this.verify(block, prev, flags); // BIP30 - Verify there are no duplicate txids. await this.verifyDuplicates(block, prev, state); // Verify scripts, spend and add coins. const view = await this.verifyInputs(block, prev, state); return [view, state]; }; /** * Perform all necessary contextual verification * on a block, without POW check. * @method * @param {Block} block * @returns {Promise} */ Chain.prototype.verifyBlock = async function verifyBlock(block) { const unlock = await this.locker.lock(); try { return await this._verifyBlock(block); } finally { unlock(); } }; /** * Perform all necessary contextual verification * on a block, without POW check (no lock). * @method * @private * @param {Block} block * @returns {Promise} */ Chain.prototype._verifyBlock = async function _verifyBlock(block) { const flags = common.flags.DEFAULT_FLAGS & ~common.flags.VERIFY_POW; return await this.verifyContext(block, this.tip, flags); }; /** * Test whether a block is the genesis block. * @param {Block} block * @returns {Boolean} */ Chain.prototype.isGenesis = function isGenesis(block) { return block.hash('hex') === this.network.genesis.hash; }; /** * Contextual verification for a block, including * version deployments (IsSuperMajority), versionbits, * coinbase height, finality checks. * @method * @private * @param {Block} block * @param {ChainEntry} prev * @param {Number} flags * @returns {Promise} - Returns {@link DeploymentState}. */ Chain.prototype.verify = async function verify(block, prev, flags) { const deployments = this.network.deployments; const hash = block.hash('hex'); const now = this.network.now(); const height = prev.height + 1; assert(typeof flags === 'number'); // Extra sanity check. if (block.prevBlock !== prev.hash) throw new VerifyError(block, 'invalid', 'bad-prevblk', 0); // Verify a checkpoint if there is one. if (!this.verifyCheckpoint(prev, hash)) { throw new VerifyError(block, 'checkpoint', 'checkpoint mismatch', 100); } // Skip everything when using checkpoints. // We can do this safely because every // block in between each checkpoint was // validated outside in the header chain. if (prev.isHistorical()) { if (this.options.spv) return new DeploymentState(); // Once segwit is active, we will still // need to check for block mutability. if (!block.hasWitness() && !block.getCommitmentHash()) return new DeploymentState(); flags &= ~common.flags.VERIFY_BODY; } // Non-contextual checks. if (flags & common.flags.VERIFY_BODY) { const [valid, reason, score] = block.checkBody(); if (!valid) throw new VerifyError(block, 'invalid', reason, score, true); } // Skip all blocks in spv mode. if (this.options.spv) return this.state; // Ensure the POW is what we expect. const bits = await this.getTarget(block.time, prev); if (block.bits !== bits) { throw new VerifyError(block, 'invalid', 'bad-diffbits', 100); } // Ensure the timestamp is correct. const mtp = await prev.getMedianTime(); if (block.time <= mtp) { throw new VerifyError(block, 'invalid', 'time-too-old', 0); } // Check timestamp against adj-time+2hours. // If this fails we may be able to accept // the block later. if (block.time > now + 2 * 60 * 60) { throw new VerifyError(block, 'invalid', 'time-too-new', 0, true); } // Only allow version 2 blocks (coinbase height) // once the majority of blocks are using it. if (block.version < 2 && height >= this.network.block.bip34height) throw new VerifyError(block, 'obsolete', 'bad-version', 0); // Only allow version 3 blocks (sig validation) // once the majority of blocks are using it. if (block.version < 3 && height >= this.network.block.bip66height) throw new VerifyError(block, 'obsolete', 'bad-version', 0); // Only allow version 4 blocks (checklocktimeverify) // once the majority of blocks are using it. if (block.version < 4 && height >= this.network.block.bip65height) throw new VerifyError(block, 'obsolete', 'bad-version', 0); // Get the new deployment state. const state = await this.getDeployments(block.time, prev); // Enforce BIP91/BIP148. if (state.hasBIP91() || state.hasBIP148()) { if (!consensus.hasBit(block.version, deployments.segwit.bit)) throw new VerifyError(block, 'invalid', 'bad-no-segwit', 0); } // Get timestamp for tx.isFinal(). const time = state.hasMTP() ? mtp : block.time; // Transactions must be finalized with // regards to nSequence and nLockTime. for (const tx of block.txs) { if (!tx.isFinal(height, time)) { throw new VerifyError(block, 'invalid', 'bad-txns-nonfinal', 10); } } // Make sure the height contained // in the coinbase is correct. if (state.hasBIP34()) { if (block.getCoinbaseHeight() !== height) { throw new VerifyError(block, 'invalid', 'bad-cb-height', 100); } } // Check the commitment hash for segwit. let commit; if (state.hasWitness()) { commit = block.getCommitmentHash(); if (commit) { // These are totally malleable. Someone // may have even accidentally sent us // the non-witness version of the block. // We don't want to consider this block // "invalid" if either of these checks // fail. if (!block.getWitnessNonce()) { throw new VerifyError(block, 'invalid', 'bad-witness-nonce-size', 100, true); } if (!commit.equals(block.createCommitmentHash())) { throw new VerifyError(block, 'invalid', 'bad-witness-merkle-match', 100, true); } } } // Blocks that do not commit to // witness data cannot contain it. if (!commit) { if (block.hasWitness()) { throw new VerifyError(block, 'invalid', 'unexpected-witness', 100, true); } } // Check block weight (different from block size // check in non-contextual verification). if (block.getWeight() > consensus.MAX_BLOCK_WEIGHT) { throw new VerifyError(block, 'invalid', 'bad-blk-weight', 100); } return state; }; /** * Check all deployments on a chain, ranging from p2sh to segwit. * @method * @param {Number} time * @param {ChainEntry} prev * @returns {Promise} - Returns {@link DeploymentState}. */ Chain.prototype.getDeployments = async function getDeployments(time, prev) { const deployments = this.network.deployments; const height = prev.height + 1; const state = new DeploymentState(); // For some reason bitcoind has p2sh in the // mandatory flags by default, when in reality // it wasn't activated until march 30th 2012. // The first p2sh output and redeem script // appeared on march 7th 2012, only it did // not have a signature. See: // 6a26d2ecb67f27d1fa5524763b49029d7106e91e3cc05743073461a719776192 // 9c08a4d78931342b37fd5f72900fb9983087e6f46c4a097d8a1f52c74e28eaf6 if (time >= consensus.BIP16_TIME) state.flags |= Script.flags.VERIFY_P2SH; // Coinbase heights are now enforced (bip34). if (height >= this.network.block.bip34height) state.bip34 = true; // Signature validation is now enforced (bip66). if (height >= this.network.block.bip66height) state.flags |= Script.flags.VERIFY_DERSIG; // CHECKLOCKTIMEVERIFY is now usable (bip65). if (height >= this.network.block.bip65height) state.flags |= Script.flags.VERIFY_CHECKLOCKTIMEVERIFY; // CHECKSEQUENCEVERIFY and median time // past locktimes are now usable (bip9 & bip113). if (await this.isActive(prev, deployments.csv)) { state.flags |= Script.flags.VERIFY_CHECKSEQUENCEVERIFY; state.lockFlags |= common.lockFlags.VERIFY_SEQUENCE; state.lockFlags |= common.lockFlags.MEDIAN_TIME_PAST; } // Check the state of the segwit deployment. const witness = await this.getState(prev, deployments.segwit); // Segregrated witness (bip141) is now usable // along with SCRIPT_VERIFY_NULLDUMMY (bip147). if (witness === thresholdStates.ACTIVE) { state.flags |= Script.flags.VERIFY_WITNESS; state.flags |= Script.flags.VERIFY_NULLDUMMY; } // Segsignal is now enforced (bip91). if (this.options.bip91) { if (witness === thresholdStates.STARTED) { if (await this.isActive(prev, deployments.segsignal)) state.bip91 = true; } } // UASF is now enforced (bip148) (mainnet-only). if (this.options.bip148 && this.network === Network.main) { if (witness !== thresholdStates.LOCKED_IN && witness !== thresholdStates.ACTIVE) { // The BIP148 MTP check is nonsensical in // that it includes the _current_ entry's // timestamp. This requires some hackery, // since bcoin only operates on the sane // assumption that deployment checks should // only ever examine the values of the // previous block (necessary for mining). const mtp = await prev.getMedianTime(time); if (mtp >= 1501545600 && mtp <= 1510704000) state.bip148 = true; } } return state; }; /** * Set a new deployment state. * @param {DeploymentState} state */ Chain.prototype.setDeploymentState = function setDeploymentState(state) { if (this.options.checkpoints && this.height < this.network.lastCheckpoint) { this.state = state; return; } if (!this.state.hasP2SH() && state.hasP2SH()) this.logger.warning('P2SH has been activated.'); if (!this.state.hasBIP34() && state.hasBIP34()) this.logger.warning('BIP34 has been activated.'); if (!this.state.hasBIP66() && state.hasBIP66()) this.logger.warning('BIP66 has been activated.'); if (!this.state.hasCLTV() && state.hasCLTV()) this.logger.warning('BIP65 has been activated.'); if (!this.state.hasCSV() && state.hasCSV()) this.logger.warning('CSV has been activated.'); if (!this.state.hasWitness() && state.hasWitness()) this.logger.warning('Segwit has been activated.'); if (!this.state.hasBIP91() && state.hasBIP91()) this.logger.warning('BIP91 has been activated.'); if (!this.state.hasBIP148() && state.hasBIP148()) this.logger.warning('BIP148 has been activated.'); this.state = state; }; /** * Determine whether to check block for duplicate txids in blockchain * history (BIP30). If we're on a chain that has bip34 activated, we * can skip this. * @method * @private * @see https://github.com/bitcoin/bips/blob/master/bip-0030.mediawiki * @param {Block} block * @param {ChainEntry} prev * @returns {Promise} */ Chain.prototype.verifyDuplicates = async function verifyDuplicates(block, prev, state) { if (this.options.spv) return; if (prev.isHistorical()) return; // BIP34 made it impossible to // create duplicate txids. if (state.hasBIP34()) return; // Check all transactions. for (const tx of block.txs) { const result = await this.db.hasCoins(tx); if (result) { const height = prev.height + 1; // Blocks 91842 and 91880 created duplicate // txids by using the same exact output script // and extraNonce. if (this.network.bip30[height]) { if (block.hash('hex') === this.network.bip30[height]) continue; } throw new VerifyError(block, 'invalid', 'bad-txns-BIP30', 100); } } }; /** * Check block transactions for all things pertaining * to inputs. This function is important because it is * what actually fills the coins into the block. This * function will check the block reward, the sigops, * the tx values, and execute and verify the scripts (it * will attempt to do this on the worker pool). If * `checkpoints` is enabled, it will skip verification * for historical data. * @method * @private * @see TX#verifyInputs * @see TX#verify * @param {Block} block * @param {ChainEntry} prev * @param {DeploymentState} state * @returns {Promise} - Returns {@link CoinView}. */ Chain.prototype.verifyInputs = async function verifyInputs(block, prev, state) { const view = new CoinView(); if (this.options.spv) return view; const interval = this.network.halvingInterval; const height = prev.height + 1; const historical = prev.isHistorical(); let sigops = 0; let reward = 0; // Check all transactions for (let i = 0; i < block.txs.length; i++) { const tx = block.txs[i]; // Ensure tx is not double spending an output. if (i > 0) { if (!await view.spendInputs(this.db, tx)) { assert(!historical, 'BUG: Spent inputs in historical data!'); throw new VerifyError(block, 'invalid', 'bad-txns-inputs-missingorspent', 100); } } // Skip everything if we're // using checkpoints. if (historical) { view.addTX(tx, height); continue; } // Verify sequence locks. if (i > 0 && tx.version >= 2) { const valid = await this.verifyLocks(prev, tx, view, state.lockFlags); if (!valid) { throw new VerifyError(block, 'invalid', 'bad-txns-nonfinal', 100); } } // Count sigops (legacy + scripthash? + witness?) sigops += tx.getSigopsCost(view, state.flags); if (sigops > consensus.MAX_BLOCK_SIGOPS_COST) { throw new VerifyError(block, 'invalid', 'bad-blk-sigops', 100); } // Contextual sanity checks. if (i > 0) { const [fee, reason, score] = tx.checkInputs(view, height); if (fee === -1) { throw new VerifyError(block, 'invalid', reason, score); } reward += fee; if (reward > consensus.MAX_MONEY) { throw new VerifyError(block, 'invalid', 'bad-cb-amount', 100); } } // Add new coins. view.addTX(tx, height); } // Skip script verification. if (historical) return view; // Make sure the miner isn't trying to conjure more coins. reward += consensus.getReward(height, interval); if (block.getClaimed() > reward) { throw new VerifyError(block, 'invalid', 'bad-cb-amount', 100); } // Push onto verification queue. const jobs = []; for (let i = 1; i < block.txs.length; i++) { const tx = block.txs[i]; jobs.push(tx.verifyAsync(view, state.flags, this.workers)); } // Verify all txs in parallel. if (!await co.every(jobs)) { throw new VerifyError(block, 'invalid', 'mandatory-script-verify-flag-failed', 100); } return view; }; /** * Get the cached height for a hash if present. * @private * @param {Hash} hash * @returns {Number} */ Chain.prototype.checkHeight = function checkHeight(hash) { const entry = this.db.getCache(hash); if (!entry) return -1; return entry.height; }; /** * Find the block at which a fork ocurred. * @private * @method * @param {ChainEntry} fork - The current chain. * @param {ChainEntry} longer - The competing chain. * @returns {Promise} */ Chain.prototype.findFork = async function findFork(fork, longer) { while (fork.hash !== longer.hash) { while (longer.height > fork.height) { longer = await longer.getPrevious(); if (!longer) throw new Error('No previous entry for new tip.'); } if (fork.hash === longer.hash) return fork; fork = await fork.getPrevious(); if (!fork) throw new Error('No previous entry for old tip.'); } return fork; }; /** * Reorganize the blockchain (connect and disconnect inputs). * Called when a competing chain with a higher chainwork * is received. * @method * @private * @param {ChainEntry} competitor - The competing chain's tip. * @returns {Promise} */ Chain.prototype.reorganize = async function reorganize(competitor) { const tip = this.tip; const fork = await this.findFork(tip, competitor); assert(fork, 'No free space or data corruption.'); // Blocks to disconnect. const disconnect = []; let entry = tip; while (entry.hash !== fork.hash) { disconnect.push(entry); entry = await entry.getPrevious(); assert(entry); } // Blocks to connect. const connect = []; entry = competitor; while (entry.hash !== fork.hash) { connect.push(entry); entry = await entry.getPrevious(); assert(entry); } // Disconnect blocks/txs. for (let i = 0; i < disconnect.length; i++) { const entry = disconnect[i]; await this.disconnect(entry); } // Connect blocks/txs. // We don't want to connect the new tip here. // That will be done outside in setBestChain. for (let i = connect.length - 1; i >= 1; i--) { const entry = connect[i]; await this.reconnect(entry); } this.logger.warning( 'Chain reorganization: old=%s(%d) new=%s(%d)', tip.rhash(), tip.height, competitor.rhash(), competitor.height ); this.emit('reorganize', tip, competitor); }; /** * Reorganize the blockchain for SPV. This * will reset the chain to the fork block. * @method * @private * @param {ChainEntry} competitor - The competing chain's tip. * @returns {Promise} */ Chain.prototype.reorganizeSPV = async function reorganizeSPV(competitor) { const tip = this.tip; const fork = await this.findFork(tip, competitor); assert(fork, 'No free space or data corruption.'); // Buffer disconnected blocks. const disconnect = []; let entry = tip; while (entry.hash !== fork.hash) { disconnect.push(entry); entry = await entry.getPrevious(); assert(entry); } // Reset the main chain back // to the fork block, causing // us to redownload the blocks // on the new main chain. await this._reset(fork.hash, true); // Emit disconnection events now that // the chain has successfully reset. for (const entry of disconnect) { const headers = entry.toHeaders(); const view = new CoinView(); await this.fire('disconnect', entry, headers, view); } this.logger.warning( 'SPV reorganization: old=%s(%d) new=%s(%d)', tip.rhash(), tip.height, competitor.rhash(), competitor.height ); this.logger.warning( 'Chain replay from height %d necessary.', fork.height); this.emit('reorganize', tip, competitor); }; /** * Disconnect an entry from the chain (updates the tip). * @method * @param {ChainEntry} entry * @returns {Promise} */ Chain.prototype.disconnect = async function disconnect(entry) { let block = await this.db.getBlock(entry.hash); if (!block) { if (!this.options.spv) throw new Error('Block not found.'); block = entry.toHeaders(); } const prev = await entry.getPrevious(); const view = await this.db.disconnect(entry, block); assert(prev); this.tip = prev; this.height = prev.height; this.emit('tip', prev); await this.fire('disconnect', entry, block, view); }; /** * Reconnect an entry to the chain (updates the tip). * This will do contextual-verification on the block * (necessary because we cannot validate the inputs * in alternate chains when they come in). * @method * @param {ChainEntry} entry * @param {Number} flags * @returns {Promise} */ Chain.prototype.reconnect = async function reconnect(entry) { const flags = common.flags.VERIFY_NONE; let block = await this.db.getBlock(entry.hash); if (!block) { if (!this.options.spv) throw new Error('Block not found.'); block = entry.toHeaders(); } const prev = await entry.getPrevious(); assert(prev); let view, state; try { [view, state] = await this.verifyContext(block, prev, flags); } catch (err) { if (err.type === 'VerifyError') { if (!err.malleated) this.setInvalid(entry.hash); this.logger.warning( 'Tried to reconnect invalid block: %s (%d).', entry.rhash(), entry.height); } throw err; } await this.db.reconnect(entry, block, view); this.tip = entry; this.height = entry.height; this.setDeploymentState(state); this.emit('tip', entry); this.emit('reconnect', entry, block); await this.fire('connect', entry, block, view); }; /** * Set the best chain. This is called on every valid block * that comes in. It may add and connect the block (main chain), * save the block without connection (alternate chain), or * reorganize the chain (a higher fork). * @method * @private * @param {ChainEntry} entry * @param {Block} block * @param {ChainEntry} prev * @param {Number} flags * @returns {Promise} */ Chain.prototype.setBestChain = async function setBestChain(entry, block, prev, flags) { // A higher fork has arrived. // Time to reorganize the chain. if (entry.prevBlock !== this.tip.hash) { this.logger.warning('WARNING: Reorganizing chain.'); // In spv-mode, we reset the // chain and redownload the blocks. if (this.options.spv) { await this.reorganizeSPV(entry); return; } await this.reorganize(entry); } // Warn of unknown versionbits. if (entry.hasUnknown()) { this.logger.warning( 'Unknown version bits in block %d: %s.', entry.height, util.hex32(entry.version)); } // Otherwise, everything is in order. // Do "contextual" verification on our block // now that we're certain its previous // block is in the chain. let view, state; try { [view, state] = await this.verifyContext(block, prev, flags); } catch (err) { if (err.type === 'VerifyError') { if (!err.malleated) this.setInvalid(entry.hash); this.logger.warning( 'Tried to connect invalid block: %s (%d).', entry.rhash(), entry.height); } throw err; } // Save block and connect inputs. await this.db.save(entry, block, view); // Expose the new state. this.tip = entry; this.height = entry.height; this.setDeploymentState(state); this.emit('tip', entry); this.emit('block', block, entry); await this.fire('connect', entry, block, view); }; /** * Save block on an alternate chain. * @method * @private * @param {ChainEntry} entry * @param {Block} block * @param {ChainEntry} prev * @param {Number} flags * @returns {Promise} */ Chain.prototype.saveAlternate = async function saveAlternate(entry, block, prev, flags) { try { // Do as much verification // as we can before saving. await this.verify(block, prev, flags); } catch (err) { if (err.type === 'VerifyError') { if (!err.malleated) this.setInvalid(entry.hash); this.logger.warning( 'Invalid block on alternate chain: %s (%d).', entry.rhash(), entry.height); } throw err; } // Warn of unknown versionbits. if (entry.hasUnknown()) { this.logger.warning( 'Unknown version bits in block %d: %s.', entry.height, util.hex32(entry.version)); } await this.db.save(entry, block); this.logger.warning('Heads up: Competing chain at height %d:' + ' tip-height=%d competitor-height=%d' + ' tip-hash=%s competitor-hash=%s' + ' tip-chainwork=%s competitor-chainwork=%s' + ' chainwork-diff=%s', entry.height, this.tip.height, entry.height, this.tip.rhash(), entry.rhash(), this.tip.chainwork.toString(), entry.chainwork.toString(), this.tip.chainwork.sub(entry.chainwork).toString()); // Emit as a "competitor" block. this.emit('competitor', block, entry); }; /** * Reset the chain to the desired block. This * is useful for replaying the blockchain download * for SPV. * @method * @param {Hash|Number} block * @returns {Promise} */ Chain.prototype.reset = async function reset(block) { const unlock = await this.locker.lock(); try { return await this._reset(block, false); } finally { unlock(); } }; /** * Reset the chain to the desired block without a lock. * @method * @private * @param {Hash|Number} block * @returns {Promise} */ Chain.prototype._reset = async function _reset(block, silent) { const tip = await this.db.reset(block); // Reset state. this.tip = tip; this.height = tip.height; this.synced = false; const state = await this.getDeploymentState(); this.setDeploymentState(state); this.emit('tip', tip); if (!silent) await this.fire('reset', tip); // Reset the orphan map completely. There may // have been some orphans on a forked chain we // no longer need. this.purgeOrphans(); this.maybeSync(); }; /** * Reset the chain to a height or hash. Useful for replaying * the blockchain download for SPV. * @method * @param {Hash|Number} block - hash/height * @returns {Promise} */ Chain.prototype.replay = async function replay(block) { const unlock = await this.locker.lock(); try { return await this._replay(block, true); } finally { unlock(); } }; /** * Reset the chain without a lock. * @method * @private * @param {Hash|Number} block - hash/height * @param {Boolean?} silent * @returns {Promise} */ Chain.prototype._replay = async function _replay(block, silent) { const entry = await this.db.getEntry(block); if (!entry) throw new Error('Block not found.'); if (!await entry.isMainChain()) throw new Error('Cannot reset on alternate chain.'); if (entry.isGenesis()) { await this._reset(entry.hash, silent); return; } await this._reset(entry.prevBlock, silent); }; /** * Invalidate block. * @method * @param {Hash} hash * @returns {Promise} */ Chain.prototype.invalidate = async function invalidate(hash) { const unlock = await this.locker.lock(); try { return await this._invalidate(hash); } finally { unlock(); } }; /** * Invalidate block (no lock). * @method * @param {Hash} hash * @returns {Promise} */ Chain.prototype._invalidate = async function _invalidate(hash) { await this._replay(hash, false); this.setInvalid(hash); }; /** * Retroactively prune the database. * @method * @returns {Promise} */ Chain.prototype.prune = async function prune() { const unlock = await this.locker.lock(); try { return await this.db.prune(this.tip.hash); } finally { unlock(); } }; /** * Scan the blockchain for transactions containing specified address hashes. * @method * @param {Hash} start - Block hash to start at. * @param {Bloom} filter - Bloom filter containing tx and address hashes. * @param {Function} iter - Iterator. * @returns {Promise} */ Chain.prototype.scan = async function scan(start, filter, iter) { const unlock = await this.locker.lock(); try { return await this.db.scan(start, filter, iter); } finally { unlock(); } }; /** * Add a block to the chain, perform all necessary verification. * @method * @param {Block} block * @param {Number?} flags * @param {Number?} id * @returns {Promise} */ Chain.prototype.add = async function add(block, flags, id) { const hash = block.hash('hex'); const unlock = await this.locker.lock(hash); try { return await this._add(block, flags, id); } finally { unlock(); } }; /** * Add a block to the chain without a lock. * @method * @private * @param {Block} block * @param {Number?} flags * @param {Number?} id * @returns {Promise} */ Chain.prototype._add = async function _add(block, flags, id) { const hash = block.hash('hex'); if (flags == null) flags = common.flags.DEFAULT_FLAGS; if (id == null) id = -1; // Special case for genesis block. if (hash === this.network.genesis.hash) { this.logger.debug('Saw genesis block: %s.', block.rhash()); throw new VerifyError(block, 'duplicate', 'duplicate', 0); } // Do we already have this block in the queue? if (this.hasPending(hash)) { this.logger.debug('Already have pending block: %s.', block.rhash()); throw new VerifyError(block, 'duplicate', 'duplicate', 0); } // If the block is already known to be // an orphan, ignore it. if (this.hasOrphan(hash)) { this.logger.debug('Already have orphan block: %s.', block.rhash()); throw new VerifyError(block, 'duplicate', 'duplicate', 0); } // Do not revalidate known invalid blocks. if (this.hasInvalid(block)) { this.logger.debug('Invalid ancestors for block: %s.', block.rhash()); throw new VerifyError(block, 'duplicate', 'duplicate', 100); } // Check the POW before doing anything. if (flags & common.flags.VERIFY_POW) { if (!block.verifyPOW()) throw new VerifyError(block, 'invalid', 'high-hash', 50); } // Do we already have this block? if (await this.db.hasEntry(hash)) { this.logger.debug('Already have block: %s.', block.rhash()); throw new VerifyError(block, 'duplicate', 'duplicate', 0); } // Find the previous block entry. const prev = await this.db.getEntry(block.prevBlock); // If previous block wasn't ever seen, // add it current to orphans and return. if (!prev) { this.storeOrphan(block, flags, id); return null; } // Connect the block. const entry = await this.connect(prev, block, flags); // Handle any orphans. if (this.hasNextOrphan(hash)) await this.handleOrphans(entry); return entry; }; /** * Connect block to chain. * @method * @private * @param {ChainEntry} prev * @param {Block} block * @param {Number} flags * @returns {Promise} */ Chain.prototype.connect = async function connect(prev, block, flags) { const start = util.hrtime(); // Sanity check. assert(block.prevBlock === prev.hash); // Explanation: we try to keep as much data // off the javascript heap as possible. Blocks // in the future may be 8mb or 20mb, who knows. // In fullnode-mode we store the blocks in // "compact" form (the headers plus the raw // Buffer object) until they're ready to be // fully validated here. They are deserialized, // validated, and connected. Hopefully the // deserialized blocks get cleaned up by the // GC quickly. if (block.isMemory()) { try { block = block.toBlock(); } catch (e) { this.logger.error(e); throw new VerifyError(block, 'malformed', 'error parsing message', 10, true); } } // Create a new chain entry. const entry = ChainEntry.fromBlock(this, block, prev); // The block is on a alternate chain if the // chainwork is less than or equal to // our tip's. Add the block but do _not_ // connect the inputs. if (entry.chainwork.cmp(this.tip.chainwork) <= 0) { // Save block to an alternate chain. await this.saveAlternate(entry, block, prev, flags); } else { // Attempt to add block to the chain index. await this.setBestChain(entry, block, prev, flags); } // Keep track of stats. this.logStatus(start, block, entry); // Check sync state. this.maybeSync(); return entry; }; /** * Handle orphans. * @method * @private * @param {ChainEntry} entry * @returns {Promise} */ Chain.prototype.handleOrphans = async function handleOrphans(entry) { let orphan = this.resolveOrphan(entry.hash); while (orphan) { const {block, flags, id} = orphan; try { entry = await this.connect(entry, block, flags); } catch (err) { if (err.type === 'VerifyError') { this.logger.warning( 'Could not resolve orphan block %s: %s.', block.rhash(), err.message); this.emit('bad orphan', err, id); break; } throw err; } this.logger.debug( 'Orphan block was resolved: %s (%d).', block.rhash(), entry.height); this.emit('resolved', block, entry); orphan = this.resolveOrphan(entry.hash); } }; /** * Test whether the chain has reached its slow height. * @private * @returns {Boolean} */ Chain.prototype.isSlow = function isSlow() { if (this.options.spv) return false; if (this.synced) return true; if (this.height === 1 || this.height % 20 === 0) return true; if (this.height >= this.network.block.slowHeight) return true; return false; }; /** * Calculate the time difference from * start time and log block. * @private * @param {Array} start * @param {Block} block * @param {ChainEntry} entry */ Chain.prototype.logStatus = function logStatus(start, block, entry) { if (!this.isSlow()) return; // Report memory for debugging. this.logger.memory(); const elapsed = util.hrtime(start); this.logger.info( 'Block %s (%d) added to chain (size=%d txs=%d time=%d).', entry.rhash(), entry.height, block.getSize(), block.txs.length, elapsed); if (this.db.coinCache.capacity > 0) { this.logger.debug('Coin Cache: size=%dmb, items=%d.', util.mb(this.db.coinCache.size), this.db.coinCache.items); } }; /** * Verify a block hash and height against the checkpoints. * @private * @param {ChainEntry} prev * @param {Hash} hash * @returns {Boolean} */ Chain.prototype.verifyCheckpoint = function verifyCheckpoint(prev, hash) { if (!this.options.checkpoints) return true; const height = prev.height + 1; const checkpoint = this.network.checkpointMap[height]; if (!checkpoint) return true; if (hash === checkpoint) { this.logger.debug('Hit checkpoint block %s (%d).', util.revHex(hash), height); this.emit('checkpoint', hash, height); return true; } // Someone is either mining on top of // an old block for no reason, or the // consensus protocol is broken and // there was a 20k+ block reorg. this.logger.warning( 'Checkpoint mismatch at height %d: expected=%s received=%s', height, util.revHex(checkpoint), util.revHex(hash) ); this.purgeOrphans(); return false; }; /** * Store an orphan. * @private * @param {Block} block * @param {Number?} flags * @param {Number?} id */ Chain.prototype.storeOrphan = function storeOrphan(block, flags, id) { const hash = block.hash('hex'); const height = block.getCoinbaseHeight(); let orphan = this.orphanPrev.get(block.prevBlock); // The orphan chain forked. if (orphan) { assert(orphan.block.hash('hex') !== hash); assert(orphan.block.prevBlock === block.prevBlock); this.logger.warning( 'Removing forked orphan block: %s (%d).', orphan.block.rhash(), height); this.removeOrphan(orphan); } this.limitOrphans(); orphan = new Orphan(block, flags, id); this.addOrphan(orphan); this.logger.debug( 'Storing orphan block: %s (%d).', block.rhash(), height); this.emit('orphan', block); }; /** * Add an orphan. * @private * @param {Orphan} orphan * @returns {Orphan} */ Chain.prototype.addOrphan = function addOrphan(orphan) { const block = orphan.block; const hash = block.hash('hex'); assert(!this.orphanMap.has(hash)); assert(!this.orphanPrev.has(block.prevBlock)); assert(this.orphanMap.size >= 0); this.orphanMap.set(hash, orphan); this.orphanPrev.set(block.prevBlock, orphan); return orphan; }; /** * Remove an orphan. * @private * @param {Orphan} orphan * @returns {Orphan} */ Chain.prototype.removeOrphan = function removeOrphan(orphan) { const block = orphan.block; const hash = block.hash('hex'); assert(this.orphanMap.has(hash)); assert(this.orphanPrev.has(block.prevBlock)); assert(this.orphanMap.size > 0); this.orphanMap.delete(hash); this.orphanPrev.delete(block.prevBlock); return orphan; }; /** * Test whether a hash would resolve the next orphan. * @private * @param {Hash} hash - Previous block hash. * @returns {Boolean} */ Chain.prototype.hasNextOrphan = function hasNextOrphan(hash) { return this.orphanPrev.has(hash); }; /** * Resolve an orphan. * @private * @param {Hash} hash - Previous block hash. * @returns {Orphan} */ Chain.prototype.resolveOrphan = function resolveOrphan(hash) { const orphan = this.orphanPrev.get(hash); if (!orphan) return null; return this.removeOrphan(orphan); }; /** * Purge any waiting orphans. */ Chain.prototype.purgeOrphans = function purgeOrphans() { const count = this.orphanMap.size; if (count === 0) return; this.orphanMap.clear(); this.orphanPrev.clear(); this.logger.debug('Purged %d orphans.', count); }; /** * Prune orphans, only keep the orphan with the highest * coinbase height (likely to be the peer's tip). */ Chain.prototype.limitOrphans = function limitOrphans() { const now = util.now(); let oldest; for (const orphan of this.orphanMap.values()) { if (now < orphan.time + 60 * 60) { if (!oldest || orphan.time < oldest.time) oldest = orphan; continue; } this.removeOrphan(orphan); } if (this.orphanMap.size < this.options.maxOrphans) return; if (!oldest) return; this.removeOrphan(oldest); }; /** * Test whether an invalid block hash has been seen. * @private * @param {Block} block * @returns {Boolean} */ Chain.prototype.hasInvalid = function hasInvalid(block) { const hash = block.hash('hex'); if (this.invalid.has(hash)) return true; if (this.invalid.has(block.prevBlock)) { this.setInvalid(hash); return true; } return false; }; /** * Mark a block as invalid. * @private * @param {Hash} hash */ Chain.prototype.setInvalid = function setInvalid(hash) { this.invalid.set(hash, true); }; /** * Forget an invalid block hash. * @private * @param {Hash} hash */ Chain.prototype.removeInvalid = function removeInvalid(hash) { this.invalid.remove(hash); }; /** * Test the chain to see if it contains * a block, or has recently seen a block. * @method * @param {Hash} hash * @returns {Promise} - Returns Boolean. */ Chain.prototype.has = async function has(hash) { if (this.hasOrphan(hash)) return true; if (this.locker.has(hash)) return true; if (this.invalid.has(hash)) return true; return await this.hasEntry(hash); }; /** * Find the corresponding block entry by hash or height. * @param {Hash|Number} hash/height * @returns {Promise} - Returns {@link ChainEntry}. */ Chain.prototype.getEntry = function getEntry(hash) { return this.db.getEntry(hash); }; /** * Test the chain to see if it contains a block. * @param {Hash} hash * @returns {Promise} - Returns Boolean. */ Chain.prototype.hasEntry = function hasEntry(hash) { return this.db.hasEntry(hash); }; /** * Get an orphan block. * @param {Hash} hash * @returns {Block} */ Chain.prototype.getOrphan = function getOrphan(hash) { return this.orphanMap.get(hash) || null; }; /** * Test the chain to see if it contains an orphan. * @param {Hash} hash * @returns {Promise} - Returns Boolean. */ Chain.prototype.hasOrphan = function hasOrphan(hash) { return this.orphanMap.has(hash); }; /** * Test the chain to see if it contains a pending block in its queue. * @param {Hash} hash * @returns {Promise} - Returns Boolean. */ Chain.prototype.hasPending = function hasPending(hash) { return this.locker.hasPending(hash); }; /** * Get coin viewpoint (spent). * @method * @param {TX} tx * @returns {Promise} - Returns {@link CoinView}. */ Chain.prototype.getSpentView = async function getSpentView(tx) { const unlock = await this.locker.lock(); try { return await this.db.getSpentView(tx); } finally { unlock(); } }; /** * Test the chain to see if it is synced. * @returns {Boolean} */ Chain.prototype.isFull = function isFull() { return this.synced; }; /** * Potentially emit a `full` event. * @private */ Chain.prototype.maybeSync = function maybeSync() { if (this.synced) return; if (this.options.checkpoints) { if (this.height < this.network.lastCheckpoint) return; } if (this.tip.time < util.now() - this.network.block.maxTipAge) return; if (!this.hasChainwork()) return; this.synced = true; this.emit('full'); }; /** * Test the chain to see if it has the * minimum required chainwork for the * network. * @returns {Boolean} */ Chain.prototype.hasChainwork = function hasChainwork() { return this.tip.chainwork.gte(this.network.pow.chainwork); }; /** * Get the fill percentage. * @returns {Number} percent - Ranges from 0.0 to 1.0. */ Chain.prototype.getProgress = function getProgress() { const start = this.network.genesis.time; const current = this.tip.time - start; const end = util.now() - start - 40 * 60; return Math.min(1, current / end); }; /** * Calculate chain locator (an array of hashes). * @method * @param {Hash?} start - Height or hash to treat as the tip. * The current tip will be used if not present. Note that this can be a * non-existent hash, which is useful for headers-first locators. * @returns {Promise} - Returns {@link Hash}[]. */ Chain.prototype.getLocator = async function getLocator(start) { const unlock = await this.locker.lock(); try { return await this._getLocator(start); } finally { unlock(); } }; /** * Calculate chain locator without a lock. * @method * @private * @param {Hash?} start * @returns {Promise} */ Chain.prototype._getLocator = async function _getLocator(start) { if (start == null) start = this.tip.hash; assert(typeof start === 'string'); let entry = await this.db.getEntry(start); const hashes = []; if (!entry) { entry = this.tip; hashes.push(start); } let hash = entry.hash; let height = entry.height; const main = await entry.isMainChain(); let step = 1; hashes.push(hash); while (height > 0) { height -= step; if (height < 0) height = 0; if (hashes.length > 10) step *= 2; if (main) { // If we're on the main chain, we can // do a fast lookup of the hash. hash = await this.db.getHash(height); assert(hash); } else { const ancestor = await entry.getAncestor(height); assert(ancestor); hash = ancestor.hash; } hashes.push(hash); } return hashes; }; /** * Calculate the orphan root of the hash (if it is an orphan). * @param {Hash} hash * @returns {Hash} */ Chain.prototype.getOrphanRoot = function getOrphanRoot(hash) { let root = null; assert(hash); for (;;) { const orphan = this.orphanMap.get(hash); if (!orphan) break; root = hash; hash = orphan.block.prevBlock; } return root; }; /** * Calculate the time difference (in seconds) * between two blocks by examining chainworks. * @param {ChainEntry} to * @param {ChainEntry} from * @returns {Number} */ Chain.prototype.getProofTime = function getProofTime(to, from) { const pow = this.network.pow; let sign, work; if (to.chainwork.cmp(from.chainwork) > 0) { work = to.chainwork.sub(from.chainwork); sign = 1; } else { work = from.chainwork.sub(to.chainwork); sign = -1; } work = work.imuln(pow.targetSpacing); work = work.div(this.tip.getProof()); if (work.bitLength() > 53) return sign * Number.MAX_SAFE_INTEGER; return sign * work.toNumber(); }; /** * Calculate the next target based on the chain tip. * @method * @returns {Promise} - returns Number * (target is in compact/mantissa form). */ Chain.prototype.getCurrentTarget = async function getCurrentTarget() { return await this.getTarget(this.network.now(), this.tip); }; /** * Calculate the next target. * @method * @param {Number} time - Next block timestamp. * @param {ChainEntry} prev - Previous entry. * @returns {Promise} - returns Number * (target is in compact/mantissa form). */ Chain.prototype.getTarget = async function getTarget(time, prev) { const pow = this.network.pow; // Genesis if (!prev) { assert(time === this.network.genesis.time); return pow.bits; } // Do not retarget if ((prev.height + 1) % pow.retargetInterval !== 0) { if (pow.targetReset) { // Special behavior for testnet: if (time > prev.time + pow.targetSpacing * 2) return pow.bits; while (prev.height !== 0 && prev.height % pow.retargetInterval !== 0 && prev.bits === pow.bits) { const cache = prev.getPrevCache(); if (cache) { prev = cache; continue; } prev = await prev.getPrevious(); assert(prev); } } return prev.bits; } // Back 2 weeks const height = prev.height - (pow.retargetInterval - 1); assert(height >= 0); const first = await prev.getAncestor(height); assert(first); return this.retarget(prev, first); }; /** * Retarget. This is called when the chain height * hits a retarget diff interval. * @param {ChainEntry} prev - Previous entry. * @param {ChainEntry} first - Chain entry from 2 weeks prior. * @returns {Number} target - Target in compact/mantissa form. */ Chain.prototype.retarget = function retarget(prev, first) { const pow = this.network.pow; const targetTimespan = pow.targetTimespan; if (pow.noRetargeting) return prev.bits; const target = consensus.fromCompact(prev.bits); let actualTimespan = prev.time - first.time; if (actualTimespan < targetTimespan / 4 | 0) actualTimespan = targetTimespan / 4 | 0; if (actualTimespan > targetTimespan * 4) actualTimespan = targetTimespan * 4; target.imuln(actualTimespan); target.idivn(targetTimespan); if (target.cmp(pow.limit) > 0) return pow.bits; return consensus.toCompact(target); }; /** * Find a locator. Analagous to bitcoind's `FindForkInGlobalIndex()`. * @method * @param {Hash[]} locator - Hashes. * @returns {Promise} - Returns {@link Hash} (the * hash of the latest known block). */ Chain.prototype.findLocator = async function findLocator(locator) { for (const hash of locator) { if (await this.db.isMainChain(hash)) return hash; } return this.network.genesis.hash; }; /** * Check whether a versionbits deployment is active (BIP9: versionbits). * @example * await chain.isActive(tip, deployments.segwit); * @method * @see https://github.com/bitcoin/bips/blob/master/bip-0009.mediawiki * @param {ChainEntry} prev - Previous chain entry. * @param {String} id - Deployment id. * @returns {Promise} - Returns Number. */ Chain.prototype.isActive = async function isActive(prev, deployment) { const state = await this.getState(prev, deployment); return state === thresholdStates.ACTIVE; }; /** * Get chain entry state for a deployment (BIP9: versionbits). * @method * @example * await chain.getState(tip, deployments.segwit); * @see https://github.com/bitcoin/bips/blob/master/bip-0009.mediawiki * @param {ChainEntry} prev - Previous chain entry. * @param {String} id - Deployment id. * @returns {Promise} - Returns Number. */ Chain.prototype.getState = async function getState(prev, deployment) { let window = this.network.minerWindow; let threshold = this.network.activationThreshold; const bit = deployment.bit; if (deployment.threshold !== -1) threshold = deployment.threshold; if (deployment.window !== -1) window = deployment.window; if (((prev.height + 1) % window) !== 0) { const height = prev.height - ((prev.height + 1) % window); prev = await prev.getAncestor(height); if (!prev) return thresholdStates.DEFINED; assert(prev.height === height); assert(((prev.height + 1) % window) === 0); } let entry = prev; let state = thresholdStates.DEFINED; const compute = []; while (entry) { const cached = this.db.stateCache.get(bit, entry); if (cached !== -1) { state = cached; break; } const time = await entry.getMedianTime(); if (time < deployment.startTime) { state = thresholdStates.DEFINED; this.db.stateCache.set(bit, entry, state); break; } compute.push(entry); const height = entry.height - window; entry = await entry.getAncestor(height); } while (compute.length) { const entry = compute.pop(); switch (state) { case thresholdStates.DEFINED: { const time = await entry.getMedianTime(); if (time >= deployment.timeout) { state = thresholdStates.FAILED; break; } if (time >= deployment.startTime) { state = thresholdStates.STARTED; break; } break; } case thresholdStates.STARTED: { const time = await entry.getMedianTime(); let block = entry; let count = 0; if (time >= deployment.timeout) { state = thresholdStates.FAILED; break; } for (let i = 0; i < window; i++) { if (block.hasBit(bit)) count++; if (count >= threshold) { state = thresholdStates.LOCKED_IN; break; } block = await block.getPrevious(); assert(block); } break; } case thresholdStates.LOCKED_IN: { state = thresholdStates.ACTIVE; break; } case thresholdStates.FAILED: case thresholdStates.ACTIVE: { break; } default: { assert(false, 'Bad state.'); break; } } this.db.stateCache.set(bit, entry, state); } return state; }; /** * Compute the version for a new block (BIP9: versionbits). * @method * @see https://github.com/bitcoin/bips/blob/master/bip-0009.mediawiki * @param {ChainEntry} prev - Previous chain entry (usually the tip). * @returns {Promise} - Returns Number. */ Chain.prototype.computeBlockVersion = async function computeBlockVersion(prev) { let version = 0; for (const deployment of this.network.deploys) { const state = await this.getState(prev, deployment); if (state === thresholdStates.LOCKED_IN || state === thresholdStates.STARTED) { version |= 1 << deployment.bit; } } version |= consensus.VERSION_TOP_BITS; version >>>= 0; return version; }; /** * Get the current deployment state of the chain. Called on load. * @method * @private * @returns {Promise} - Returns {@link DeploymentState}. */ Chain.prototype.getDeploymentState = async function getDeploymentState() { const prev = await this.tip.getPrevious(); if (!prev) { assert(this.tip.isGenesis()); return this.state; } if (this.options.spv) return this.state; return await this.getDeployments(this.tip.time, prev); }; /** * Check transaction finality, taking into account MEDIAN_TIME_PAST * if it is present in the lock flags. * @method * @param {ChainEntry} prev - Previous chain entry. * @param {TX} tx * @param {LockFlags} flags * @returns {Promise} - Returns Boolean. */ Chain.prototype.verifyFinal = async function verifyFinal(prev, tx, flags) { const height = prev.height + 1; // We can skip MTP if the locktime is height. if (tx.locktime < consensus.LOCKTIME_THRESHOLD) return tx.isFinal(height, -1); if (flags & common.lockFlags.MEDIAN_TIME_PAST) { const time = await prev.getMedianTime(); return tx.isFinal(height, time); } return tx.isFinal(height, this.network.now()); }; /** * Get the necessary minimum time and height sequence locks for a transaction. * @method * @param {ChainEntry} prev * @param {TX} tx * @param {CoinView} view * @param {LockFlags} flags * @returns {Promise} */ Chain.prototype.getLocks = async function getLocks(prev, tx, view, flags) { const GRANULARITY = consensus.SEQUENCE_GRANULARITY; const DISABE_FLAG = consensus.SEQUENCE_DISABLE_FLAG; const TYPE_FLAG = consensus.SEQUENCE_TYPE_FLAG; const MASK = consensus.SEQUENCE_MASK; if (!(flags & common.lockFlags.VERIFY_SEQUENCE)) return [-1, -1]; if (tx.isCoinbase() || tx.version < 2) return [-1, -1]; let minHeight = -1; let minTime = -1; for (const {prevout, sequence} of tx.inputs) { if (sequence & DISABE_FLAG) continue; let height = view.getHeight(prevout); if (height === -1) height = this.height + 1; if (!(sequence & TYPE_FLAG)) { height += (sequence & MASK) - 1; minHeight = Math.max(minHeight, height); continue; } height = Math.max(height - 1, 0); const entry = await prev.getAncestor(height); assert(entry, 'Database is corrupt.'); let time = await entry.getMedianTime(); time += ((sequence & MASK) << GRANULARITY) - 1; minTime = Math.max(minTime, time); } return [minHeight, minTime]; }; /** * Verify sequence locks. * @method * @param {ChainEntry} prev * @param {TX} tx * @param {CoinView} view * @param {LockFlags} flags * @returns {Promise} - Returns Boolean. */ Chain.prototype.verifyLocks = async function verifyLocks(prev, tx, view, flags) { const [height, time] = await this.getLocks(prev, tx, view, flags); // Also catches case where // height is `-1`. Fall through. if (height >= prev.height + 1) return false; if (time === -1) return true; const mtp = await prev.getMedianTime(); if (time >= mtp) return false; return true; }; /** * ChainOptions * @alias module:blockchain.ChainOptions * @constructor * @param {Object} options */ function ChainOptions(options) { if (!(this instanceof ChainOptions)) return new ChainOptions(options); this.network = Network.primary; this.logger = Logger.global; this.workers = null; this.prefix = null; this.location = null; this.db = 'memory'; this.maxFiles = 64; this.cacheSize = 32 << 20; this.compression = true; this.bufferKeys = ChainDB.layout.binary; this.spv = false; this.bip91 = false; this.bip148 = false; this.prune = false; this.indexTX = false; this.indexAddress = false; this.forceFlags = false; this.coinCache = 0; this.entryCache = 5000; this.maxOrphans = 20; this.checkpoints = true; if (options) this.fromOptions(options); } /** * Inject properties from object. * @private * @param {Object} options * @returns {ChainOptions} */ ChainOptions.prototype.fromOptions = function fromOptions(options) { if (options.network != null) this.network = Network.get(options.network); if (options.logger != null) { assert(typeof options.logger === 'object'); this.logger = options.logger; } if (options.workers != null) { assert(typeof options.workers === 'object'); this.workers = options.workers; } if (options.spv != null) { assert(typeof options.spv === 'boolean'); this.spv = options.spv; } if (options.prefix != null) { assert(typeof options.prefix === 'string'); this.prefix = options.prefix; this.location = this.spv ? path.join(this.prefix, 'spvchain') : path.join(this.prefix, 'chain'); } if (options.location != null) { assert(typeof options.location === 'string'); this.location = options.location; } if (options.db != null) { assert(typeof options.db === 'string'); this.db = options.db; } if (options.maxFiles != null) { assert(util.isU32(options.maxFiles)); this.maxFiles = options.maxFiles; } if (options.cacheSize != null) { assert(util.isU64(options.cacheSize)); this.cacheSize = options.cacheSize; } if (options.compression != null) { assert(typeof options.compression === 'boolean'); this.compression = options.compression; } if (options.prune != null) { assert(typeof options.prune === 'boolean'); this.prune = options.prune; } if (options.indexTX != null) { assert(typeof options.indexTX === 'boolean'); this.indexTX = options.indexTX; } if (options.indexAddress != null) { assert(typeof options.indexAddress === 'boolean'); this.indexAddress = options.indexAddress; } if (options.forceFlags != null) { assert(typeof options.forceFlags === 'boolean'); this.forceFlags = options.forceFlags; } if (options.bip91 != null) { assert(typeof options.bip91 === 'boolean'); this.bip91 = options.bip91; } if (options.bip148 != null) { assert(typeof options.bip148 === 'boolean'); this.bip148 = options.bip148; } if (options.coinCache != null) { assert(util.isU64(options.coinCache)); this.coinCache = options.coinCache; } if (options.entryCache != null) { assert(util.isU32(options.entryCache)); this.entryCache = options.entryCache; } if (options.maxOrphans != null) { assert(util.isU32(options.maxOrphans)); this.maxOrphans = options.maxOrphans; } if (options.checkpoints != null) { assert(typeof options.checkpoints === 'boolean'); this.checkpoints = options.checkpoints; } return this; }; /** * Instantiate chain options from object. * @param {Object} options * @returns {ChainOptions} */ ChainOptions.fromOptions = function fromOptions(options) { return new ChainOptions().fromOptions(options); }; /** * Represents the deployment state of the chain. * @alias module:blockchain.DeploymentState * @constructor * @property {VerifyFlags} flags * @property {LockFlags} lockFlags * @property {Boolean} bip34 */ function DeploymentState() { if (!(this instanceof DeploymentState)) return new DeploymentState(); this.flags = Script.flags.MANDATORY_VERIFY_FLAGS; this.flags &= ~Script.flags.VERIFY_P2SH; this.lockFlags = common.lockFlags.MANDATORY_LOCKTIME_FLAGS; this.bip34 = false; this.bip91 = false; this.bip148 = false; } /** * Test whether p2sh is active. * @returns {Boolean} */ DeploymentState.prototype.hasP2SH = function hasP2SH() { return (this.flags & Script.flags.VERIFY_P2SH) !== 0; }; /** * Test whether bip34 (coinbase height) is active. * @returns {Boolean} */ DeploymentState.prototype.hasBIP34 = function hasBIP34() { return this.bip34; }; /** * Test whether bip66 (VERIFY_DERSIG) is active. * @returns {Boolean} */ DeploymentState.prototype.hasBIP66 = function hasBIP66() { return (this.flags & Script.flags.VERIFY_DERSIG) !== 0; }; /** * Test whether cltv is active. * @returns {Boolean} */ DeploymentState.prototype.hasCLTV = function hasCLTV() { return (this.flags & Script.flags.VERIFY_CHECKLOCKTIMEVERIFY) !== 0; }; /** * Test whether median time past locktime is active. * @returns {Boolean} */ DeploymentState.prototype.hasMTP = function hasMTP() { return (this.lockFlags & common.lockFlags.MEDIAN_TIME_PAST) !== 0; }; /** * Test whether csv is active. * @returns {Boolean} */ DeploymentState.prototype.hasCSV = function hasCSV() { return (this.flags & Script.flags.VERIFY_CHECKSEQUENCEVERIFY) !== 0; }; /** * Test whether segwit is active. * @returns {Boolean} */ DeploymentState.prototype.hasWitness = function hasWitness() { return (this.flags & Script.flags.VERIFY_WITNESS) !== 0; }; /** * Test whether bip91 is active. * @returns {Boolean} */ DeploymentState.prototype.hasBIP91 = function hasBIP91() { return this.bip91; }; /** * Test whether bip148 is active. * @returns {Boolean} */ DeploymentState.prototype.hasBIP148 = function hasBIP148() { return this.bip148; }; /** * Orphan * @constructor * @ignore */ function Orphan(block, flags, id) { this.block = block; this.flags = flags; this.id = id; this.time = util.now(); } /* * Expose */ module.exports = Chain;