2021-03-18 17:47:40 +01:00
|
|
|
|
import { CpfpInfo, TransactionExtended, TransactionStripped } from '../mempool.interfaces';
|
2021-08-05 01:03:52 +02:00
|
|
|
|
import config from '../config';
|
2020-05-24 11:29:30 +02:00
|
|
|
|
export class Common {
|
2021-12-28 14:59:11 +01:00
|
|
|
|
static nativeAssetId = config.MEMPOOL.NETWORK === 'liquidtestnet' ?
|
|
|
|
|
'144c654344aa716d6f3abcc1ca90e5641e4e2a7f633bc09fe3baf64585819a49'
|
|
|
|
|
: '6f0279e9ed041c3d710a9f57d0c02928416460c4b722ae3457a11eec381c526d';
|
2021-12-30 23:28:40 +01:00
|
|
|
|
static _isLiquid = config.MEMPOOL.NETWORK === 'liquid' || config.MEMPOOL.NETWORK === 'liquidtestnet';
|
|
|
|
|
|
|
|
|
|
static isLiquid(): boolean {
|
|
|
|
|
return this._isLiquid;
|
|
|
|
|
}
|
2021-09-18 11:37:25 +02:00
|
|
|
|
|
2020-05-24 11:29:30 +02:00
|
|
|
|
static median(numbers: number[]) {
|
|
|
|
|
let medianNr = 0;
|
|
|
|
|
const numsLen = numbers.length;
|
|
|
|
|
if (numsLen % 2 === 0) {
|
|
|
|
|
medianNr = (numbers[numsLen / 2 - 1] + numbers[numsLen / 2]) / 2;
|
|
|
|
|
} else {
|
|
|
|
|
medianNr = numbers[(numsLen - 1) / 2];
|
|
|
|
|
}
|
|
|
|
|
return medianNr;
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-18 06:52:46 +01:00
|
|
|
|
static percentile(numbers: number[], percentile: number) {
|
2021-04-10 19:26:05 +02:00
|
|
|
|
if (percentile === 50) {
|
|
|
|
|
return this.median(numbers);
|
|
|
|
|
}
|
2021-03-18 06:52:46 +01:00
|
|
|
|
const index = Math.ceil(numbers.length * (100 - percentile) * 1e-2);
|
2021-04-10 19:26:05 +02:00
|
|
|
|
if (index < 0 || index > numbers.length - 1) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2021-03-18 06:52:46 +01:00
|
|
|
|
return numbers[index];
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-23 19:20:59 +02:00
|
|
|
|
static getFeesInRange(transactions: TransactionExtended[], rangeLength: number) {
|
2021-03-18 17:47:40 +01:00
|
|
|
|
const arr = [transactions[transactions.length - 1].effectiveFeePerVsize];
|
2020-05-24 11:29:30 +02:00
|
|
|
|
const chunk = 1 / (rangeLength - 1);
|
|
|
|
|
let itemsToAdd = rangeLength - 2;
|
|
|
|
|
|
|
|
|
|
while (itemsToAdd > 0) {
|
2021-03-18 17:47:40 +01:00
|
|
|
|
arr.push(transactions[Math.floor(transactions.length * chunk * itemsToAdd)].effectiveFeePerVsize);
|
2020-05-24 11:29:30 +02:00
|
|
|
|
itemsToAdd--;
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-18 17:47:40 +01:00
|
|
|
|
arr.push(transactions[0].effectiveFeePerVsize);
|
2020-05-24 11:29:30 +02:00
|
|
|
|
return arr;
|
|
|
|
|
}
|
2020-06-08 13:55:53 +02:00
|
|
|
|
|
|
|
|
|
static findRbfTransactions(added: TransactionExtended[], deleted: TransactionExtended[]): { [txid: string]: TransactionExtended } {
|
|
|
|
|
const matches: { [txid: string]: TransactionExtended } = {};
|
|
|
|
|
deleted
|
|
|
|
|
// The replaced tx must have at least one input with nSequence < maxint-1 (That’s the opt-in)
|
|
|
|
|
.filter((tx) => tx.vin.some((vin) => vin.sequence < 0xfffffffe))
|
|
|
|
|
.forEach((deletedTx) => {
|
|
|
|
|
const foundMatches = added.find((addedTx) => {
|
|
|
|
|
// The new tx must, absolutely speaking, pay at least as much fee as the replaced tx.
|
|
|
|
|
return addedTx.fee > deletedTx.fee
|
|
|
|
|
// The new transaction must pay more fee per kB than the replaced tx.
|
|
|
|
|
&& addedTx.feePerVsize > deletedTx.feePerVsize
|
|
|
|
|
// Spends one or more of the same inputs
|
|
|
|
|
&& deletedTx.vin.some((deletedVin) =>
|
|
|
|
|
addedTx.vin.some((vin) => vin.txid === deletedVin.txid));
|
|
|
|
|
});
|
|
|
|
|
if (foundMatches) {
|
|
|
|
|
matches[deletedTx.txid] = foundMatches;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return matches;
|
|
|
|
|
}
|
2020-09-25 21:11:30 +02:00
|
|
|
|
|
|
|
|
|
static stripTransaction(tx: TransactionExtended): TransactionStripped {
|
|
|
|
|
return {
|
|
|
|
|
txid: tx.txid,
|
|
|
|
|
fee: tx.fee,
|
2021-01-24 19:09:42 +01:00
|
|
|
|
vsize: tx.weight / 4,
|
2020-12-21 17:08:34 +01:00
|
|
|
|
value: tx.vout.reduce((acc, vout) => acc + (vout.value ? vout.value : 0), 0),
|
2020-09-25 21:11:30 +02:00
|
|
|
|
};
|
|
|
|
|
}
|
2020-12-28 14:17:32 +01:00
|
|
|
|
|
|
|
|
|
static sleep(ms: number): Promise<void> {
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
resolve();
|
|
|
|
|
}, ms);
|
|
|
|
|
});
|
|
|
|
|
}
|
2021-04-10 19:26:05 +02:00
|
|
|
|
|
2021-03-19 07:47:37 +01:00
|
|
|
|
static shuffleArray(array: any[]) {
|
|
|
|
|
for (let i = array.length - 1; i > 0; i--) {
|
|
|
|
|
const j = Math.floor(Math.random() * (i + 1));
|
|
|
|
|
[array[i], array[j]] = [array[j], array[i]];
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-03-18 17:47:40 +01:00
|
|
|
|
|
|
|
|
|
static setRelativesAndGetCpfpInfo(tx: TransactionExtended, memPool: { [txid: string]: TransactionExtended }): CpfpInfo {
|
|
|
|
|
const parents = this.findAllParents(tx, memPool);
|
2021-04-10 19:26:05 +02:00
|
|
|
|
const lowerFeeParents = parents.filter((parent) => parent.feePerVsize < tx.effectiveFeePerVsize);
|
2021-03-18 17:47:40 +01:00
|
|
|
|
|
2021-04-05 21:45:47 +02:00
|
|
|
|
let totalWeight = tx.weight + lowerFeeParents.reduce((prev, val) => prev + val.weight, 0);
|
|
|
|
|
let totalFees = tx.fee + lowerFeeParents.reduce((prev, val) => prev + val.fee, 0);
|
2021-03-18 17:47:40 +01:00
|
|
|
|
|
|
|
|
|
tx.ancestors = parents
|
|
|
|
|
.map((t) => {
|
|
|
|
|
return {
|
|
|
|
|
txid: t.txid,
|
|
|
|
|
weight: t.weight,
|
|
|
|
|
fee: t.fee,
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Add high (high fee) decendant weight and fees
|
|
|
|
|
if (tx.bestDescendant) {
|
|
|
|
|
totalWeight += tx.bestDescendant.weight;
|
|
|
|
|
totalFees += tx.bestDescendant.fee;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-30 23:28:40 +01:00
|
|
|
|
tx.effectiveFeePerVsize = Math.max(Common.isLiquid() ? 0.1 : 1, totalFees / (totalWeight / 4));
|
2021-03-18 17:47:40 +01:00
|
|
|
|
tx.cpfpChecked = true;
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
ancestors: tx.ancestors,
|
|
|
|
|
bestDescendant: tx.bestDescendant || null,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private static findAllParents(tx: TransactionExtended, memPool: { [txid: string]: TransactionExtended }): TransactionExtended[] {
|
|
|
|
|
let parents: TransactionExtended[] = [];
|
|
|
|
|
tx.vin.forEach((parent) => {
|
2021-04-01 22:30:51 +02:00
|
|
|
|
if (parents.find((p) => p.txid === parent.txid)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-18 17:47:40 +01:00
|
|
|
|
const parentTx = memPool[parent.txid];
|
|
|
|
|
if (parentTx) {
|
|
|
|
|
if (tx.bestDescendant && tx.bestDescendant.fee / (tx.bestDescendant.weight / 4) > parentTx.feePerVsize) {
|
|
|
|
|
if (parentTx.bestDescendant && parentTx.bestDescendant.fee < tx.fee + tx.bestDescendant.fee) {
|
|
|
|
|
parentTx.bestDescendant = {
|
|
|
|
|
weight: tx.weight + tx.bestDescendant.weight,
|
|
|
|
|
fee: tx.fee + tx.bestDescendant.fee,
|
|
|
|
|
txid: tx.txid,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
} else if (tx.feePerVsize > parentTx.feePerVsize) {
|
|
|
|
|
parentTx.bestDescendant = {
|
|
|
|
|
weight: tx.weight,
|
|
|
|
|
fee: tx.fee,
|
|
|
|
|
txid: tx.txid
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
parents.push(parentTx);
|
|
|
|
|
parents = parents.concat(this.findAllParents(parentTx, memPool));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return parents;
|
|
|
|
|
}
|
2022-02-09 11:41:05 +01:00
|
|
|
|
|
|
|
|
|
static getSqlInterval(interval: string | null): string | null {
|
|
|
|
|
switch (interval) {
|
|
|
|
|
case '24h': return '1 DAY';
|
|
|
|
|
case '3d': return '3 DAY';
|
|
|
|
|
case '1w': return '1 WEEK';
|
|
|
|
|
case '1m': return '1 MONTH';
|
|
|
|
|
case '3m': return '3 MONTH';
|
|
|
|
|
case '6m': return '6 MONTH';
|
|
|
|
|
case '1y': return '1 YEAR';
|
|
|
|
|
case '2y': return '2 YEAR';
|
|
|
|
|
case '3y': return '3 YEAR';
|
|
|
|
|
default: return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-05-24 11:29:30 +02:00
|
|
|
|
}
|