mempool/backend/src/api/blocks.ts

133 lines
5.0 KiB
TypeScript
Raw Normal View History

import config from '../config';
import bitcoinApi from './bitcoin/bitcoin-api-factory';
import logger from '../logger';
2020-02-23 13:16:50 +01:00
import memPool from './mempool';
2020-12-29 19:47:07 +01:00
import { BlockExtended, TransactionExtended } from '../mempool.interfaces';
2020-05-24 11:29:30 +02:00
import { Common } from './common';
import diskCache from './disk-cache';
2020-12-21 17:08:34 +01:00
import transactionUtils from './transaction-utils';
2019-07-21 16:59:47 +02:00
class Blocks {
2020-12-29 19:47:07 +01:00
private static INITIAL_BLOCK_AMOUNT = 8;
private blocks: BlockExtended[] = [];
private currentBlockHeight = 0;
private lastDifficultyAdjustmentTime = 0;
private newBlockCallbacks: ((block: BlockExtended, txIds: string[], transactions: TransactionExtended[]) => void)[] = [];
constructor() { }
2019-07-21 16:59:47 +02:00
public getBlocks(): BlockExtended[] {
2019-07-21 16:59:47 +02:00
return this.blocks;
}
public setBlocks(blocks: BlockExtended[]) {
this.blocks = blocks;
}
public setNewBlockCallback(fn: (block: BlockExtended, txIds: string[], transactions: TransactionExtended[]) => void) {
this.newBlockCallbacks.push(fn);
2019-07-21 16:59:47 +02:00
}
public async $updateBlocks() {
2020-12-21 17:08:34 +01:00
const blockHeightTip = await bitcoinApi.$getBlockHeightTip();
2019-07-21 16:59:47 +02:00
if (this.blocks.length === 0) {
2020-12-29 19:47:07 +01:00
this.currentBlockHeight = blockHeightTip - Blocks.INITIAL_BLOCK_AMOUNT;
} else {
this.currentBlockHeight = this.blocks[this.blocks.length - 1].height;
}
2020-12-29 19:47:07 +01:00
if (blockHeightTip - this.currentBlockHeight > Blocks.INITIAL_BLOCK_AMOUNT * 2) {
logger.info(`${blockHeightTip - this.currentBlockHeight} blocks since tip. Fast forwarding to the ${Blocks.INITIAL_BLOCK_AMOUNT} recent blocks`);
this.currentBlockHeight = blockHeightTip - Blocks.INITIAL_BLOCK_AMOUNT;
}
if (!this.lastDifficultyAdjustmentTime) {
const heightDiff = blockHeightTip % 2016;
2020-12-21 17:08:34 +01:00
const blockHash = await bitcoinApi.$getBlockHash(blockHeightTip - heightDiff);
const block = await bitcoinApi.$getBlock(blockHash);
this.lastDifficultyAdjustmentTime = block.timestamp;
}
while (this.currentBlockHeight < blockHeightTip) {
if (this.currentBlockHeight === 0) {
this.currentBlockHeight = blockHeightTip;
2019-07-21 16:59:47 +02:00
} else {
this.currentBlockHeight++;
logger.debug(`New block found (#${this.currentBlockHeight})!`);
2019-07-21 16:59:47 +02:00
}
2020-12-21 17:08:34 +01:00
const transactions: TransactionExtended[] = [];
2020-12-21 17:08:34 +01:00
const blockHash = await bitcoinApi.$getBlockHash(this.currentBlockHeight);
const block = await bitcoinApi.$getBlock(blockHash);
const txIds: string[] = await bitcoinApi.$getTxIdsForBlock(blockHash);
const mempool = memPool.getMempool();
let transactionsFound = 0;
2019-07-21 16:59:47 +02:00
for (let i = 0; i < txIds.length; i++) {
2020-12-22 00:04:31 +01:00
// When using bitcoind, just fetch the coinbase tx for now
if (config.MEMPOOL.BACKEND !== 'esplora' && i === 0) {
let txFound = false;
let findCoinbaseTxTries = 0;
// It takes Electrum Server a few seconds to index the transaction after a block is found
while (findCoinbaseTxTries < 5 && !txFound) {
const tx = await transactionUtils.$getTransactionExtended(txIds[i]);
if (tx) {
txFound = true;
transactions.push(tx);
} else {
await Common.sleep(1000);
findCoinbaseTxTries++;
}
2020-12-22 00:04:31 +01:00
}
}
if (mempool[txIds[i]]) {
transactions.push(mempool[txIds[i]]);
transactionsFound++;
} else if (config.MEMPOOL.BACKEND === 'esplora') {
2020-12-22 00:04:31 +01:00
logger.debug(`Fetching block tx ${i} of ${txIds.length}`);
const tx = await transactionUtils.$getTransactionExtended(txIds[i]);
2020-12-22 00:04:31 +01:00
if (tx) {
transactions.push(tx);
2020-02-23 13:16:50 +01:00
}
}
}
2020-02-23 13:16:50 +01:00
logger.debug(`${transactionsFound} of ${txIds.length} found in mempool. ${txIds.length - transactionsFound} not found.`);
const blockExtended: BlockExtended = Object.assign({}, block);
blockExtended.reward = transactions[0].vout.reduce((acc, curr) => acc + curr.value, 0);
blockExtended.coinbaseTx = transactionUtils.stripCoinbaseTransaction(transactions[0]);
transactions.sort((a, b) => b.feePerVsize - a.feePerVsize);
blockExtended.medianFee = transactions.length > 1 ? Common.median(transactions.map((tx) => tx.feePerVsize)) : 0;
blockExtended.feeRange = transactions.length > 1 ? Common.getFeesInRange(transactions.slice(0, transactions.length - 1), 8) : [0, 0];
if (block.height % 2016 === 0) {
this.lastDifficultyAdjustmentTime = block.timestamp;
}
2019-07-21 16:59:47 +02:00
this.blocks.push(blockExtended);
2020-12-29 19:47:07 +01:00
if (this.blocks.length > Blocks.INITIAL_BLOCK_AMOUNT * 4) {
this.blocks = this.blocks.slice(-Blocks.INITIAL_BLOCK_AMOUNT * 4);
2019-07-21 16:59:47 +02:00
}
if (this.newBlockCallbacks.length) {
this.newBlockCallbacks.forEach((cb) => cb(blockExtended, txIds, transactions));
}
diskCache.$saveCacheToDisk();
}
}
public getLastDifficultyAdjustmentTime(): number {
return this.lastDifficultyAdjustmentTime;
}
2020-12-21 17:08:34 +01:00
public getCurrentBlockHeight(): number {
return this.currentBlockHeight;
}
2019-07-21 16:59:47 +02:00
}
export default new Blocks();