// Copyright 2018 The Grin Developers // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. //! Transaction pool implementation. //! Used for both the txpool and stempool layers in the pool. use std::collections::HashSet; use std::sync::Arc; use core::core::hash::Hashed; use core::core::id::ShortIdentifiable; use core::core::transaction; use core::core::{Block, CompactBlock, Transaction, TxKernel}; use types::*; use util::LOGGER; pub struct Pool { /// Entries in the pool (tx + info + timer) in simple insertion order. pub entries: Vec, /// The blockchain pub blockchain: Arc, pub name: String, } impl Pool where T: BlockChain, { pub fn new(chain: Arc, name: String) -> Pool { Pool { entries: vec![], blockchain: chain.clone(), name, } } /// Query the tx pool for all known txs based on kernel short_ids /// from the provided compact_block. /// Note: does not validate that we return the full set of required txs. /// The caller will need to validate that themselves. pub fn retrieve_transactions(&self, cb: &CompactBlock) -> Vec { let mut txs = vec![]; for x in &self.entries { for kernel in &x.tx.kernels { // rehash each kernel to calculate the block specific short_id let short_id = kernel.short_id(&cb.hash(), cb.nonce); // if any kernel matches then keep the tx for later if cb.kern_ids.contains(&short_id) { txs.push(x.tx.clone()); break; } } } txs } /// Take the first num_to_fetch txs based on insertion order. pub fn prepare_mineable_transactions(&self, num_to_fetch: u32) -> Vec { self.entries .iter() .take(num_to_fetch as usize) .map(|x| x.tx.clone()) .collect() } pub fn all_transactions(&self) -> Vec { self.entries.iter().map(|x| x.tx.clone()).collect() } pub fn aggregate_transaction(&self) -> Result, PoolError> { let txs = self.all_transactions(); if txs.is_empty() { return Ok(None); } let tx = transaction::aggregate(txs)?; Ok(Some(tx)) } pub fn select_valid_transactions( &mut self, from_state: PoolEntryState, to_state: PoolEntryState, extra_tx: Option, ) -> Result, PoolError> { let entries = &mut self.entries .iter_mut() .filter(|x| x.state == from_state) .collect::>(); let candidate_txs = entries.iter().map(|x| x.tx.clone()).collect(); let valid_txs = self.blockchain.validate_raw_txs(candidate_txs, extra_tx)?; // Update state on all entries included in final vec of valid txs. for x in &mut entries.iter_mut() { if valid_txs.contains(&x.tx) { x.state = to_state.clone(); } } Ok(valid_txs) } // Aggregate this new tx with all existing txs in the pool. // If we can validate the aggregated tx against the current chain state // then we can safely add the tx to the pool. pub fn add_to_pool( &mut self, entry: PoolEntry, extra_txs: Vec, ) -> Result<(), PoolError> { debug!( LOGGER, "pool [{}]: add_to_pool: {}, {:?}", self.name, entry.tx.hash(), entry.src, ); // Combine all the txs from the pool with any extra txs provided. let mut txs = self.all_transactions(); txs.extend(extra_txs); let agg_tx = if txs.is_empty() { // If we have nothing to aggregate then simply return the tx itself. entry.tx.clone() } else { // Create a single aggregated tx from the existing pool txs (to check pool is // valid). let agg_tx = transaction::aggregate(txs)?; // Then check new tx would not introduce a duplicate output in the pool. for x in &entry.tx.outputs { if agg_tx.outputs.contains(&x) { return Err(PoolError::DuplicateCommitment); } } // Finally aggregate the new tx with everything in the pool (with any extra // txs). transaction::aggregate(vec![agg_tx, entry.tx.clone()])? }; // Validate aggregated tx against the current chain state (via txhashset // extension). self.blockchain.validate_raw_txs(vec![], Some(agg_tx))?; // If we get here successfully then we can safely add the entry to the pool. self.entries.push(entry); Ok(()) } pub fn reconcile(&mut self, extra_tx: Option) -> Result<(), PoolError> { let candidate_txs = self.all_transactions(); let existing_len = candidate_txs.len(); if candidate_txs.is_empty() { return Ok(()); } // Go through the candidate txs and keep everything that validates incrementally // against the current chain state, accounting for the "extra tx" as necessary. let valid_txs = self.blockchain.validate_raw_txs(candidate_txs, extra_tx)?; self.entries.retain(|x| valid_txs.contains(&x.tx)); debug!( LOGGER, "pool [{}]: reconcile: existing txs {}, retained txs {}", self.name, existing_len, self.entries.len(), ); Ok(()) } // Filter txs in the pool based on the latest block. // Reject any txs where we see a matching tx kernel in the block. // Also reject any txs where we see a conflicting tx, // where an input is spent in a different tx. fn remaining_transactions(&self, block: &Block) -> Vec { self.entries .iter() .filter(|x| !x.tx.kernels.iter().any(|y| block.kernels.contains(y))) .filter(|x| !x.tx.inputs.iter().any(|y| block.inputs.contains(y))) .map(|x| x.tx.clone()) .collect() } pub fn find_matching_transactions(&self, kernels: Vec) -> Vec { // While the inputs outputs can be cut-through the kernel will stay intact // In order to deaggregate tx we look for tx with the same kernel let mut found_txs = vec![]; // Gather all the kernels of the multi-kernel transaction in one set let kernel_set = kernels.into_iter().collect::>(); // Check each transaction in the pool for entry in &self.entries { let entry_kernel_set = entry.tx.kernels.iter().cloned().collect::>(); if entry_kernel_set.is_subset(&kernel_set) { found_txs.push(entry.tx.clone()); } } found_txs } /// Quick reconciliation step - we can evict any txs in the pool where /// inputs or kernels intersect with the block. pub fn reconcile_block(&mut self, block: &Block) -> Result<(), PoolError> { let candidate_txs = self.remaining_transactions(block); self.entries.retain(|x| candidate_txs.contains(&x.tx)); Ok(()) } pub fn size(&self) -> usize { self.entries.len() } }