Keep track of local transactions
This commit is contained in:
		
							parent
							
								
									78b5c743f6
								
							
						
					
					
						commit
						66e327dfcb
					
				
							
								
								
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @ -297,6 +297,7 @@ dependencies = [ | |||||||
|  "heapsize 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", |  "heapsize 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  "hyper 0.9.4 (git+https://github.com/ethcore/hyper)", |  "hyper 0.9.4 (git+https://github.com/ethcore/hyper)", | ||||||
|  "lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", |  "lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "linked-hash-map 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  "log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", |  "log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  "lru-cache 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", |  "lru-cache 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  "num_cpus 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)", |  "num_cpus 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  | |||||||
| @ -27,6 +27,7 @@ time = "0.1" | |||||||
| rand = "0.3" | rand = "0.3" | ||||||
| byteorder = "0.5" | byteorder = "0.5" | ||||||
| transient-hashmap = "0.1" | transient-hashmap = "0.1" | ||||||
|  | linked-hash-map = "0.3.0" | ||||||
| evmjit = { path = "../evmjit", optional = true } | evmjit = { path = "../evmjit", optional = true } | ||||||
| clippy = { version = "0.0.96", optional = true} | clippy = { version = "0.0.96", optional = true} | ||||||
| ethash = { path = "../ethash" } | ethash = { path = "../ethash" } | ||||||
|  | |||||||
| @ -102,6 +102,7 @@ extern crate rlp; | |||||||
| extern crate ethcore_bloom_journal as bloom_journal; | extern crate ethcore_bloom_journal as bloom_journal; | ||||||
| extern crate byteorder; | extern crate byteorder; | ||||||
| extern crate transient_hashmap; | extern crate transient_hashmap; | ||||||
|  | extern crate linked_hash_map; | ||||||
| 
 | 
 | ||||||
| #[macro_use] | #[macro_use] | ||||||
| extern crate log; | extern crate log; | ||||||
|  | |||||||
							
								
								
									
										193
									
								
								ethcore/src/miner/local_transactions.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										193
									
								
								ethcore/src/miner/local_transactions.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,193 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (UK) Ltd.
 | ||||||
|  | // This file is part of Parity.
 | ||||||
|  | 
 | ||||||
|  | // Parity is free software: you can redistribute it and/or modify
 | ||||||
|  | // it under the terms of the GNU General Public License as published by
 | ||||||
|  | // the Free Software Foundation, either version 3 of the License, or
 | ||||||
|  | // (at your option) any later version.
 | ||||||
|  | 
 | ||||||
|  | // Parity is distributed in the hope that it will be useful,
 | ||||||
|  | // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||||||
|  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||||||
|  | // GNU General Public License for more details.
 | ||||||
|  | 
 | ||||||
|  | // You should have received a copy of the GNU General Public License
 | ||||||
|  | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  | 
 | ||||||
|  | //! Local Transactions List.
 | ||||||
|  | 
 | ||||||
|  | use linked_hash_map::LinkedHashMap; | ||||||
|  | use transaction::SignedTransaction; | ||||||
|  | use error::TransactionError; | ||||||
|  | use util::{U256, H256}; | ||||||
|  | 
 | ||||||
|  | #[derive(Debug, PartialEq, Clone)] | ||||||
|  | pub enum Status { | ||||||
|  | 	/// The transaction is currently in the transaction queue.
 | ||||||
|  | 	Pending, | ||||||
|  | 	/// The transaction is in future part of the queue.
 | ||||||
|  | 	Future, | ||||||
|  | 	/// Transaction is already mined.
 | ||||||
|  | 	Mined(SignedTransaction), | ||||||
|  | 	/// Transaction is dropped because of limit
 | ||||||
|  | 	Dropped(SignedTransaction), | ||||||
|  | 	/// Replaced because of higher gas price of another transaction.
 | ||||||
|  | 	Replaced(SignedTransaction, U256, H256), | ||||||
|  | 	/// Transaction was never accepted to the queue.
 | ||||||
|  | 	Rejected(SignedTransaction, TransactionError), | ||||||
|  | 	/// Transaction is invalid.
 | ||||||
|  | 	Invalid(SignedTransaction), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Status { | ||||||
|  | 	fn is_current(&self) -> bool { | ||||||
|  | 		*self == Status::Pending || *self == Status::Future | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Keeps track of local transactions that are in the queue or were mined/dropped recently.
 | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub struct LocalTransactionsList { | ||||||
|  | 	max_old: usize, | ||||||
|  | 	transactions: LinkedHashMap<H256, Status>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Default for LocalTransactionsList { | ||||||
|  | 	fn default() -> Self { | ||||||
|  | 		Self::new(10) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl LocalTransactionsList { | ||||||
|  | 	pub fn new(max_old: usize) -> Self { | ||||||
|  | 		LocalTransactionsList { | ||||||
|  | 			max_old: max_old, | ||||||
|  | 			transactions: Default::default(), | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn mark_pending(&mut self, hash: H256) { | ||||||
|  | 		self.clear_old(); | ||||||
|  | 		self.transactions.insert(hash, Status::Pending); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn mark_future(&mut self, hash: H256) { | ||||||
|  | 		self.transactions.insert(hash, Status::Future); | ||||||
|  | 		self.clear_old(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn mark_rejected(&mut self, tx: SignedTransaction, err: TransactionError) { | ||||||
|  | 		self.transactions.insert(tx.hash(), Status::Rejected(tx, err)); | ||||||
|  | 		self.clear_old(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn mark_replaced(&mut self, tx: SignedTransaction, gas_price: U256, hash: H256) { | ||||||
|  | 		self.transactions.insert(tx.hash(), Status::Replaced(tx, gas_price, hash)); | ||||||
|  | 		self.clear_old(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn mark_invalid(&mut self, tx: SignedTransaction) { | ||||||
|  | 		self.transactions.insert(tx.hash(), Status::Invalid(tx)); | ||||||
|  | 		self.clear_old(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn mark_dropped(&mut self, tx: SignedTransaction) { | ||||||
|  | 		self.transactions.insert(tx.hash(), Status::Dropped(tx)); | ||||||
|  | 		self.clear_old(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn mark_mined(&mut self, tx: SignedTransaction) { | ||||||
|  | 		self.transactions.insert(tx.hash(), Status::Mined(tx)); | ||||||
|  | 		self.clear_old(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn contains(&self, hash: &H256) -> bool { | ||||||
|  | 		self.transactions.contains_key(hash) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn all_transactions(&self) -> &LinkedHashMap<H256, Status> { | ||||||
|  | 		&self.transactions | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn clear_old(&mut self) { | ||||||
|  | 		let number_of_old = self.transactions | ||||||
|  | 			.values() | ||||||
|  | 			.filter(|status| !status.is_current()) | ||||||
|  | 			.count(); | ||||||
|  | 
 | ||||||
|  | 		if self.max_old >= number_of_old { | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		let to_remove = self.transactions | ||||||
|  | 			.iter() | ||||||
|  | 			.filter(|&(_, status)| !status.is_current()) | ||||||
|  | 			.map(|(hash, _)| *hash) | ||||||
|  | 			.take(number_of_old - self.max_old) | ||||||
|  | 			.collect::<Vec<_>>(); | ||||||
|  | 
 | ||||||
|  | 		for hash in to_remove { | ||||||
|  | 			self.transactions.remove(&hash); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  | 	use util::U256; | ||||||
|  | 	use ethkey::{Random, Generator}; | ||||||
|  | 	use transaction::{Action, Transaction, SignedTransaction}; | ||||||
|  | 	use super::{LocalTransactionsList, Status}; | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn should_add_transaction_as_pending() { | ||||||
|  | 		// given
 | ||||||
|  | 		let mut list = LocalTransactionsList::default(); | ||||||
|  | 
 | ||||||
|  | 		// when
 | ||||||
|  | 		list.mark_pending(10.into()); | ||||||
|  | 		list.mark_future(20.into()); | ||||||
|  | 
 | ||||||
|  | 		// then
 | ||||||
|  | 		assert!(list.contains(&10.into()), "Should contain the transaction."); | ||||||
|  | 		assert!(list.contains(&20.into()), "Should contain the transaction."); | ||||||
|  | 		let statuses = list.all_transactions().values().cloned().collect::<Vec<Status>>(); | ||||||
|  | 		assert_eq!(statuses, vec![Status::Pending, Status::Future]); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn should_clear_old_transactions() { | ||||||
|  | 		// given
 | ||||||
|  | 		let mut list = LocalTransactionsList::new(1); | ||||||
|  | 		let tx1 = new_tx(10.into()); | ||||||
|  | 		let tx1_hash = tx1.hash(); | ||||||
|  | 		let tx2 = new_tx(50.into()); | ||||||
|  | 		let tx2_hash = tx2.hash(); | ||||||
|  | 
 | ||||||
|  | 		list.mark_pending(10.into()); | ||||||
|  | 		list.mark_invalid(tx1); | ||||||
|  | 		list.mark_dropped(tx2); | ||||||
|  | 		assert!(list.contains(&tx2_hash)); | ||||||
|  | 		assert!(!list.contains(&tx1_hash)); | ||||||
|  | 		assert!(list.contains(&10.into())); | ||||||
|  | 
 | ||||||
|  | 		// when
 | ||||||
|  | 		list.mark_future(15.into()); | ||||||
|  | 
 | ||||||
|  | 		// then
 | ||||||
|  | 		assert!(list.contains(&10.into())); | ||||||
|  | 		assert!(list.contains(&15.into())); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn new_tx(nonce: U256) -> SignedTransaction { | ||||||
|  | 		let keypair = Random.generate().unwrap(); | ||||||
|  | 		Transaction { | ||||||
|  | 			action: Action::Create, | ||||||
|  | 			value: U256::from(100), | ||||||
|  | 			data: Default::default(), | ||||||
|  | 			gas: U256::from(10), | ||||||
|  | 			gas_price: U256::from(1245), | ||||||
|  | 			nonce: nonce | ||||||
|  | 		}.sign(keypair.secret(), None) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -43,6 +43,7 @@ | |||||||
| 
 | 
 | ||||||
| mod banning_queue; | mod banning_queue; | ||||||
| mod external; | mod external; | ||||||
|  | mod local_transactions; | ||||||
| mod miner; | mod miner; | ||||||
| mod price_info; | mod price_info; | ||||||
| mod transaction_queue; | mod transaction_queue; | ||||||
|  | |||||||
| @ -91,6 +91,7 @@ use util::table::Table; | |||||||
| use transaction::*; | use transaction::*; | ||||||
| use error::{Error, TransactionError}; | use error::{Error, TransactionError}; | ||||||
| use client::TransactionImportResult; | use client::TransactionImportResult; | ||||||
|  | use miner::local_transactions::LocalTransactionsList; | ||||||
| 
 | 
 | ||||||
| /// Transaction origin
 | /// Transaction origin
 | ||||||
| #[derive(Clone, Copy, Debug, PartialEq, Eq)] | #[derive(Clone, Copy, Debug, PartialEq, Eq)] | ||||||
| @ -125,6 +126,12 @@ impl Ord for TransactionOrigin { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | impl TransactionOrigin { | ||||||
|  | 	fn is_local(&self) -> bool { | ||||||
|  | 		*self == TransactionOrigin::Local | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #[derive(Clone, Debug)] | #[derive(Clone, Debug)] | ||||||
| /// Light structure used to identify transaction and its order
 | /// Light structure used to identify transaction and its order
 | ||||||
| struct TransactionOrder { | struct TransactionOrder { | ||||||
| @ -352,7 +359,7 @@ impl TransactionSet { | |||||||
| 	///
 | 	///
 | ||||||
| 	/// It drops transactions from this set but also removes associated `VerifiedTransaction`.
 | 	/// It drops transactions from this set but also removes associated `VerifiedTransaction`.
 | ||||||
| 	/// Returns addresses and lowest nonces of transactions removed because of limit.
 | 	/// Returns addresses and lowest nonces of transactions removed because of limit.
 | ||||||
| 	fn enforce_limit(&mut self, by_hash: &mut HashMap<H256, VerifiedTransaction>) -> Option<HashMap<Address, U256>> { | 	fn enforce_limit(&mut self, by_hash: &mut HashMap<H256, VerifiedTransaction>, local: &mut LocalTransactionsList) -> Option<HashMap<Address, U256>> { | ||||||
| 		let mut count = 0; | 		let mut count = 0; | ||||||
| 		let mut gas: U256 = 0.into(); | 		let mut gas: U256 = 0.into(); | ||||||
| 		let to_drop : Vec<(Address, U256)> = { | 		let to_drop : Vec<(Address, U256)> = { | ||||||
| @ -379,9 +386,13 @@ impl TransactionSet { | |||||||
| 					.expect("Transaction has just been found in `by_priority`; so it is in `by_address` also."); | 					.expect("Transaction has just been found in `by_priority`; so it is in `by_address` also."); | ||||||
| 				trace!(target: "txqueue", "Dropped out of limit transaction: {:?}", order.hash); | 				trace!(target: "txqueue", "Dropped out of limit transaction: {:?}", order.hash); | ||||||
| 
 | 
 | ||||||
| 				by_hash.remove(&order.hash) | 				let order = by_hash.remove(&order.hash) | ||||||
| 					.expect("hash is in `by_priorty`; all hashes in `by_priority` must be in `by_hash`; qed"); | 					.expect("hash is in `by_priorty`; all hashes in `by_priority` must be in `by_hash`; qed"); | ||||||
| 
 | 
 | ||||||
|  | 				if order.origin.is_local() { | ||||||
|  | 					local.mark_dropped(order.transaction); | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
| 				let min = removed.get(&sender).map_or(nonce, |val| cmp::min(*val, nonce)); | 				let min = removed.get(&sender).map_or(nonce, |val| cmp::min(*val, nonce)); | ||||||
| 				removed.insert(sender, min); | 				removed.insert(sender, min); | ||||||
| 				removed | 				removed | ||||||
| @ -488,6 +499,8 @@ pub struct TransactionQueue { | |||||||
| 	by_hash: HashMap<H256, VerifiedTransaction>, | 	by_hash: HashMap<H256, VerifiedTransaction>, | ||||||
| 	/// Last nonce of transaction in current (to quickly check next expected transaction)
 | 	/// Last nonce of transaction in current (to quickly check next expected transaction)
 | ||||||
| 	last_nonces: HashMap<Address, U256>, | 	last_nonces: HashMap<Address, U256>, | ||||||
|  | 	/// List of local transactions and their statuses.
 | ||||||
|  | 	local_transactions: LocalTransactionsList, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Default for TransactionQueue { | impl Default for TransactionQueue { | ||||||
| @ -529,6 +542,7 @@ impl TransactionQueue { | |||||||
| 			future: future, | 			future: future, | ||||||
| 			by_hash: HashMap::new(), | 			by_hash: HashMap::new(), | ||||||
| 			last_nonces: HashMap::new(), | 			last_nonces: HashMap::new(), | ||||||
|  | 			local_transactions: LocalTransactionsList::default(), | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -537,8 +551,8 @@ impl TransactionQueue { | |||||||
| 		self.current.set_limit(limit); | 		self.current.set_limit(limit); | ||||||
| 		self.future.set_limit(limit); | 		self.future.set_limit(limit); | ||||||
| 		// And ensure the limits
 | 		// And ensure the limits
 | ||||||
| 		self.current.enforce_limit(&mut self.by_hash); | 		self.current.enforce_limit(&mut self.by_hash, &mut self.local_transactions); | ||||||
| 		self.future.enforce_limit(&mut self.by_hash); | 		self.future.enforce_limit(&mut self.by_hash, &mut self.local_transactions); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Returns current limit of transactions in the queue.
 | 	/// Returns current limit of transactions in the queue.
 | ||||||
| @ -578,7 +592,7 @@ impl TransactionQueue { | |||||||
| 	pub fn set_total_gas_limit(&mut self, gas_limit: U256) { | 	pub fn set_total_gas_limit(&mut self, gas_limit: U256) { | ||||||
| 		self.future.gas_limit = gas_limit; | 		self.future.gas_limit = gas_limit; | ||||||
| 		self.current.gas_limit = gas_limit; | 		self.current.gas_limit = gas_limit; | ||||||
| 		self.future.enforce_limit(&mut self.by_hash); | 		self.future.enforce_limit(&mut self.by_hash, &mut self.local_transactions); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Set the new limit for the amount of gas any individual transaction may have.
 | 	/// Set the new limit for the amount of gas any individual transaction may have.
 | ||||||
| @ -609,6 +623,42 @@ impl TransactionQueue { | |||||||
| 		F: Fn(&Address) -> AccountDetails, | 		F: Fn(&Address) -> AccountDetails, | ||||||
| 		G: Fn(&SignedTransaction) -> U256, | 		G: Fn(&SignedTransaction) -> U256, | ||||||
| 	{ | 	{ | ||||||
|  | 		if origin == TransactionOrigin::Local { | ||||||
|  | 			let hash = tx.hash(); | ||||||
|  | 			let cloned_tx = tx.clone(); | ||||||
|  | 
 | ||||||
|  | 			let result = self.add_internal(tx, origin, fetch_account, gas_estimator); | ||||||
|  | 			match result { | ||||||
|  | 				Ok(TransactionImportResult::Current) => { | ||||||
|  | 					self.local_transactions.mark_pending(hash); | ||||||
|  | 				}, | ||||||
|  | 				Ok(TransactionImportResult::Future) => { | ||||||
|  | 					self.local_transactions.mark_future(hash); | ||||||
|  | 				}, | ||||||
|  | 				Err(Error::Transaction(ref err)) => { | ||||||
|  | 					self.local_transactions.mark_rejected(cloned_tx, err.clone()); | ||||||
|  | 				}, | ||||||
|  | 				Err(_) => { | ||||||
|  | 					self.local_transactions.mark_invalid(cloned_tx); | ||||||
|  | 				}, | ||||||
|  | 			} | ||||||
|  | 			result | ||||||
|  | 		} else { | ||||||
|  | 			self.add_internal(tx, origin, fetch_account, gas_estimator) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Adds signed transaction to the queue.
 | ||||||
|  | 	fn add_internal<F, G>( | ||||||
|  | 		&mut self, | ||||||
|  | 		tx: SignedTransaction, | ||||||
|  | 		origin: TransactionOrigin, | ||||||
|  | 		fetch_account: &F, | ||||||
|  | 		gas_estimator: &G, | ||||||
|  | 	) -> Result<TransactionImportResult, Error> where | ||||||
|  | 		F: Fn(&Address) -> AccountDetails, | ||||||
|  | 		G: Fn(&SignedTransaction) -> U256, | ||||||
|  | 	{ | ||||||
| 
 | 
 | ||||||
| 		if tx.gas_price < self.minimal_gas_price && origin != TransactionOrigin::Local { | 		if tx.gas_price < self.minimal_gas_price && origin != TransactionOrigin::Local { | ||||||
| 			trace!(target: "txqueue", | 			trace!(target: "txqueue", | ||||||
| @ -647,7 +697,6 @@ impl TransactionQueue { | |||||||
| 				self.gas_limit, | 				self.gas_limit, | ||||||
| 				self.tx_gas_limit | 				self.tx_gas_limit | ||||||
| 			); | 			); | ||||||
| 
 |  | ||||||
| 			return Err(Error::Transaction(TransactionError::GasLimitExceeded { | 			return Err(Error::Transaction(TransactionError::GasLimitExceeded { | ||||||
| 				limit: self.gas_limit, | 				limit: self.gas_limit, | ||||||
| 				got: tx.gas, | 				got: tx.gas, | ||||||
| @ -766,6 +815,11 @@ impl TransactionQueue { | |||||||
| 
 | 
 | ||||||
| 		trace!(target: "txqueue", "Removing invalid transaction: {:?}", transaction.hash()); | 		trace!(target: "txqueue", "Removing invalid transaction: {:?}", transaction.hash()); | ||||||
| 
 | 
 | ||||||
|  | 		// Mark in locals
 | ||||||
|  | 		if self.local_transactions.contains(transaction_hash) { | ||||||
|  | 			self.local_transactions.mark_invalid(transaction.transaction.clone()); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		// Remove from future
 | 		// Remove from future
 | ||||||
| 		let order = self.future.drop(&sender, &nonce); | 		let order = self.future.drop(&sender, &nonce); | ||||||
| 		if order.is_some() { | 		if order.is_some() { | ||||||
| @ -821,15 +875,21 @@ impl TransactionQueue { | |||||||
| 															 qed");
 | 															 qed");
 | ||||||
| 			if k >= current_nonce { | 			if k >= current_nonce { | ||||||
| 				let order = order.update_height(k, current_nonce); | 				let order = order.update_height(k, current_nonce); | ||||||
|  | 				if order.origin.is_local() { | ||||||
|  | 					self.local_transactions.mark_future(order.hash); | ||||||
|  | 				} | ||||||
| 				if let Some(old) = self.future.insert(*sender, k, order.clone()) { | 				if let Some(old) = self.future.insert(*sender, k, order.clone()) { | ||||||
| 					Self::replace_orders(*sender, k, old, order, &mut self.future, &mut self.by_hash); | 					Self::replace_orders(*sender, k, old, order, &mut self.future, &mut self.by_hash, &mut self.local_transactions); | ||||||
| 				} | 				} | ||||||
| 			} else { | 			} else { | ||||||
| 				trace!(target: "txqueue", "Removing old transaction: {:?} (nonce: {} < {})", order.hash, k, current_nonce); | 				trace!(target: "txqueue", "Removing old transaction: {:?} (nonce: {} < {})", order.hash, k, current_nonce); | ||||||
| 				self.by_hash.remove(&order.hash).expect("All transactions in `future` are also in `by_hash`"); | 				let tx = self.by_hash.remove(&order.hash).expect("All transactions in `future` are also in `by_hash`"); | ||||||
|  | 				if tx.origin.is_local() { | ||||||
|  | 					self.local_transactions.mark_mined(tx.transaction); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		self.future.enforce_limit(&mut self.by_hash); | 		} | ||||||
|  | 		self.future.enforce_limit(&mut self.by_hash, &mut self.local_transactions); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Returns top transactions from the queue ordered by priority.
 | 	/// Returns top transactions from the queue ordered by priority.
 | ||||||
| @ -897,8 +957,11 @@ impl TransactionQueue { | |||||||
| 				self.future.by_gas_price.remove(&order.gas_price, &order.hash); | 				self.future.by_gas_price.remove(&order.gas_price, &order.hash); | ||||||
| 				// Put to current
 | 				// Put to current
 | ||||||
| 				let order = order.update_height(current_nonce, first_nonce); | 				let order = order.update_height(current_nonce, first_nonce); | ||||||
|  | 				if order.origin.is_local() { | ||||||
|  | 					self.local_transactions.mark_pending(order.hash); | ||||||
|  | 				} | ||||||
| 				if let Some(old) = self.current.insert(address, current_nonce, order.clone()) { | 				if let Some(old) = self.current.insert(address, current_nonce, order.clone()) { | ||||||
| 					Self::replace_orders(address, current_nonce, old, order, &mut self.current, &mut self.by_hash); | 					Self::replace_orders(address, current_nonce, old, order, &mut self.current, &mut self.by_hash, &mut self.local_transactions); | ||||||
| 				} | 				} | ||||||
| 				update_last_nonce_to = Some(current_nonce); | 				update_last_nonce_to = Some(current_nonce); | ||||||
| 				current_nonce = current_nonce + U256::one(); | 				current_nonce = current_nonce + U256::one(); | ||||||
| @ -957,9 +1020,11 @@ impl TransactionQueue { | |||||||
| 		if nonce > next_nonce { | 		if nonce > next_nonce { | ||||||
| 			// We have a gap - put to future.
 | 			// We have a gap - put to future.
 | ||||||
| 			// Insert transaction (or replace old one with lower gas price)
 | 			// Insert transaction (or replace old one with lower gas price)
 | ||||||
| 			try!(check_too_cheap(Self::replace_transaction(tx, state_nonce, min_gas_price, &mut self.future, &mut self.by_hash))); | 			try!(check_too_cheap( | ||||||
|  | 				Self::replace_transaction(tx, state_nonce, min_gas_price, &mut self.future, &mut self.by_hash, &mut self.local_transactions) | ||||||
|  | 			)); | ||||||
| 			// Enforce limit in Future
 | 			// Enforce limit in Future
 | ||||||
| 			let removed = self.future.enforce_limit(&mut self.by_hash); | 			let removed = self.future.enforce_limit(&mut self.by_hash, &mut self.local_transactions); | ||||||
| 			// Return an error if this transaction was not imported because of limit.
 | 			// Return an error if this transaction was not imported because of limit.
 | ||||||
| 			try!(check_if_removed(&address, &nonce, removed)); | 			try!(check_if_removed(&address, &nonce, removed)); | ||||||
| 
 | 
 | ||||||
| @ -973,13 +1038,15 @@ impl TransactionQueue { | |||||||
| 		self.move_matching_future_to_current(address, nonce + U256::one(), state_nonce); | 		self.move_matching_future_to_current(address, nonce + U256::one(), state_nonce); | ||||||
| 
 | 
 | ||||||
| 		// Replace transaction if any
 | 		// Replace transaction if any
 | ||||||
| 		try!(check_too_cheap(Self::replace_transaction(tx, state_nonce, min_gas_price, &mut self.current, &mut self.by_hash))); | 		try!(check_too_cheap( | ||||||
|  | 			Self::replace_transaction(tx, state_nonce, min_gas_price, &mut self.current, &mut self.by_hash, &mut self.local_transactions) | ||||||
|  | 		)); | ||||||
| 		// Keep track of highest nonce stored in current
 | 		// Keep track of highest nonce stored in current
 | ||||||
| 		let new_max = self.last_nonces.get(&address).map_or(nonce, |n| cmp::max(nonce, *n)); | 		let new_max = self.last_nonces.get(&address).map_or(nonce, |n| cmp::max(nonce, *n)); | ||||||
| 		self.last_nonces.insert(address, new_max); | 		self.last_nonces.insert(address, new_max); | ||||||
| 
 | 
 | ||||||
| 		// Also enforce the limit
 | 		// Also enforce the limit
 | ||||||
| 		let removed = self.current.enforce_limit(&mut self.by_hash); | 		let removed = self.current.enforce_limit(&mut self.by_hash, &mut self.local_transactions); | ||||||
| 		// If some transaction were removed because of limit we need to update last_nonces also.
 | 		// If some transaction were removed because of limit we need to update last_nonces also.
 | ||||||
| 		self.update_last_nonces(&removed); | 		self.update_last_nonces(&removed); | ||||||
| 		// Trigger error if the transaction we are importing was removed.
 | 		// Trigger error if the transaction we are importing was removed.
 | ||||||
| @ -1010,7 +1077,14 @@ impl TransactionQueue { | |||||||
| 	///
 | 	///
 | ||||||
| 	/// Returns `true` if transaction actually got to the queue (`false` if there was already a transaction with higher
 | 	/// Returns `true` if transaction actually got to the queue (`false` if there was already a transaction with higher
 | ||||||
| 	/// gas_price)
 | 	/// gas_price)
 | ||||||
| 	fn replace_transaction(tx: VerifiedTransaction, base_nonce: U256, min_gas_price: (U256, PrioritizationStrategy), set: &mut TransactionSet, by_hash: &mut HashMap<H256, VerifiedTransaction>) -> bool { | 	fn replace_transaction( | ||||||
|  | 		tx: VerifiedTransaction, | ||||||
|  | 		base_nonce: U256, | ||||||
|  | 		min_gas_price: (U256, PrioritizationStrategy), | ||||||
|  | 		set: &mut TransactionSet, | ||||||
|  | 		by_hash: &mut HashMap<H256, VerifiedTransaction>, | ||||||
|  | 		local: &mut LocalTransactionsList, | ||||||
|  | 	) -> bool { | ||||||
| 		let order = TransactionOrder::for_transaction(&tx, base_nonce, min_gas_price.0, min_gas_price.1); | 		let order = TransactionOrder::for_transaction(&tx, base_nonce, min_gas_price.0, min_gas_price.1); | ||||||
| 		let hash = tx.hash(); | 		let hash = tx.hash(); | ||||||
| 		let address = tx.sender(); | 		let address = tx.sender(); | ||||||
| @ -1021,14 +1095,24 @@ impl TransactionQueue { | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 		if let Some(old) = set.insert(address, nonce, order.clone()) { | 		if let Some(old) = set.insert(address, nonce, order.clone()) { | ||||||
| 			Self::replace_orders(address, nonce, old, order, set, by_hash) | 			Self::replace_orders(address, nonce, old, order, set, by_hash, local) | ||||||
| 		} else { | 		} else { | ||||||
| 			true | 			true | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	fn replace_orders(address: Address, nonce: U256, old: TransactionOrder, order: TransactionOrder, set: &mut TransactionSet, by_hash: &mut HashMap<H256, VerifiedTransaction>) -> bool { | 	fn replace_orders( | ||||||
|  | 		address: Address, | ||||||
|  | 		nonce: U256, | ||||||
|  | 		old: TransactionOrder, | ||||||
|  | 		order: TransactionOrder, | ||||||
|  | 		set: &mut TransactionSet, | ||||||
|  | 		by_hash: &mut HashMap<H256, VerifiedTransaction>, | ||||||
|  | 		local: &mut LocalTransactionsList, | ||||||
|  | 	) -> bool { | ||||||
| 		// There was already transaction in queue. Let's check which one should stay
 | 		// There was already transaction in queue. Let's check which one should stay
 | ||||||
|  | 		let old_hash = old.hash; | ||||||
|  | 		let new_hash = order.hash; | ||||||
| 		let old_fee = old.gas_price; | 		let old_fee = old.gas_price; | ||||||
| 		let new_fee = order.gas_price; | 		let new_fee = order.gas_price; | ||||||
| 		if old_fee.cmp(&new_fee) == Ordering::Greater { | 		if old_fee.cmp(&new_fee) == Ordering::Greater { | ||||||
| @ -1036,12 +1120,18 @@ impl TransactionQueue { | |||||||
| 			// Put back old transaction since it has greater priority (higher gas_price)
 | 			// Put back old transaction since it has greater priority (higher gas_price)
 | ||||||
| 			set.insert(address, nonce, old); | 			set.insert(address, nonce, old); | ||||||
| 			// and remove new one
 | 			// and remove new one
 | ||||||
| 			by_hash.remove(&order.hash).expect("The hash has been just inserted and no other line is altering `by_hash`."); | 			let order = by_hash.remove(&order.hash).expect("The hash has been just inserted and no other line is altering `by_hash`."); | ||||||
|  | 			if order.origin.is_local() { | ||||||
|  | 				local.mark_replaced(order.transaction, old_fee, old_hash); | ||||||
|  | 			} | ||||||
| 			false | 			false | ||||||
| 		} else { | 		} else { | ||||||
| 			trace!(target: "txqueue", "Replaced transaction: {:?} with transaction with higher gas price: {:?}", old.hash, order.hash); | 			trace!(target: "txqueue", "Replaced transaction: {:?} with transaction with higher gas price: {:?}", old.hash, order.hash); | ||||||
| 			// Make sure we remove old transaction entirely
 | 			// Make sure we remove old transaction entirely
 | ||||||
| 			by_hash.remove(&old.hash).expect("The hash is coming from `future` so it has to be in `by_hash`."); | 			let old = by_hash.remove(&old.hash).expect("The hash is coming from `future` so it has to be in `by_hash`."); | ||||||
|  | 			if old.origin.is_local() { | ||||||
|  | 				local.mark_replaced(old.transaction, new_fee, new_hash); | ||||||
|  | 			} | ||||||
| 			true | 			true | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -1078,6 +1168,7 @@ mod test { | |||||||
| 	use error::{Error, TransactionError}; | 	use error::{Error, TransactionError}; | ||||||
| 	use super::*; | 	use super::*; | ||||||
| 	use super::{TransactionSet, TransactionOrder, VerifiedTransaction}; | 	use super::{TransactionSet, TransactionOrder, VerifiedTransaction}; | ||||||
|  | 	use miner::local_transactions::LocalTransactionsList; | ||||||
| 	use client::TransactionImportResult; | 	use client::TransactionImportResult; | ||||||
| 
 | 
 | ||||||
| 	fn unwrap_tx_err(err: Result<TransactionImportResult, Error>) -> TransactionError { | 	fn unwrap_tx_err(err: Result<TransactionImportResult, Error>) -> TransactionError { | ||||||
| @ -1208,6 +1299,7 @@ mod test { | |||||||
| 	#[test] | 	#[test] | ||||||
| 	fn should_create_transaction_set() { | 	fn should_create_transaction_set() { | ||||||
| 		// given
 | 		// given
 | ||||||
|  | 		let mut local = LocalTransactionsList::default(); | ||||||
| 		let mut set = TransactionSet { | 		let mut set = TransactionSet { | ||||||
| 			by_priority: BTreeSet::new(), | 			by_priority: BTreeSet::new(), | ||||||
| 			by_address: Table::new(), | 			by_address: Table::new(), | ||||||
| @ -1235,7 +1327,7 @@ mod test { | |||||||
| 		assert_eq!(set.by_address.len(), 2); | 		assert_eq!(set.by_address.len(), 2); | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		set.enforce_limit(&mut by_hash); | 		set.enforce_limit(&mut by_hash, &mut local); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
| 		assert_eq!(by_hash.len(), 1); | 		assert_eq!(by_hash.len(), 1); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user