mempool/backend/src/api/blocks.ts

119 lines
3.6 KiB
TypeScript
Raw Normal View History

2019-07-21 16:59:47 +02:00
const config = require('../../mempool-config.json');
import bitcoinApi from './bitcoin/electrs-api';
2020-02-23 13:16:50 +01:00
import memPool from './mempool';
import { Block, TransactionExtended } from '../interfaces';
2019-07-21 16:59:47 +02:00
class Blocks {
private blocks: Block[] = [];
private currentBlockHeight = 0;
private newBlockCallback: Function = () => {};
constructor() { }
2019-07-21 16:59:47 +02:00
public getBlocks(): Block[] {
2019-07-21 16:59:47 +02:00
return this.blocks;
}
public setBlocks(blocks: Block[]) {
this.blocks = blocks;
}
public setNewBlockCallback(fn: Function) {
this.newBlockCallback = fn;
2019-07-21 16:59:47 +02:00
}
public async updateBlocks() {
try {
const blockHeightTip = await bitcoinApi.getBlockHeightTip();
2019-07-21 16:59:47 +02:00
if (this.blocks.length === 0) {
this.currentBlockHeight = blockHeightTip - config.INITIAL_BLOCK_AMOUNT;
2019-07-21 16:59:47 +02:00
} else {
this.currentBlockHeight = this.blocks[this.blocks.length - 1].height;
2019-07-21 16:59:47 +02: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;
}
while (this.currentBlockHeight < blockHeightTip) {
if (this.currentBlockHeight === 0) {
this.currentBlockHeight = blockHeightTip;
2019-07-21 16:59:47 +02:00
} else {
this.currentBlockHeight++;
console.log(`New block found (#${this.currentBlockHeight})!`);
2019-07-21 16:59:47 +02: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[] = [];
for (let i = 1; i < txIds.length; i++) {
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-02-23 13:16:50 +01:00
transactions.sort((a, b) => b.feePerVsize - a.feePerVsize);
2020-02-24 04:58:52 +01:00
block.medianFee = transactions.length ? this.median(transactions.map((tx) => tx.feePerVsize)) : 0;
2020-02-24 16:51:27 +01:00
block.feeRange = transactions.length ? this.getFeesInRange(transactions, 8) : [0, 0];
2019-07-21 16:59:47 +02:00
this.blocks.push(block);
if (this.blocks.length > config.KEEP_BLOCK_AMOUNT) {
this.blocks.shift();
}
2020-02-23 13:16:50 +01:00
this.newBlockCallback(block, txIds, transactions);
2019-07-21 16:59:47 +02:00
}
} catch (err) {
console.log('updateBlocks error', err);
}
}
2019-07-21 16:59:47 +02:00
private median(numbers: number[]) {
let medianNr = 0;
const numsLen = numbers.length;
numbers.sort();
if (numsLen % 2 === 0) {
medianNr = (numbers[numsLen / 2 - 1] + numbers[numsLen / 2]) / 2;
} else {
medianNr = numbers[(numsLen - 1) / 2];
}
return medianNr;
}
private getFeesInRange(transactions: any[], rangeLength: number) {
const arr = [transactions[transactions.length - 1].feePerVsize];
const chunk = 1 / (rangeLength - 1);
let itemsToAdd = rangeLength - 2;
while (itemsToAdd > 0) {
arr.push(transactions[Math.floor(transactions.length * chunk * itemsToAdd)].feePerVsize);
itemsToAdd--;
}
arr.push(transactions[0].feePerVsize);
return arr;
}
2019-07-21 16:59:47 +02:00
}
export default new Blocks();