2020-10-19 06:57:02 +02:00
|
|
|
import config from '../config';
|
2020-12-20 16:36:36 +01:00
|
|
|
import bitcoinApi from './bitcoin/bitcoin-api-factory';
|
2020-12-27 22:47:22 +01:00
|
|
|
import { TransactionExtended, VbytesPerSecond } from '../mempool.interfaces';
|
2020-10-13 10:27:52 +02:00
|
|
|
import logger from '../logger';
|
2020-09-25 21:11:30 +02:00
|
|
|
import { Common } from './common';
|
2020-12-21 17:08:34 +01:00
|
|
|
import transactionUtils from './transaction-utils';
|
2020-12-27 22:47:22 +01:00
|
|
|
import { IBitcoinApi } from './bitcoin/bitcoin-api.interface';
|
2021-01-05 12:57:06 +01:00
|
|
|
import loadingIndicators from './loading-indicators';
|
2021-09-14 23:47:24 +02:00
|
|
|
import bitcoinClient from './bitcoin/bitcoin-client';
|
|
|
|
import bitcoinSecondClient from './bitcoin/bitcoin-second-client';
|
2019-07-21 16:59:47 +02:00
|
|
|
|
|
|
|
class Mempool {
|
2021-01-06 16:49:28 +01:00
|
|
|
private static WEBSOCKET_REFRESH_RATE_MS = 10000;
|
2021-03-21 00:06:03 +01:00
|
|
|
private static LAZY_DELETE_AFTER_SECONDS = 30;
|
2020-04-01 15:06:44 +02:00
|
|
|
private inSync: boolean = false;
|
2020-06-07 12:30:32 +02:00
|
|
|
private mempoolCache: { [txId: string]: TransactionExtended } = {};
|
2020-12-27 22:47:22 +01:00
|
|
|
private mempoolInfo: IBitcoinApi.MempoolInfo = { loaded: false, size: 0, bytes: 0, usage: 0,
|
2021-04-06 09:07:38 +02:00
|
|
|
maxmempool: 300000000, mempoolminfee: 0.00001000, minrelaytxfee: 0.00001000 };
|
2020-12-27 22:47:22 +01:00
|
|
|
private mempoolChangedCallback: ((newMempool: {[txId: string]: TransactionExtended; }, newTransactions: TransactionExtended[],
|
2020-06-08 21:08:46 +02:00
|
|
|
deletedTransactions: TransactionExtended[]) => void) | undefined;
|
2019-07-21 16:59:47 +02:00
|
|
|
|
|
|
|
private txPerSecondArray: number[] = [];
|
|
|
|
private txPerSecond: number = 0;
|
|
|
|
|
2020-06-10 18:52:14 +02:00
|
|
|
private vBytesPerSecondArray: VbytesPerSecond[] = [];
|
2019-07-21 16:59:47 +02:00
|
|
|
private vBytesPerSecond: number = 0;
|
2020-06-18 08:54:54 +02:00
|
|
|
private mempoolProtection = 0;
|
2020-09-25 21:11:30 +02:00
|
|
|
private latestTransactions: any[] = [];
|
2019-07-21 16:59:47 +02:00
|
|
|
|
|
|
|
constructor() {
|
|
|
|
setInterval(this.updateTxPerSecond.bind(this), 1000);
|
2021-03-21 00:06:03 +01:00
|
|
|
setInterval(this.deleteExpiredTransactions.bind(this), 20000);
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2021-01-20 11:16:43 +01:00
|
|
|
public isInSync(): boolean {
|
2020-04-01 15:06:44 +02:00
|
|
|
return this.inSync;
|
|
|
|
}
|
|
|
|
|
2021-01-20 11:16:43 +01:00
|
|
|
public setOutOfSync(): void {
|
|
|
|
this.inSync = false;
|
|
|
|
loadingIndicators.setProgress('mempool', 99);
|
|
|
|
}
|
|
|
|
|
2020-09-25 21:11:30 +02:00
|
|
|
public getLatestTransactions() {
|
|
|
|
return this.latestTransactions;
|
|
|
|
}
|
|
|
|
|
2020-06-08 21:08:46 +02:00
|
|
|
public setMempoolChangedCallback(fn: (newMempool: { [txId: string]: TransactionExtended; },
|
|
|
|
newTransactions: TransactionExtended[], deletedTransactions: TransactionExtended[]) => void) {
|
2019-07-21 16:59:47 +02:00
|
|
|
this.mempoolChangedCallback = fn;
|
|
|
|
}
|
|
|
|
|
2020-02-23 13:16:50 +01:00
|
|
|
public getMempool(): { [txid: string]: TransactionExtended } {
|
2020-02-16 16:15:07 +01:00
|
|
|
return this.mempoolCache;
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-06-07 12:30:32 +02:00
|
|
|
public setMempool(mempoolData: { [txId: string]: TransactionExtended }) {
|
2020-02-16 16:15:07 +01:00
|
|
|
this.mempoolCache = mempoolData;
|
2020-07-03 18:45:19 +02:00
|
|
|
if (this.mempoolChangedCallback) {
|
|
|
|
this.mempoolChangedCallback(this.mempoolCache, [], []);
|
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-10-18 16:47:47 +02:00
|
|
|
public async $updateMemPoolInfo() {
|
2021-09-14 23:47:24 +02:00
|
|
|
this.mempoolInfo = await this.$getMempoolInfo();
|
2020-02-17 14:39:20 +01:00
|
|
|
}
|
|
|
|
|
2021-02-24 06:26:55 +01:00
|
|
|
public getMempoolInfo(): IBitcoinApi.MempoolInfo {
|
2019-07-21 16:59:47 +02:00
|
|
|
return this.mempoolInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
public getTxPerSecond(): number {
|
|
|
|
return this.txPerSecond;
|
|
|
|
}
|
|
|
|
|
|
|
|
public getVBytesPerSecond(): number {
|
|
|
|
return this.vBytesPerSecond;
|
|
|
|
}
|
|
|
|
|
2020-02-27 19:09:07 +01:00
|
|
|
public getFirstSeenForTransactions(txIds: string[]): number[] {
|
|
|
|
const txTimes: number[] = [];
|
|
|
|
txIds.forEach((txId: string) => {
|
2020-12-27 22:47:22 +01:00
|
|
|
const tx = this.mempoolCache[txId];
|
|
|
|
if (tx && tx.firstSeen) {
|
|
|
|
txTimes.push(tx.firstSeen);
|
2020-02-27 19:09:07 +01:00
|
|
|
} else {
|
|
|
|
txTimes.push(0);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return txTimes;
|
|
|
|
}
|
|
|
|
|
2020-10-18 16:47:47 +02:00
|
|
|
public async $updateMempool() {
|
2020-10-13 11:00:58 +02:00
|
|
|
logger.debug('Updating mempool');
|
2019-07-21 16:59:47 +02:00
|
|
|
const start = new Date().getTime();
|
|
|
|
let hasChange: boolean = false;
|
2020-06-18 08:54:54 +02:00
|
|
|
const currentMempoolSize = Object.keys(this.mempoolCache).length;
|
2019-07-21 16:59:47 +02:00
|
|
|
let txCount = 0;
|
2020-12-21 17:08:34 +01:00
|
|
|
const transactions = await bitcoinApi.$getRawMempool();
|
2020-10-18 16:47:47 +02:00
|
|
|
const diff = transactions.length - currentMempoolSize;
|
|
|
|
const newTransactions: TransactionExtended[] = [];
|
|
|
|
|
2021-01-05 12:57:06 +01:00
|
|
|
if (!this.inSync) {
|
|
|
|
loadingIndicators.setProgress('mempool', Object.keys(this.mempoolCache).length / transactions.length * 100);
|
|
|
|
}
|
|
|
|
|
2020-10-18 16:47:47 +02:00
|
|
|
for (const txid of transactions) {
|
|
|
|
if (!this.mempoolCache[txid]) {
|
2021-01-23 20:51:22 +01:00
|
|
|
try {
|
2021-01-23 22:15:06 +01:00
|
|
|
const transaction = await transactionUtils.$getTransactionExtended(txid);
|
2020-10-18 16:47:47 +02:00
|
|
|
this.mempoolCache[txid] = transaction;
|
|
|
|
txCount++;
|
|
|
|
if (this.inSync) {
|
|
|
|
this.txPerSecondArray.push(new Date().getTime());
|
|
|
|
this.vBytesPerSecondArray.push({
|
|
|
|
unixTime: new Date().getTime(),
|
|
|
|
vSize: transaction.vsize,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
hasChange = true;
|
|
|
|
if (diff > 0) {
|
|
|
|
logger.debug('Fetched transaction ' + txCount + ' / ' + diff);
|
2019-07-21 16:59:47 +02:00
|
|
|
} else {
|
2020-10-18 16:47:47 +02:00
|
|
|
logger.debug('Fetched transaction ' + txCount);
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
2020-10-18 16:47:47 +02:00
|
|
|
newTransactions.push(transaction);
|
2021-01-23 20:51:22 +01:00
|
|
|
} catch (e) {
|
2021-08-31 14:09:33 +02:00
|
|
|
logger.debug('Error finding transaction in mempool: ' + (e instanceof Error ? e.message : e));
|
2019-11-15 10:27:12 +01:00
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2021-01-06 16:49:28 +01:00
|
|
|
if ((new Date().getTime()) - start > Mempool.WEBSOCKET_REFRESH_RATE_MS) {
|
2020-10-18 16:47:47 +02:00
|
|
|
break;
|
2020-06-18 08:54:54 +02:00
|
|
|
}
|
2020-10-18 16:47:47 +02:00
|
|
|
}
|
2020-06-08 21:32:24 +02:00
|
|
|
|
2020-10-18 16:47:47 +02:00
|
|
|
// Prevent mempool from clear on bitcoind restart by delaying the deletion
|
2020-10-19 12:30:47 +02:00
|
|
|
if (this.mempoolProtection === 0
|
|
|
|
&& currentMempoolSize > 20000
|
|
|
|
&& transactions.length / currentMempoolSize <= 0.80
|
|
|
|
) {
|
2020-10-18 16:47:47 +02:00
|
|
|
this.mempoolProtection = 1;
|
|
|
|
this.inSync = false;
|
|
|
|
logger.warn(`Mempool clear protection triggered because transactions.length: ${transactions.length} and currentMempoolSize: ${currentMempoolSize}.`);
|
|
|
|
setTimeout(() => {
|
|
|
|
this.mempoolProtection = 2;
|
|
|
|
logger.warn('Mempool clear protection resumed.');
|
2021-02-14 14:32:00 +01:00
|
|
|
}, 1000 * 60 * config.MEMPOOL.CLEAR_PROTECTION_MINUTES);
|
2020-10-18 16:47:47 +02:00
|
|
|
}
|
2020-06-18 08:54:54 +02:00
|
|
|
|
2020-10-18 16:47:47 +02:00
|
|
|
const deletedTransactions: TransactionExtended[] = [];
|
|
|
|
|
|
|
|
if (this.mempoolProtection !== 1) {
|
|
|
|
this.mempoolProtection = 0;
|
|
|
|
// Index object for faster search
|
|
|
|
const transactionsObject = {};
|
|
|
|
transactions.forEach((txId) => transactionsObject[txId] = true);
|
|
|
|
|
2021-03-21 00:06:03 +01:00
|
|
|
// Flag transactions for lazy deletion
|
2020-10-18 16:47:47 +02:00
|
|
|
for (const tx in this.mempoolCache) {
|
2021-03-21 00:06:03 +01:00
|
|
|
if (!transactionsObject[tx] && !this.mempoolCache[tx].deleteAfter) {
|
2020-10-18 16:47:47 +02:00
|
|
|
deletedTransactions.push(this.mempoolCache[tx]);
|
2021-03-21 00:06:03 +01:00
|
|
|
this.mempoolCache[tx].deleteAfter = new Date().getTime() + Mempool.LAZY_DELETE_AFTER_SECONDS * 1000;
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
2020-06-08 13:55:53 +02:00
|
|
|
}
|
2020-10-18 16:47:47 +02:00
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
|
2020-10-18 16:47:47 +02:00
|
|
|
const newTransactionsStripped = newTransactions.map((tx) => Common.stripTransaction(tx));
|
|
|
|
this.latestTransactions = newTransactionsStripped.concat(this.latestTransactions).slice(0, 6);
|
2020-09-25 21:11:30 +02:00
|
|
|
|
2021-03-21 00:06:03 +01:00
|
|
|
if (!this.inSync && transactions.length === Object.keys(this.mempoolCache).length) {
|
2020-10-18 16:47:47 +02:00
|
|
|
this.inSync = true;
|
2021-04-13 07:03:36 +02:00
|
|
|
logger.notice('The mempool is now in sync!');
|
2021-01-05 12:57:06 +01:00
|
|
|
loadingIndicators.setProgress('mempool', 100);
|
2020-10-18 16:47:47 +02:00
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
|
2020-10-18 16:47:47 +02:00
|
|
|
if (this.mempoolChangedCallback && (hasChange || deletedTransactions.length)) {
|
|
|
|
this.mempoolChangedCallback(this.mempoolCache, newTransactions, deletedTransactions);
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
2020-10-18 16:47:47 +02:00
|
|
|
|
|
|
|
const end = new Date().getTime();
|
|
|
|
const time = end - start;
|
2021-03-21 00:06:03 +01:00
|
|
|
logger.debug(`New mempool size: ${Object.keys(this.mempoolCache).length} Change: ${diff}`);
|
2020-10-18 16:47:47 +02:00
|
|
|
logger.debug('Mempool updated in ' + time / 1000 + ' seconds');
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private updateTxPerSecond() {
|
2020-10-19 06:57:02 +02:00
|
|
|
const nowMinusTimeSpan = new Date().getTime() - (1000 * config.STATISTICS.TX_PER_SECOND_SAMPLE_PERIOD);
|
2019-07-21 16:59:47 +02:00
|
|
|
this.txPerSecondArray = this.txPerSecondArray.filter((unixTime) => unixTime > nowMinusTimeSpan);
|
2020-10-19 06:57:02 +02:00
|
|
|
this.txPerSecond = this.txPerSecondArray.length / config.STATISTICS.TX_PER_SECOND_SAMPLE_PERIOD || 0;
|
2019-07-21 16:59:47 +02:00
|
|
|
|
|
|
|
this.vBytesPerSecondArray = this.vBytesPerSecondArray.filter((data) => data.unixTime > nowMinusTimeSpan);
|
|
|
|
if (this.vBytesPerSecondArray.length) {
|
|
|
|
this.vBytesPerSecond = Math.round(
|
2020-10-19 06:57:02 +02:00
|
|
|
this.vBytesPerSecondArray.map((data) => data.vSize).reduce((a, b) => a + b) / config.STATISTICS.TX_PER_SECOND_SAMPLE_PERIOD
|
2019-07-21 16:59:47 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2021-03-21 00:06:03 +01:00
|
|
|
|
|
|
|
private deleteExpiredTransactions() {
|
|
|
|
const now = new Date().getTime();
|
|
|
|
for (const tx in this.mempoolCache) {
|
|
|
|
const lazyDeleteAt = this.mempoolCache[tx].deleteAfter;
|
|
|
|
if (lazyDeleteAt && lazyDeleteAt < now) {
|
|
|
|
delete this.mempoolCache[tx];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-09-14 23:47:24 +02:00
|
|
|
|
|
|
|
private $getMempoolInfo() {
|
2021-09-19 00:40:16 +02:00
|
|
|
if (config.MEMPOOL.USE_SECOND_NODE_FOR_MINFEE) {
|
2021-09-14 23:47:24 +02:00
|
|
|
return Promise.all([
|
|
|
|
bitcoinClient.getMempoolInfo(),
|
|
|
|
bitcoinSecondClient.getMempoolInfo()
|
|
|
|
]).then(([mempoolInfo, secondMempoolInfo]) => {
|
|
|
|
mempoolInfo.maxmempool = secondMempoolInfo.maxmempool;
|
|
|
|
mempoolInfo.mempoolminfee = secondMempoolInfo.mempoolminfee;
|
|
|
|
mempoolInfo.minrelaytxfee = secondMempoolInfo.minrelaytxfee;
|
|
|
|
return mempoolInfo;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return bitcoinClient.getMempoolInfo();
|
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export default new Mempool();
|