Merge pull request #2793 from mempool/mononaut/gbt-thread-optimization
Refactor advanced GBT implementation to minimize inter-thread data transfer
This commit is contained in:
		
						commit
						606e6df834
					
				| @ -1,17 +1,14 @@ | |||||||
| import logger from '../logger'; | import logger from '../logger'; | ||||||
| import { MempoolBlock, TransactionExtended, TransactionStripped, MempoolBlockWithTransactions, MempoolBlockDelta } from '../mempool.interfaces'; | import { MempoolBlock, TransactionExtended, ThreadTransaction, TransactionStripped, MempoolBlockWithTransactions, MempoolBlockDelta, Ancestor } from '../mempool.interfaces'; | ||||||
| import { Common } from './common'; | import { Common } from './common'; | ||||||
| import config from '../config'; | import config from '../config'; | ||||||
| import { StaticPool } from 'node-worker-threads-pool'; | import { Worker } from 'worker_threads'; | ||||||
| import path from 'path'; | import path from 'path'; | ||||||
| 
 | 
 | ||||||
| class MempoolBlocks { | class MempoolBlocks { | ||||||
|   private mempoolBlocks: MempoolBlockWithTransactions[] = []; |   private mempoolBlocks: MempoolBlockWithTransactions[] = []; | ||||||
|   private mempoolBlockDeltas: MempoolBlockDelta[] = []; |   private mempoolBlockDeltas: MempoolBlockDelta[] = []; | ||||||
|   private makeTemplatesPool = new StaticPool({ |   private txSelectionWorker: Worker | null = null; | ||||||
|     size: 1, |  | ||||||
|     task: path.resolve(__dirname, './tx-selection-worker.js'), |  | ||||||
|   }); |  | ||||||
| 
 | 
 | ||||||
|   constructor() {} |   constructor() {} | ||||||
| 
 | 
 | ||||||
| @ -146,27 +143,159 @@ class MempoolBlocks { | |||||||
|     return mempoolBlockDeltas; |     return mempoolBlockDeltas; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   public async makeBlockTemplates(newMempool: { [txid: string]: TransactionExtended }, blockLimit: number, weightLimit: number | null = null, condenseRest = false): Promise<void> { |   public async makeBlockTemplates(newMempool: { [txid: string]: TransactionExtended }): Promise<void> { | ||||||
|     const { mempool, blocks } = await this.makeTemplatesPool.exec({ mempool: newMempool, blockLimit, weightLimit, condenseRest }); |     // prepare a stripped down version of the mempool with only the minimum necessary data
 | ||||||
|     const deltas = this.calculateMempoolDeltas(this.mempoolBlocks, blocks); |     // to reduce the overhead of passing this data to the worker thread
 | ||||||
| 
 |     const strippedMempool: { [txid: string]: ThreadTransaction } = {}; | ||||||
|     // copy CPFP info across to main thread's mempool
 |     Object.values(newMempool).forEach(entry => { | ||||||
|     Object.keys(newMempool).forEach((txid) => { |       strippedMempool[entry.txid] = { | ||||||
|       if (newMempool[txid] && mempool[txid]) { |         txid: entry.txid, | ||||||
|         newMempool[txid].effectiveFeePerVsize = mempool[txid].effectiveFeePerVsize; |         fee: entry.fee, | ||||||
|         newMempool[txid].ancestors = mempool[txid].ancestors; |         weight: entry.weight, | ||||||
|         newMempool[txid].descendants = mempool[txid].descendants; |         feePerVsize: entry.fee / (entry.weight / 4), | ||||||
|         newMempool[txid].bestDescendant = mempool[txid].bestDescendant; |         effectiveFeePerVsize: entry.fee / (entry.weight / 4), | ||||||
|         newMempool[txid].cpfpChecked = mempool[txid].cpfpChecked; |         vin: entry.vin.map(v => v.txid), | ||||||
|       } |       }; | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     this.mempoolBlocks = blocks; |     // (re)initialize tx selection worker thread
 | ||||||
|  |     if (!this.txSelectionWorker) { | ||||||
|  |       this.txSelectionWorker = new Worker(path.resolve(__dirname, './tx-selection-worker.js')); | ||||||
|  |       // if the thread throws an unexpected error, or exits for any other reason,
 | ||||||
|  |       // reset worker state so that it will be re-initialized on the next run
 | ||||||
|  |       this.txSelectionWorker.once('error', () => { | ||||||
|  |         this.txSelectionWorker = null; | ||||||
|  |       }); | ||||||
|  |       this.txSelectionWorker.once('exit', () => { | ||||||
|  |         this.txSelectionWorker = null; | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // run the block construction algorithm in a separate thread, and wait for a result
 | ||||||
|  |     let threadErrorListener; | ||||||
|  |     try { | ||||||
|  |       const workerResultPromise = new Promise<{ blocks: ThreadTransaction[][], clusters: { [root: string]: string[] } }>((resolve, reject) => { | ||||||
|  |         threadErrorListener = reject; | ||||||
|  |         this.txSelectionWorker?.once('message', (result): void => { | ||||||
|  |           resolve(result); | ||||||
|  |         }); | ||||||
|  |         this.txSelectionWorker?.once('error', reject); | ||||||
|  |       }); | ||||||
|  |       this.txSelectionWorker.postMessage({ type: 'set', mempool: strippedMempool }); | ||||||
|  |       const { blocks, clusters } = await workerResultPromise; | ||||||
|  | 
 | ||||||
|  |       this.processBlockTemplates(newMempool, blocks, clusters); | ||||||
|  | 
 | ||||||
|  |       // clean up thread error listener
 | ||||||
|  |       this.txSelectionWorker?.removeListener('error', threadErrorListener); | ||||||
|  |     } catch (e) { | ||||||
|  |       logger.err('makeBlockTemplates failed. ' + (e instanceof Error ? e.message : e)); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   public async updateBlockTemplates(newMempool: { [txid: string]: TransactionExtended }, added: TransactionExtended[], removed: string[]): Promise<void> { | ||||||
|  |     if (!this.txSelectionWorker) { | ||||||
|  |       // need to reset the worker
 | ||||||
|  |       return this.makeBlockTemplates(newMempool); | ||||||
|  |     } | ||||||
|  |     // prepare a stripped down version of the mempool with only the minimum necessary data
 | ||||||
|  |     // to reduce the overhead of passing this data to the worker thread
 | ||||||
|  |     const addedStripped: ThreadTransaction[] = added.map(entry => { | ||||||
|  |       return { | ||||||
|  |         txid: entry.txid, | ||||||
|  |         fee: entry.fee, | ||||||
|  |         weight: entry.weight, | ||||||
|  |         feePerVsize: entry.fee / (entry.weight / 4), | ||||||
|  |         effectiveFeePerVsize: entry.fee / (entry.weight / 4), | ||||||
|  |         vin: entry.vin.map(v => v.txid), | ||||||
|  |       }; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     // run the block construction algorithm in a separate thread, and wait for a result
 | ||||||
|  |     let threadErrorListener; | ||||||
|  |     try { | ||||||
|  |       const workerResultPromise = new Promise<{ blocks: ThreadTransaction[][], clusters: { [root: string]: string[] } }>((resolve, reject) => { | ||||||
|  |         threadErrorListener = reject; | ||||||
|  |         this.txSelectionWorker?.once('message', (result): void => { | ||||||
|  |           resolve(result); | ||||||
|  |         }); | ||||||
|  |         this.txSelectionWorker?.once('error', reject); | ||||||
|  |       }); | ||||||
|  |       this.txSelectionWorker.postMessage({ type: 'update', added: addedStripped, removed }); | ||||||
|  |       const { blocks, clusters } = await workerResultPromise; | ||||||
|  | 
 | ||||||
|  |       this.processBlockTemplates(newMempool, blocks, clusters); | ||||||
|  | 
 | ||||||
|  |       // clean up thread error listener
 | ||||||
|  |       this.txSelectionWorker?.removeListener('error', threadErrorListener); | ||||||
|  |     } catch (e) { | ||||||
|  |       logger.err('updateBlockTemplates failed. ' + (e instanceof Error ? e.message : e)); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   private processBlockTemplates(mempool, blocks, clusters): void { | ||||||
|  |     // update this thread's mempool with the results
 | ||||||
|  |     blocks.forEach(block => { | ||||||
|  |       block.forEach(tx => { | ||||||
|  |         if (tx.txid in mempool) { | ||||||
|  |           if (tx.effectiveFeePerVsize != null) { | ||||||
|  |             mempool[tx.txid].effectiveFeePerVsize = tx.effectiveFeePerVsize; | ||||||
|  |           } | ||||||
|  |           if (tx.cpfpRoot && tx.cpfpRoot in clusters) { | ||||||
|  |             const ancestors: Ancestor[] = []; | ||||||
|  |             const descendants: Ancestor[] = []; | ||||||
|  |             const cluster = clusters[tx.cpfpRoot]; | ||||||
|  |             let matched = false; | ||||||
|  |             cluster.forEach(txid => { | ||||||
|  |               if (txid === tx.txid) { | ||||||
|  |                 matched = true; | ||||||
|  |               } else { | ||||||
|  |                 const relative = { | ||||||
|  |                   txid: txid, | ||||||
|  |                   fee: mempool[txid].fee, | ||||||
|  |                   weight: mempool[txid].weight, | ||||||
|  |                 }; | ||||||
|  |                 if (matched) { | ||||||
|  |                   descendants.push(relative); | ||||||
|  |                 } else { | ||||||
|  |                   ancestors.push(relative); | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             }); | ||||||
|  |             mempool[tx.txid].ancestors = ancestors; | ||||||
|  |             mempool[tx.txid].descendants = descendants; | ||||||
|  |             mempool[tx.txid].bestDescendant = null; | ||||||
|  |           } | ||||||
|  |           mempool[tx.txid].cpfpChecked = tx.cpfpChecked; | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     // unpack the condensed blocks into proper mempool blocks
 | ||||||
|  |     const mempoolBlocks = blocks.map((transactions, blockIndex) => { | ||||||
|  |       return this.dataToMempoolBlocks(transactions.map(tx => { | ||||||
|  |         return mempool[tx.txid] || null; | ||||||
|  |       }).filter(tx => !!tx), undefined, undefined, blockIndex); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     const deltas = this.calculateMempoolDeltas(this.mempoolBlocks, mempoolBlocks); | ||||||
|  | 
 | ||||||
|  |     this.mempoolBlocks = mempoolBlocks; | ||||||
|     this.mempoolBlockDeltas = deltas; |     this.mempoolBlockDeltas = deltas; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   private dataToMempoolBlocks(transactions: TransactionExtended[], |   private dataToMempoolBlocks(transactions: TransactionExtended[], | ||||||
|     blockSize: number, blockWeight: number, blocksIndex: number): MempoolBlockWithTransactions { |     blockSize: number | undefined, blockWeight: number | undefined, blocksIndex: number): MempoolBlockWithTransactions { | ||||||
|  |     let totalSize = blockSize || 0; | ||||||
|  |     let totalWeight = blockWeight || 0; | ||||||
|  |     if (blockSize === undefined && blockWeight === undefined) { | ||||||
|  |       totalSize = 0; | ||||||
|  |       totalWeight = 0; | ||||||
|  |       transactions.forEach(tx => { | ||||||
|  |         totalSize += tx.size; | ||||||
|  |         totalWeight += tx.weight; | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|     let rangeLength = 4; |     let rangeLength = 4; | ||||||
|     if (blocksIndex === 0) { |     if (blocksIndex === 0) { | ||||||
|       rangeLength = 8; |       rangeLength = 8; | ||||||
| @ -177,8 +306,8 @@ class MempoolBlocks { | |||||||
|       rangeLength = 8; |       rangeLength = 8; | ||||||
|     } |     } | ||||||
|     return { |     return { | ||||||
|       blockSize: blockSize, |       blockSize: totalSize, | ||||||
|       blockVSize: blockWeight / 4, |       blockVSize: totalWeight / 4, | ||||||
|       nTx: transactions.length, |       nTx: transactions.length, | ||||||
|       totalFees: transactions.reduce((acc, cur) => acc + cur.fee, 0), |       totalFees: transactions.reduce((acc, cur) => acc + cur.fee, 0), | ||||||
|       medianFee: Common.percentile(transactions.map((tx) => tx.effectiveFeePerVsize), config.MEMPOOL.RECOMMENDED_FEE_PERCENTILE), |       medianFee: Common.percentile(transactions.map((tx) => tx.effectiveFeePerVsize), config.MEMPOOL.RECOMMENDED_FEE_PERCENTILE), | ||||||
|  | |||||||
| @ -21,7 +21,7 @@ class Mempool { | |||||||
|   private mempoolChangedCallback: ((newMempool: {[txId: string]: TransactionExtended; }, newTransactions: TransactionExtended[], |   private mempoolChangedCallback: ((newMempool: {[txId: string]: TransactionExtended; }, newTransactions: TransactionExtended[], | ||||||
|     deletedTransactions: TransactionExtended[]) => void) | undefined; |     deletedTransactions: TransactionExtended[]) => void) | undefined; | ||||||
|   private asyncMempoolChangedCallback: ((newMempool: {[txId: string]: TransactionExtended; }, newTransactions: TransactionExtended[], |   private asyncMempoolChangedCallback: ((newMempool: {[txId: string]: TransactionExtended; }, newTransactions: TransactionExtended[], | ||||||
|     deletedTransactions: TransactionExtended[]) => void) | undefined; |     deletedTransactions: TransactionExtended[]) => Promise<void>) | undefined; | ||||||
| 
 | 
 | ||||||
|   private txPerSecondArray: number[] = []; |   private txPerSecondArray: number[] = []; | ||||||
|   private txPerSecond: number = 0; |   private txPerSecond: number = 0; | ||||||
|  | |||||||
| @ -1,17 +1,30 @@ | |||||||
| import config from '../config'; | import config from '../config'; | ||||||
| import logger from '../logger'; | import logger from '../logger'; | ||||||
| import { TransactionExtended, MempoolBlockWithTransactions, AuditTransaction } from '../mempool.interfaces'; | import { ThreadTransaction, MempoolBlockWithTransactions, AuditTransaction } from '../mempool.interfaces'; | ||||||
| import { PairingHeap } from '../utils/pairing-heap'; | import { PairingHeap } from '../utils/pairing-heap'; | ||||||
| import { Common } from './common'; | import { Common } from './common'; | ||||||
| import { parentPort } from 'worker_threads'; | import { parentPort } from 'worker_threads'; | ||||||
| 
 | 
 | ||||||
|  | let mempool: { [txid: string]: ThreadTransaction } = {}; | ||||||
|  | 
 | ||||||
| if (parentPort) { | if (parentPort) { | ||||||
|   parentPort.on('message', (params: { mempool: { [txid: string]: TransactionExtended }, blockLimit: number, weightLimit: number | null, condenseRest: boolean}) => { |   parentPort.on('message', (params) => { | ||||||
|     const { mempool, blocks } = makeBlockTemplates(params); |     if (params.type === 'set') { | ||||||
|  |       mempool = params.mempool; | ||||||
|  |     } else if (params.type === 'update') { | ||||||
|  |       params.added.forEach(tx => { | ||||||
|  |         mempool[tx.txid] = tx; | ||||||
|  |       }); | ||||||
|  |       params.removed.forEach(txid => { | ||||||
|  |         delete mempool[txid]; | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const { blocks, clusters } = makeBlockTemplates(mempool); | ||||||
| 
 | 
 | ||||||
|     // return the result to main thread.
 |     // return the result to main thread.
 | ||||||
|     if (parentPort) { |     if (parentPort) { | ||||||
|      parentPort.postMessage({ mempool, blocks }); |      parentPort.postMessage({ blocks, clusters }); | ||||||
|     } |     } | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
| @ -19,35 +32,24 @@ if (parentPort) { | |||||||
| /* | /* | ||||||
| * Build projected mempool blocks using an approximation of the transaction selection algorithm from Bitcoin Core | * Build projected mempool blocks using an approximation of the transaction selection algorithm from Bitcoin Core | ||||||
| * (see BlockAssembler in https://github.com/bitcoin/bitcoin/blob/master/src/node/miner.cpp)
 | * (see BlockAssembler in https://github.com/bitcoin/bitcoin/blob/master/src/node/miner.cpp)
 | ||||||
| * |  | ||||||
| * blockLimit: number of blocks to build in total. |  | ||||||
| * weightLimit: maximum weight of transactions to consider using the selection algorithm. |  | ||||||
| *              if weightLimit is significantly lower than the mempool size, results may start to diverge from getBlockTemplate  |  | ||||||
| * condenseRest: whether to ignore excess transactions or append them to the final block. |  | ||||||
| */ | */ | ||||||
| function makeBlockTemplates({ mempool, blockLimit, weightLimit, condenseRest }: { mempool: { [txid: string]: TransactionExtended }, blockLimit: number, weightLimit?: number | null, condenseRest?: boolean | null }) | function makeBlockTemplates(mempool: { [txid: string]: ThreadTransaction }) | ||||||
|   : { mempool: { [txid: string]: TransactionExtended }, blocks: MempoolBlockWithTransactions[] } { |   : { blocks: ThreadTransaction[][], clusters: { [root: string]: string[] } } { | ||||||
|   const start = Date.now(); |   const start = Date.now(); | ||||||
|   const auditPool: { [txid: string]: AuditTransaction } = {}; |   const auditPool: { [txid: string]: AuditTransaction } = {}; | ||||||
|   const mempoolArray: AuditTransaction[] = []; |   const mempoolArray: AuditTransaction[] = []; | ||||||
|   const restOfArray: TransactionExtended[] = []; |   const restOfArray: ThreadTransaction[] = []; | ||||||
|  |   const cpfpClusters: { [root: string]: string[] } = {}; | ||||||
|    |    | ||||||
|   let weight = 0; |  | ||||||
|   const maxWeight = weightLimit ? Math.max(4_000_000 * blockLimit, weightLimit) : Infinity; |  | ||||||
|   // grab the top feerate txs up to maxWeight
 |   // grab the top feerate txs up to maxWeight
 | ||||||
|   Object.values(mempool).sort((a, b) => b.feePerVsize - a.feePerVsize).forEach(tx => { |   Object.values(mempool).sort((a, b) => b.feePerVsize - a.feePerVsize).forEach(tx => { | ||||||
|     weight += tx.weight; |  | ||||||
|     if (weight >= maxWeight) { |  | ||||||
|       restOfArray.push(tx); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     // initializing everything up front helps V8 optimize property access later
 |     // initializing everything up front helps V8 optimize property access later
 | ||||||
|     auditPool[tx.txid] = { |     auditPool[tx.txid] = { | ||||||
|       txid: tx.txid, |       txid: tx.txid, | ||||||
|       fee: tx.fee, |       fee: tx.fee, | ||||||
|       size: tx.size, |  | ||||||
|       weight: tx.weight, |       weight: tx.weight, | ||||||
|       feePerVsize: tx.feePerVsize, |       feePerVsize: tx.feePerVsize, | ||||||
|  |       effectiveFeePerVsize: tx.feePerVsize, | ||||||
|       vin: tx.vin, |       vin: tx.vin, | ||||||
|       relativesSet: false, |       relativesSet: false, | ||||||
|       ancestorMap: new Map<string, AuditTransaction>(), |       ancestorMap: new Map<string, AuditTransaction>(), | ||||||
| @ -74,7 +76,7 @@ function makeBlockTemplates({ mempool, blockLimit, weightLimit, condenseRest }: | |||||||
| 
 | 
 | ||||||
|   // Build blocks by greedily choosing the highest feerate package
 |   // Build blocks by greedily choosing the highest feerate package
 | ||||||
|   // (i.e. the package rooted in the transaction with the best ancestor score)
 |   // (i.e. the package rooted in the transaction with the best ancestor score)
 | ||||||
|   const blocks: MempoolBlockWithTransactions[] = []; |   const blocks: ThreadTransaction[][] = []; | ||||||
|   let blockWeight = 4000; |   let blockWeight = 4000; | ||||||
|   let blockSize = 0; |   let blockSize = 0; | ||||||
|   let transactions: AuditTransaction[] = []; |   let transactions: AuditTransaction[] = []; | ||||||
| @ -82,7 +84,7 @@ function makeBlockTemplates({ mempool, blockLimit, weightLimit, condenseRest }: | |||||||
|   let overflow: AuditTransaction[] = []; |   let overflow: AuditTransaction[] = []; | ||||||
|   let failures = 0; |   let failures = 0; | ||||||
|   let top = 0; |   let top = 0; | ||||||
|   while ((top < mempoolArray.length || !modified.isEmpty()) && (condenseRest || blocks.length < blockLimit)) { |   while ((top < mempoolArray.length || !modified.isEmpty())) { | ||||||
|     // skip invalid transactions
 |     // skip invalid transactions
 | ||||||
|     while (top < mempoolArray.length && (mempoolArray[top].used || mempoolArray[top].modified)) { |     while (top < mempoolArray.length && (mempoolArray[top].used || mempoolArray[top].modified)) { | ||||||
|       top++; |       top++; | ||||||
| @ -107,9 +109,13 @@ function makeBlockTemplates({ mempool, blockLimit, weightLimit, condenseRest }: | |||||||
|       // Check if the package fits into this block
 |       // Check if the package fits into this block
 | ||||||
|       if (blockWeight + nextTx.ancestorWeight < config.MEMPOOL.BLOCK_WEIGHT_UNITS) { |       if (blockWeight + nextTx.ancestorWeight < config.MEMPOOL.BLOCK_WEIGHT_UNITS) { | ||||||
|         const ancestors: AuditTransaction[] = Array.from(nextTx.ancestorMap.values()); |         const ancestors: AuditTransaction[] = Array.from(nextTx.ancestorMap.values()); | ||||||
|         const descendants: AuditTransaction[] = []; |  | ||||||
|         // sort ancestors by dependency graph (equivalent to sorting by ascending ancestor count)
 |         // sort ancestors by dependency graph (equivalent to sorting by ascending ancestor count)
 | ||||||
|         const sortedTxSet = [...ancestors.sort((a, b) => { return (a.ancestorMap.size || 0) - (b.ancestorMap.size || 0); }), nextTx]; |         const sortedTxSet = [...ancestors.sort((a, b) => { return (a.ancestorMap.size || 0) - (b.ancestorMap.size || 0); }), nextTx]; | ||||||
|  |         let isCluster = false; | ||||||
|  |         if (sortedTxSet.length > 1) { | ||||||
|  |           cpfpClusters[nextTx.txid] = sortedTxSet.map(tx => tx.txid); | ||||||
|  |           isCluster = true; | ||||||
|  |         } | ||||||
|         const effectiveFeeRate = nextTx.ancestorFee / (nextTx.ancestorWeight / 4); |         const effectiveFeeRate = nextTx.ancestorFee / (nextTx.ancestorWeight / 4); | ||||||
|         const used: AuditTransaction[] = []; |         const used: AuditTransaction[] = []; | ||||||
|         while (sortedTxSet.length) { |         while (sortedTxSet.length) { | ||||||
| @ -119,21 +125,9 @@ function makeBlockTemplates({ mempool, blockLimit, weightLimit, condenseRest }: | |||||||
|           ancestor.usedBy = nextTx.txid; |           ancestor.usedBy = nextTx.txid; | ||||||
|           // update original copy of this tx with effective fee rate & relatives data
 |           // update original copy of this tx with effective fee rate & relatives data
 | ||||||
|           mempoolTx.effectiveFeePerVsize = effectiveFeeRate; |           mempoolTx.effectiveFeePerVsize = effectiveFeeRate; | ||||||
|           mempoolTx.ancestors = sortedTxSet.map((a) => { |           if (isCluster) { | ||||||
|             return { |             mempoolTx.cpfpRoot = nextTx.txid; | ||||||
|               txid: a.txid, |           } | ||||||
|               fee: a.fee, |  | ||||||
|               weight: a.weight, |  | ||||||
|             }; |  | ||||||
|           }).reverse(); |  | ||||||
|           mempoolTx.descendants = descendants.map((a) => { |  | ||||||
|             return { |  | ||||||
|               txid: a.txid, |  | ||||||
|               fee: a.fee, |  | ||||||
|               weight: a.weight, |  | ||||||
|             }; |  | ||||||
|           }); |  | ||||||
|           descendants.push(ancestor); |  | ||||||
|           mempoolTx.cpfpChecked = true; |           mempoolTx.cpfpChecked = true; | ||||||
|           transactions.push(ancestor); |           transactions.push(ancestor); | ||||||
|           blockSize += ancestor.size; |           blockSize += ancestor.size; | ||||||
| @ -159,10 +153,10 @@ function makeBlockTemplates({ mempool, blockLimit, weightLimit, condenseRest }: | |||||||
|     // this block is full
 |     // this block is full
 | ||||||
|     const exceededPackageTries = failures > 1000 && blockWeight > (config.MEMPOOL.BLOCK_WEIGHT_UNITS - 4000); |     const exceededPackageTries = failures > 1000 && blockWeight > (config.MEMPOOL.BLOCK_WEIGHT_UNITS - 4000); | ||||||
|     const queueEmpty = top >= mempoolArray.length && modified.isEmpty(); |     const queueEmpty = top >= mempoolArray.length && modified.isEmpty(); | ||||||
|     if ((exceededPackageTries || queueEmpty) && (!condenseRest || blocks.length < blockLimit - 1)) { |     if ((exceededPackageTries || queueEmpty) && blocks.length < 7) { | ||||||
|       // construct this block
 |       // construct this block
 | ||||||
|       if (transactions.length) { |       if (transactions.length) { | ||||||
|         blocks.push(dataToMempoolBlocks(transactions.map(t => mempool[t.txid]), blockSize, blockWeight, blocks.length)); |         blocks.push(transactions.map(t => mempool[t.txid])); | ||||||
|       } |       } | ||||||
|       // reset for the next block
 |       // reset for the next block
 | ||||||
|       transactions = []; |       transactions = []; | ||||||
| @ -181,55 +175,40 @@ function makeBlockTemplates({ mempool, blockLimit, weightLimit, condenseRest }: | |||||||
|       overflow = []; |       overflow = []; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   if (condenseRest) { |   // pack any leftover transactions into the last block
 | ||||||
|     // pack any leftover transactions into the last block
 |   for (const tx of overflow) { | ||||||
|     for (const tx of overflow) { |     if (!tx || tx?.used) { | ||||||
|       if (!tx || tx?.used) { |       continue; | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
|       blockWeight += tx.weight; |  | ||||||
|       blockSize += tx.size; |  | ||||||
|       const mempoolTx = mempool[tx.txid]; |  | ||||||
|       // update original copy of this tx with effective fee rate & relatives data
 |  | ||||||
|       mempoolTx.effectiveFeePerVsize = tx.score; |  | ||||||
|       mempoolTx.ancestors = (Array.from(tx.ancestorMap?.values()) as AuditTransaction[]).map((a) => { |  | ||||||
|         return { |  | ||||||
|           txid: a.txid, |  | ||||||
|           fee: a.fee, |  | ||||||
|           weight: a.weight, |  | ||||||
|         }; |  | ||||||
|       }); |  | ||||||
|       mempoolTx.bestDescendant = null; |  | ||||||
|       mempoolTx.cpfpChecked = true; |  | ||||||
|       transactions.push(tx); |  | ||||||
|       tx.used = true; |  | ||||||
|     } |     } | ||||||
|     const blockTransactions = transactions.map(t => mempool[t.txid]); |     blockWeight += tx.weight; | ||||||
|     restOfArray.forEach(tx => { |     const mempoolTx = mempool[tx.txid]; | ||||||
|       blockWeight += tx.weight; |     // update original copy of this tx with effective fee rate & relatives data
 | ||||||
|       blockSize += tx.size; |     mempoolTx.effectiveFeePerVsize = tx.score; | ||||||
|       tx.effectiveFeePerVsize = tx.feePerVsize; |     if (tx.ancestorMap.size > 0) { | ||||||
|       tx.cpfpChecked = false; |       cpfpClusters[tx.txid] = Array.from(tx.ancestorMap?.values()).map(a => a.txid); | ||||||
|       tx.ancestors = []; |       mempoolTx.cpfpRoot = tx.txid; | ||||||
|       tx.bestDescendant = null; |  | ||||||
|       blockTransactions.push(tx); |  | ||||||
|     }); |  | ||||||
|     if (blockTransactions.length) { |  | ||||||
|       blocks.push(dataToMempoolBlocks(blockTransactions, blockSize, blockWeight, blocks.length)); |  | ||||||
|     } |     } | ||||||
|     transactions = []; |     mempoolTx.cpfpChecked = true; | ||||||
|   } else if (transactions.length) { |     transactions.push(tx); | ||||||
|     blocks.push(dataToMempoolBlocks(transactions.map(t => mempool[t.txid]), blockSize, blockWeight, blocks.length)); |     tx.used = true; | ||||||
|   } |   } | ||||||
|  |   const blockTransactions = transactions.map(t => mempool[t.txid]); | ||||||
|  |   restOfArray.forEach(tx => { | ||||||
|  |     blockWeight += tx.weight; | ||||||
|  |     tx.effectiveFeePerVsize = tx.feePerVsize; | ||||||
|  |     tx.cpfpChecked = false; | ||||||
|  |     blockTransactions.push(tx); | ||||||
|  |   }); | ||||||
|  |   if (blockTransactions.length) { | ||||||
|  |     blocks.push(blockTransactions); | ||||||
|  |   } | ||||||
|  |   transactions = []; | ||||||
| 
 | 
 | ||||||
|   const end = Date.now(); |   const end = Date.now(); | ||||||
|   const time = end - start; |   const time = end - start; | ||||||
|   logger.debug('Mempool templates calculated in ' + time / 1000 + ' seconds'); |   logger.debug('Mempool templates calculated in ' + time / 1000 + ' seconds'); | ||||||
| 
 | 
 | ||||||
|   return { |   return { blocks, clusters: cpfpClusters }; | ||||||
|     mempool, |  | ||||||
|     blocks |  | ||||||
|   }; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // traverse in-mempool ancestors
 | // traverse in-mempool ancestors
 | ||||||
| @ -239,9 +218,9 @@ function setRelatives( | |||||||
|   mempool: { [txid: string]: AuditTransaction }, |   mempool: { [txid: string]: AuditTransaction }, | ||||||
| ): void { | ): void { | ||||||
|   for (const parent of tx.vin) { |   for (const parent of tx.vin) { | ||||||
|     const parentTx = mempool[parent.txid]; |     const parentTx = mempool[parent]; | ||||||
|     if (parentTx && !tx.ancestorMap?.has(parent.txid)) { |     if (parentTx && !tx.ancestorMap?.has(parent)) { | ||||||
|       tx.ancestorMap.set(parent.txid, parentTx); |       tx.ancestorMap.set(parent, parentTx); | ||||||
|       parentTx.children.add(tx); |       parentTx.children.add(tx); | ||||||
|       // visit each node only once
 |       // visit each node only once
 | ||||||
|       if (!parentTx.relativesSet) { |       if (!parentTx.relativesSet) { | ||||||
| @ -313,26 +292,3 @@ function updateDescendants( | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| function dataToMempoolBlocks(transactions: TransactionExtended[], |  | ||||||
|   blockSize: number, blockWeight: number, blocksIndex: number): MempoolBlockWithTransactions { |  | ||||||
|   let rangeLength = 4; |  | ||||||
|   if (blocksIndex === 0) { |  | ||||||
|     rangeLength = 8; |  | ||||||
|   } |  | ||||||
|   if (transactions.length > 4000) { |  | ||||||
|     rangeLength = 6; |  | ||||||
|   } else if (transactions.length > 10000) { |  | ||||||
|     rangeLength = 8; |  | ||||||
|   } |  | ||||||
|   return { |  | ||||||
|     blockSize: blockSize, |  | ||||||
|     blockVSize: blockWeight / 4, |  | ||||||
|     nTx: transactions.length, |  | ||||||
|     totalFees: transactions.reduce((acc, cur) => acc + cur.fee, 0), |  | ||||||
|     medianFee: Common.percentile(transactions.map((tx) => tx.effectiveFeePerVsize), config.MEMPOOL.RECOMMENDED_FEE_PERCENTILE), |  | ||||||
|     feeRange: Common.getFeesInRange(transactions, rangeLength), |  | ||||||
|     transactionIds: transactions.map((tx) => tx.txid), |  | ||||||
|     transactions: transactions.map((tx) => Common.stripTransaction(tx)), |  | ||||||
|   }; |  | ||||||
| } |  | ||||||
| @ -251,7 +251,7 @@ class WebsocketHandler { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (config.MEMPOOL.ADVANCED_GBT_MEMPOOL) { |     if (config.MEMPOOL.ADVANCED_GBT_MEMPOOL) { | ||||||
|       await mempoolBlocks.makeBlockTemplates(newMempool, 8, null, true); |       await mempoolBlocks.updateBlockTemplates(newMempool, newTransactions, deletedTransactions.map(tx => tx.txid)); | ||||||
|     } else { |     } else { | ||||||
|       mempoolBlocks.updateMempoolBlocks(newMempool); |       mempoolBlocks.updateMempoolBlocks(newMempool); | ||||||
|     } |     } | ||||||
| @ -419,7 +419,7 @@ class WebsocketHandler { | |||||||
|     const _memPool = memPool.getMempool(); |     const _memPool = memPool.getMempool(); | ||||||
| 
 | 
 | ||||||
|     if (config.MEMPOOL.ADVANCED_GBT_AUDIT) { |     if (config.MEMPOOL.ADVANCED_GBT_AUDIT) { | ||||||
|       await mempoolBlocks.makeBlockTemplates(_memPool, 2); |       await mempoolBlocks.makeBlockTemplates(_memPool); | ||||||
|     } else { |     } else { | ||||||
|       mempoolBlocks.updateMempoolBlocks(_memPool); |       mempoolBlocks.updateMempoolBlocks(_memPool); | ||||||
|     } |     } | ||||||
| @ -462,13 +462,15 @@ class WebsocketHandler { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     const removed: string[] = []; | ||||||
|     // Update mempool to remove transactions included in the new block
 |     // Update mempool to remove transactions included in the new block
 | ||||||
|     for (const txId of txIds) { |     for (const txId of txIds) { | ||||||
|       delete _memPool[txId]; |       delete _memPool[txId]; | ||||||
|  |       removed.push(txId); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (config.MEMPOOL.ADVANCED_GBT_MEMPOOL) { |     if (config.MEMPOOL.ADVANCED_GBT_MEMPOOL) { | ||||||
|       await mempoolBlocks.makeBlockTemplates(_memPool, 8, null, true); |       await mempoolBlocks.updateBlockTemplates(_memPool, [], removed); | ||||||
|     } else { |     } else { | ||||||
|       mempoolBlocks.updateMempoolBlocks(_memPool); |       mempoolBlocks.updateMempoolBlocks(_memPool); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -81,10 +81,10 @@ export interface TransactionExtended extends IEsploraApi.Transaction { | |||||||
| export interface AuditTransaction { | export interface AuditTransaction { | ||||||
|   txid: string; |   txid: string; | ||||||
|   fee: number; |   fee: number; | ||||||
|   size: number; |  | ||||||
|   weight: number; |   weight: number; | ||||||
|   feePerVsize: number; |   feePerVsize: number; | ||||||
|   vin: IEsploraApi.Vin[]; |   effectiveFeePerVsize: number; | ||||||
|  |   vin: string[]; | ||||||
|   relativesSet: boolean; |   relativesSet: boolean; | ||||||
|   ancestorMap: Map<string, AuditTransaction>; |   ancestorMap: Map<string, AuditTransaction>; | ||||||
|   children: Set<AuditTransaction>; |   children: Set<AuditTransaction>; | ||||||
| @ -96,6 +96,17 @@ export interface AuditTransaction { | |||||||
|   modifiedNode: HeapNode<AuditTransaction>; |   modifiedNode: HeapNode<AuditTransaction>; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export interface ThreadTransaction { | ||||||
|  |   txid: string; | ||||||
|  |   fee: number; | ||||||
|  |   weight: number; | ||||||
|  |   feePerVsize: number; | ||||||
|  |   effectiveFeePerVsize?: number; | ||||||
|  |   vin: string[]; | ||||||
|  |   cpfpRoot?: string; | ||||||
|  |   cpfpChecked?: boolean; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export interface Ancestor { | export interface Ancestor { | ||||||
|   txid: string; |   txid: string; | ||||||
|   weight: number; |   weight: number; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user