2019-07-21 16:59:47 +02:00
|
|
|
const config = require('../../mempool-config.json');
|
2020-02-16 16:15:07 +01:00
|
|
|
import bitcoinApi from './bitcoin/electrs-api';
|
2020-02-23 13:16:50 +01:00
|
|
|
import memPool from './mempool';
|
2020-06-08 21:08:46 +02:00
|
|
|
import { Block, TransactionExtended, TransactionMinerInfo } from '../interfaces';
|
2020-05-24 11:29:30 +02:00
|
|
|
import { Common } from './common';
|
2019-07-21 16:59:47 +02:00
|
|
|
|
|
|
|
class Blocks {
|
2020-02-16 16:15:07 +01:00
|
|
|
private blocks: Block[] = [];
|
2019-08-29 01:17:31 +02:00
|
|
|
private currentBlockHeight = 0;
|
2020-06-08 21:08:46 +02:00
|
|
|
private newBlockCallback: ((block: Block, txIds: string[], transactions: TransactionExtended[]) => void) | undefined;
|
2019-08-29 01:17:31 +02:00
|
|
|
|
2020-02-16 16:15:07 +01:00
|
|
|
constructor() { }
|
2019-07-21 16:59:47 +02:00
|
|
|
|
2020-02-16 16:15:07 +01:00
|
|
|
public getBlocks(): Block[] {
|
2019-07-21 16:59:47 +02:00
|
|
|
return this.blocks;
|
|
|
|
}
|
|
|
|
|
2020-02-29 15:52:04 +01:00
|
|
|
public setBlocks(blocks: Block[]) {
|
|
|
|
this.blocks = blocks;
|
|
|
|
}
|
|
|
|
|
2020-06-08 21:08:46 +02:00
|
|
|
public setNewBlockCallback(fn: (block: Block, txIds: string[], transactions: TransactionExtended[]) => void) {
|
2020-02-16 16:15:07 +01:00
|
|
|
this.newBlockCallback = fn;
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public async updateBlocks() {
|
|
|
|
try {
|
2020-02-16 16:15:07 +01:00
|
|
|
const blockHeightTip = await bitcoinApi.getBlockHeightTip();
|
2019-07-21 16:59:47 +02:00
|
|
|
|
|
|
|
if (this.blocks.length === 0) {
|
2020-02-16 16:15:07 +01:00
|
|
|
this.currentBlockHeight = blockHeightTip - config.INITIAL_BLOCK_AMOUNT;
|
2019-07-21 16:59:47 +02:00
|
|
|
} else {
|
2019-08-29 01:17:31 +02:00
|
|
|
this.currentBlockHeight = this.blocks[this.blocks.length - 1].height;
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-03-03 09:11:14 +01:00
|
|
|
if (blockHeightTip - this.currentBlockHeight > config.INITIAL_BLOCK_AMOUNT * 2) {
|
|
|
|
console.log(`${blockHeightTip - this.currentBlockHeight} blocks since tip. Fast forwarding to the ${config.INITIAL_BLOCK_AMOUNT} recent blocks`);
|
|
|
|
this.currentBlockHeight = blockHeightTip - config.INITIAL_BLOCK_AMOUNT;
|
|
|
|
}
|
|
|
|
|
2020-02-16 16:15:07 +01:00
|
|
|
while (this.currentBlockHeight < blockHeightTip) {
|
|
|
|
if (this.currentBlockHeight === 0) {
|
|
|
|
this.currentBlockHeight = blockHeightTip;
|
2019-07-21 16:59:47 +02:00
|
|
|
} else {
|
2020-02-16 16:15:07 +01:00
|
|
|
this.currentBlockHeight++;
|
|
|
|
console.log(`New block found (#${this.currentBlockHeight})!`);
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-02-16 16:15:07 +01:00
|
|
|
const blockHash = await bitcoinApi.getBlockHash(this.currentBlockHeight);
|
|
|
|
const block = await bitcoinApi.getBlock(blockHash);
|
|
|
|
const txIds = await bitcoinApi.getTxIdsForBlock(blockHash);
|
|
|
|
|
2020-02-23 13:16:50 +01:00
|
|
|
const mempool = memPool.getMempool();
|
|
|
|
let found = 0;
|
|
|
|
let notFound = 0;
|
|
|
|
|
|
|
|
const transactions: TransactionExtended[] = [];
|
|
|
|
|
2020-03-04 09:10:30 +01:00
|
|
|
for (let i = 0; i < txIds.length; i++) {
|
2020-02-23 13:16:50 +01:00
|
|
|
if (mempool[txIds[i]]) {
|
|
|
|
transactions.push(mempool[txIds[i]]);
|
|
|
|
found++;
|
|
|
|
} else {
|
|
|
|
console.log(`Fetching block tx ${i} of ${txIds.length}`);
|
|
|
|
const tx = await memPool.getTransactionExtended(txIds[i]);
|
|
|
|
if (tx) {
|
|
|
|
transactions.push(tx);
|
|
|
|
}
|
|
|
|
notFound++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-23 21:42:29 +01:00
|
|
|
console.log(`${found} of ${txIds.length} found in mempool. ${notFound} not found.`);
|
|
|
|
|
2020-03-04 09:10:30 +01:00
|
|
|
block.reward = transactions[0].vout.reduce((acc, curr) => acc + curr.value, 0);
|
2020-02-23 13:16:50 +01:00
|
|
|
transactions.sort((a, b) => b.feePerVsize - a.feePerVsize);
|
2020-05-24 11:29:30 +02:00
|
|
|
block.medianFee = transactions.length > 1 ? Common.median(transactions.map((tx) => tx.feePerVsize)) : 0;
|
2020-05-26 06:02:12 +02:00
|
|
|
block.feeRange = transactions.length > 1 ? Common.getFeesInRange(transactions, 8, 1) : [0, 0];
|
2020-05-24 17:45:45 +02:00
|
|
|
block.coinbaseTx = this.stripCoinbaseTransaction(transactions[0]);
|
2020-02-16 16:15:07 +01:00
|
|
|
|
2019-07-21 16:59:47 +02:00
|
|
|
this.blocks.push(block);
|
|
|
|
if (this.blocks.length > config.KEEP_BLOCK_AMOUNT) {
|
|
|
|
this.blocks.shift();
|
|
|
|
}
|
|
|
|
|
2020-06-08 21:08:46 +02:00
|
|
|
if (this.newBlockCallback) {
|
|
|
|
this.newBlockCallback(block, txIds, transactions);
|
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-02-16 16:15:07 +01:00
|
|
|
} catch (err) {
|
|
|
|
console.log('updateBlocks error', err);
|
2019-08-29 01:17:31 +02:00
|
|
|
}
|
|
|
|
}
|
2020-05-24 17:45:45 +02:00
|
|
|
|
2020-06-08 21:08:46 +02:00
|
|
|
private stripCoinbaseTransaction(tx: TransactionExtended): TransactionMinerInfo {
|
2020-05-24 17:45:45 +02:00
|
|
|
return {
|
|
|
|
vin: [{
|
|
|
|
scriptsig: tx.vin[0].scriptsig
|
|
|
|
}],
|
2020-06-19 18:32:17 +02:00
|
|
|
vout: tx.vout.map((vout) => ({ scriptpubkey_address: vout.scriptpubkey_address, value: vout.value }))
|
2020-05-24 17:45:45 +02:00
|
|
|
};
|
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export default new Blocks();
|