Merge branch 'master' of github.com:ethcore/parity into db_writer
This commit is contained in:
		
						commit
						9ce9fd390d
					
				| @ -122,7 +122,7 @@ const CLIENT_DB_VER_STR: &'static str = "5.3"; | |||||||
| 
 | 
 | ||||||
| impl Client<CanonVerifier> { | impl Client<CanonVerifier> { | ||||||
| 	/// Create a new client with given spec and DB path.
 | 	/// Create a new client with given spec and DB path.
 | ||||||
| 	pub fn new(config: ClientConfig, spec: Spec, path: &Path, message_channel: IoChannel<NetSyncMessage> ) -> Result<Arc<Client>, Error> { | 	pub fn new(config: ClientConfig, spec: Spec, path: &Path, message_channel: IoChannel<NetSyncMessage> ) -> Arc<Client> { | ||||||
| 		Client::<CanonVerifier>::new_with_verifier(config, spec, path, message_channel) | 		Client::<CanonVerifier>::new_with_verifier(config, spec, path, message_channel) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -146,7 +146,7 @@ pub fn append_path(path: &Path, item: &str) -> String { | |||||||
| 
 | 
 | ||||||
| impl<V> Client<V> where V: Verifier { | impl<V> Client<V> where V: Verifier { | ||||||
| 	///  Create a new client with given spec and DB path and custom verifier.
 | 	///  Create a new client with given spec and DB path and custom verifier.
 | ||||||
| 	pub fn new_with_verifier(config: ClientConfig, spec: Spec, path: &Path, message_channel: IoChannel<NetSyncMessage> ) -> Result<Arc<Client<V>>, Error> { | 	pub fn new_with_verifier(config: ClientConfig, spec: Spec, path: &Path, message_channel: IoChannel<NetSyncMessage> ) -> Arc<Client<V>> { | ||||||
| 		let path = get_db_path(path, config.pruning, spec.genesis_header().hash()); | 		let path = get_db_path(path, config.pruning, spec.genesis_header().hash()); | ||||||
| 		let gb = spec.genesis_block(); | 		let gb = spec.genesis_block(); | ||||||
| 		let chain = Arc::new(BlockChain::new(config.blockchain, &gb, &path)); | 		let chain = Arc::new(BlockChain::new(config.blockchain, &gb, &path)); | ||||||
| @ -163,7 +163,7 @@ impl<V> Client<V> where V: Verifier { | |||||||
| 		let panic_handler = PanicHandler::new_in_arc(); | 		let panic_handler = PanicHandler::new_in_arc(); | ||||||
| 		panic_handler.forward_from(&block_queue); | 		panic_handler.forward_from(&block_queue); | ||||||
| 
 | 
 | ||||||
| 		Ok(Arc::new(Client { | 		Arc::new(Client { | ||||||
| 			chain: chain, | 			chain: chain, | ||||||
| 			engine: engine, | 			engine: engine, | ||||||
| 			state_db: Mutex::new(state_db), | 			state_db: Mutex::new(state_db), | ||||||
| @ -172,7 +172,7 @@ impl<V> Client<V> where V: Verifier { | |||||||
| 			import_lock: Mutex::new(()), | 			import_lock: Mutex::new(()), | ||||||
| 			panic_handler: panic_handler, | 			panic_handler: panic_handler, | ||||||
| 			verifier: PhantomData, | 			verifier: PhantomData, | ||||||
| 		})) | 		}) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Flush the block import queue.
 | 	/// Flush the block import queue.
 | ||||||
|  | |||||||
| @ -62,13 +62,18 @@ pub enum ExecutionError { | |||||||
| 	Internal | 	Internal | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Debug)] | #[derive(Debug, PartialEq)] | ||||||
| /// Errors concerning transaction processing.
 | /// Errors concerning transaction processing.
 | ||||||
| pub enum TransactionError { | pub enum TransactionError { | ||||||
| 	/// Transaction is already imported to the queue
 | 	/// Transaction is already imported to the queue
 | ||||||
| 	AlreadyImported, | 	AlreadyImported, | ||||||
| 	/// Transaction is not valid anymore (state already has higher nonce)
 | 	/// Transaction is not valid anymore (state already has higher nonce)
 | ||||||
| 	Old, | 	Old, | ||||||
|  | 	/// Transaction has too low fee
 | ||||||
|  | 	/// (there is already a transaction with the same sender-nonce but higher gas price)
 | ||||||
|  | 	TooCheapToReplace, | ||||||
|  | 	/// Transaction was not imported to the queue because limit has been reached.
 | ||||||
|  | 	LimitReached, | ||||||
| 	/// Transaction's gas price is below threshold.
 | 	/// Transaction's gas price is below threshold.
 | ||||||
| 	InsufficientGasPrice { | 	InsufficientGasPrice { | ||||||
| 		/// Minimal expected gas price
 | 		/// Minimal expected gas price
 | ||||||
| @ -153,7 +158,7 @@ pub enum BlockError { | |||||||
| 	UnknownUncleParent(H256), | 	UnknownUncleParent(H256), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Debug)] | #[derive(Debug, PartialEq)] | ||||||
| /// Import to the block queue result
 | /// Import to the block queue result
 | ||||||
| pub enum ImportError { | pub enum ImportError { | ||||||
| 	/// Already in the block chain.
 | 	/// Already in the block chain.
 | ||||||
|  | |||||||
| @ -53,7 +53,7 @@ pub fn json_chain_test(json_data: &[u8], era: ChainEra) -> Vec<String> { | |||||||
| 
 | 
 | ||||||
| 			let temp = RandomTempPath::new(); | 			let temp = RandomTempPath::new(); | ||||||
| 			{ | 			{ | ||||||
| 				let client = Client::new(ClientConfig::default(), spec, temp.as_path(), IoChannel::disconnected()).unwrap(); | 				let client = Client::new(ClientConfig::default(), spec, temp.as_path(), IoChannel::disconnected()); | ||||||
| 				for b in &blockchain.blocks_rlp() { | 				for b in &blockchain.blocks_rlp() { | ||||||
| 					if Block::is_good(&b) { | 					if Block::is_good(&b) { | ||||||
| 						let _ = client.import_block(b.clone()); | 						let _ = client.import_block(b.clone()); | ||||||
|  | |||||||
| @ -17,11 +17,9 @@ | |||||||
| use super::test_common::*; | use super::test_common::*; | ||||||
| use state::*; | use state::*; | ||||||
| use executive::*; | use executive::*; | ||||||
| use spec::*; |  | ||||||
| use engine::*; | use engine::*; | ||||||
| use evm; | use evm; | ||||||
| use evm::{Schedule, Ext, Factory, VMType, ContractCreateResult, MessageCallResult}; | use evm::{Schedule, Ext, Factory, VMType, ContractCreateResult, MessageCallResult}; | ||||||
| use ethereum; |  | ||||||
| use externalities::*; | use externalities::*; | ||||||
| use substate::*; | use substate::*; | ||||||
| use tests::helpers::*; | use tests::helpers::*; | ||||||
|  | |||||||
| @ -61,7 +61,7 @@ impl ClientService { | |||||||
| 
 | 
 | ||||||
| 		info!("Starting {}", net_service.host_info()); | 		info!("Starting {}", net_service.host_info()); | ||||||
| 		info!("Configured for {} using {:?} engine", spec.name, spec.engine.name()); | 		info!("Configured for {} using {:?} engine", spec.name, spec.engine.name()); | ||||||
| 		let client = try!(Client::new(config, spec, db_path, net_service.io().channel())); | 		let client = Client::new(config, spec, db_path, net_service.io().channel()); | ||||||
| 		panic_handler.forward_from(client.deref()); | 		panic_handler.forward_from(client.deref()); | ||||||
| 		let client_io = Arc::new(ClientIoHandler { | 		let client_io = Arc::new(ClientIoHandler { | ||||||
| 			client: client.clone() | 			client: client.clone() | ||||||
|  | |||||||
| @ -20,17 +20,10 @@ use tests::helpers::*; | |||||||
| use common::*; | use common::*; | ||||||
| use devtools::*; | use devtools::*; | ||||||
| 
 | 
 | ||||||
| #[test] |  | ||||||
| fn created() { |  | ||||||
| 	let dir = RandomTempPath::new(); |  | ||||||
| 	let client_result = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()); |  | ||||||
| 	assert!(client_result.is_ok()); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[test] | #[test] | ||||||
| fn imports_from_empty() { | fn imports_from_empty() { | ||||||
| 	let dir = RandomTempPath::new(); | 	let dir = RandomTempPath::new(); | ||||||
| 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()).unwrap(); | 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()); | ||||||
| 	client.import_verified_blocks(&IoChannel::disconnected()); | 	client.import_verified_blocks(&IoChannel::disconnected()); | ||||||
| 	client.flush_queue(); | 	client.flush_queue(); | ||||||
| } | } | ||||||
| @ -48,7 +41,7 @@ fn returns_state_root_basic() { | |||||||
| #[test] | #[test] | ||||||
| fn imports_good_block() { | fn imports_good_block() { | ||||||
| 	let dir = RandomTempPath::new(); | 	let dir = RandomTempPath::new(); | ||||||
| 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()).unwrap(); | 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()); | ||||||
| 	let good_block = get_good_dummy_block(); | 	let good_block = get_good_dummy_block(); | ||||||
| 	if let Err(_) = client.import_block(good_block) { | 	if let Err(_) = client.import_block(good_block) { | ||||||
| 		panic!("error importing block being good by definition"); | 		panic!("error importing block being good by definition"); | ||||||
| @ -63,7 +56,7 @@ fn imports_good_block() { | |||||||
| #[test] | #[test] | ||||||
| fn query_none_block() { | fn query_none_block() { | ||||||
| 	let dir = RandomTempPath::new(); | 	let dir = RandomTempPath::new(); | ||||||
| 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()).unwrap(); | 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()); | ||||||
| 
 | 
 | ||||||
|     let non_existant = client.block_header(BlockId::Number(188)); |     let non_existant = client.block_header(BlockId::Number(188)); | ||||||
| 	assert!(non_existant.is_none()); | 	assert!(non_existant.is_none()); | ||||||
|  | |||||||
| @ -145,7 +145,7 @@ pub fn create_test_block_with_data(header: &Header, transactions: &[&SignedTrans | |||||||
| pub fn generate_dummy_client(block_number: u32) -> GuardedTempResult<Arc<Client>> { | pub fn generate_dummy_client(block_number: u32) -> GuardedTempResult<Arc<Client>> { | ||||||
| 	let dir = RandomTempPath::new(); | 	let dir = RandomTempPath::new(); | ||||||
| 
 | 
 | ||||||
| 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()).unwrap(); | 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()); | ||||||
| 	let test_spec = get_test_spec(); | 	let test_spec = get_test_spec(); | ||||||
| 	let test_engine = &test_spec.engine; | 	let test_engine = &test_spec.engine; | ||||||
| 	let state_root = test_spec.genesis_header().state_root; | 	let state_root = test_spec.genesis_header().state_root; | ||||||
| @ -211,7 +211,7 @@ pub fn push_blocks_to_client(client: &Arc<Client>, timestamp_salt: u64, starting | |||||||
| 
 | 
 | ||||||
| pub fn get_test_client_with_blocks(blocks: Vec<Bytes>) -> GuardedTempResult<Arc<Client>> { | pub fn get_test_client_with_blocks(blocks: Vec<Bytes>) -> GuardedTempResult<Arc<Client>> { | ||||||
| 	let dir = RandomTempPath::new(); | 	let dir = RandomTempPath::new(); | ||||||
| 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()).unwrap(); | 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()); | ||||||
| 	for block in &blocks { | 	for block in &blocks { | ||||||
| 		if let Err(_) = client.import_block(block.clone()) { | 		if let Err(_) = client.import_block(block.clone()) { | ||||||
| 			panic!("panic importing block which is well-formed"); | 			panic!("panic importing block which is well-formed"); | ||||||
|  | |||||||
| @ -38,7 +38,7 @@ | |||||||
| //! fn main() {
 | //! fn main() {
 | ||||||
| //! 	let mut service = NetworkService::start(NetworkConfiguration::new()).unwrap();
 | //! 	let mut service = NetworkService::start(NetworkConfiguration::new()).unwrap();
 | ||||||
| //! 	let dir = env::temp_dir();
 | //! 	let dir = env::temp_dir();
 | ||||||
| //! 	let client = Client::new(ClientConfig::default(), ethereum::new_frontier(), &dir, service.io().channel()).unwrap();
 | //! 	let client = Client::new(ClientConfig::default(), ethereum::new_frontier(), &dir, service.io().channel());
 | ||||||
| //!
 | //!
 | ||||||
| //!		let miner: Miner = Miner::default();
 | //!		let miner: Miner = Miner::default();
 | ||||||
| //!		// get status
 | //!		// get status
 | ||||||
| @ -61,7 +61,7 @@ extern crate rayon; | |||||||
| mod miner; | mod miner; | ||||||
| mod transaction_queue; | mod transaction_queue; | ||||||
| 
 | 
 | ||||||
| pub use transaction_queue::{TransactionQueue, AccountDetails}; | pub use transaction_queue::{TransactionQueue, AccountDetails, TransactionImportResult}; | ||||||
| pub use miner::{Miner}; | pub use miner::{Miner}; | ||||||
| 
 | 
 | ||||||
| use util::{H256, U256, Address, Bytes}; | use util::{H256, U256, Address, Bytes}; | ||||||
| @ -100,6 +100,12 @@ pub trait MinerService : Send + Sync { | |||||||
| 	/// Set the gas limit we wish to target when sealing a new block.
 | 	/// Set the gas limit we wish to target when sealing a new block.
 | ||||||
| 	fn set_gas_floor_target(&self, target: U256); | 	fn set_gas_floor_target(&self, target: U256); | ||||||
| 
 | 
 | ||||||
|  | 	/// Get current transactions limit in queue.
 | ||||||
|  | 	fn transactions_limit(&self) -> usize; | ||||||
|  | 
 | ||||||
|  | 	/// Set maximal number of transactions kept in the queue (both current and future).
 | ||||||
|  | 	fn set_transactions_limit(&self, limit: usize); | ||||||
|  | 
 | ||||||
| 	/// Imports transactions to transaction queue.
 | 	/// Imports transactions to transaction queue.
 | ||||||
| 	fn import_transactions<T>(&self, transactions: Vec<SignedTransaction>, fetch_account: T) -> | 	fn import_transactions<T>(&self, transactions: Vec<SignedTransaction>, fetch_account: T) -> | ||||||
| 		Vec<Result<TransactionImportResult, Error>> | 		Vec<Result<TransactionImportResult, Error>> | ||||||
| @ -145,15 +151,6 @@ pub trait MinerService : Send + Sync { | |||||||
| 	fn sensible_gas_limit(&self) -> U256 { x!(21000) } | 	fn sensible_gas_limit(&self) -> U256 { x!(21000) } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Represents the result of importing transaction.
 |  | ||||||
| #[derive(Debug)] |  | ||||||
| pub enum TransactionImportResult { |  | ||||||
| 	/// Transaction was imported to current queue.
 |  | ||||||
| 	Current, |  | ||||||
| 	/// Transaction was imported to future queue.
 |  | ||||||
| 	Future |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Mining status
 | /// Mining status
 | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct MinerStatus { | pub struct MinerStatus { | ||||||
|  | |||||||
| @ -205,6 +205,14 @@ impl MinerService for Miner { | |||||||
| 		*self.gas_floor_target.read().unwrap() / x!(5) | 		*self.gas_floor_target.read().unwrap() / x!(5) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	fn transactions_limit(&self) -> usize { | ||||||
|  | 		self.transaction_queue.lock().unwrap().limit() | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn set_transactions_limit(&self, limit: usize) { | ||||||
|  | 		self.transaction_queue.lock().unwrap().set_limit(limit) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/// Get the author that we will seal blocks as.
 | 	/// Get the author that we will seal blocks as.
 | ||||||
| 	fn author(&self) -> Address { | 	fn author(&self) -> Address { | ||||||
| 		*self.author.read().unwrap() | 		*self.author.read().unwrap() | ||||||
|  | |||||||
| @ -86,14 +86,13 @@ | |||||||
| 
 | 
 | ||||||
| use std::default::Default; | use std::default::Default; | ||||||
| use std::cmp::{Ordering}; | use std::cmp::{Ordering}; | ||||||
|  | use std::cmp; | ||||||
| use std::collections::{HashMap, BTreeSet}; | use std::collections::{HashMap, BTreeSet}; | ||||||
| use util::numbers::{Uint, U256}; | use util::numbers::{Uint, U256}; | ||||||
| use util::hash::{Address, H256}; | use util::hash::{Address, H256}; | ||||||
| use util::table::*; | use util::table::*; | ||||||
| use ethcore::transaction::*; | use ethcore::transaction::*; | ||||||
| use ethcore::error::{Error, TransactionError}; | use ethcore::error::{Error, TransactionError}; | ||||||
| use super::TransactionImportResult; |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| #[derive(Clone, Debug)] | #[derive(Clone, Debug)] | ||||||
| /// Light structure used to identify transaction and it's order
 | /// Light structure used to identify transaction and it's order
 | ||||||
| @ -206,10 +205,11 @@ impl TransactionSet { | |||||||
| 	/// Remove low priority transactions if there is more then specified by given `limit`.
 | 	/// Remove low priority transactions if there is more then specified by given `limit`.
 | ||||||
| 	///
 | 	///
 | ||||||
| 	/// It drops transactions from this set but also removes associated `VerifiedTransaction`.
 | 	/// It drops transactions from this set but also removes associated `VerifiedTransaction`.
 | ||||||
| 	fn enforce_limit(&mut self, by_hash: &mut HashMap<H256, VerifiedTransaction>) { | 	/// Returns addresses and highes nonces of transactions removed because of limit.
 | ||||||
|  | 	fn enforce_limit(&mut self, by_hash: &mut HashMap<H256, VerifiedTransaction>) -> Option<HashMap<Address, U256>> { | ||||||
| 		let len = self.by_priority.len(); | 		let len = self.by_priority.len(); | ||||||
| 		if len <= self.limit { | 		if len <= self.limit { | ||||||
| 			return; | 			return None; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		let to_drop : Vec<(Address, U256)> = { | 		let to_drop : Vec<(Address, U256)> = { | ||||||
| @ -222,10 +222,18 @@ impl TransactionSet { | |||||||
| 				.collect() | 				.collect() | ||||||
| 		}; | 		}; | ||||||
| 
 | 
 | ||||||
| 		for (sender, nonce) in to_drop { | 		Some(to_drop.into_iter() | ||||||
| 			let order = self.drop(&sender, &nonce).expect("Transaction has just been found in `by_priority`; so it is in `by_address` also."); | 			.fold(HashMap::new(), |mut removed, (sender, nonce)| { | ||||||
| 			by_hash.remove(&order.hash).expect("Hash found in `by_priorty` matches the one dropped; so it is included in `by_hash`"); | 				let order = self.drop(&sender, &nonce) | ||||||
| 		} | 					.expect("Transaction has just been found in `by_priority`; so it is in `by_address` also."); | ||||||
|  | 
 | ||||||
|  | 				by_hash.remove(&order.hash) | ||||||
|  | 					.expect("Hash found in `by_priorty` matches the one dropped; so it is included in `by_hash`"); | ||||||
|  | 
 | ||||||
|  | 				let max = removed.get(&sender).map(|val| cmp::max(*val, nonce)).unwrap_or(nonce); | ||||||
|  | 				removed.insert(sender, max); | ||||||
|  | 				removed | ||||||
|  | 			})) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Drop transaction from this set (remove from `by_priority` and `by_address`)
 | 	/// Drop transaction from this set (remove from `by_priority` and `by_address`)
 | ||||||
| @ -242,6 +250,12 @@ impl TransactionSet { | |||||||
| 		self.by_priority.clear(); | 		self.by_priority.clear(); | ||||||
| 		self.by_address.clear(); | 		self.by_address.clear(); | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Sets new limit for number of transactions in this `TransactionSet`.
 | ||||||
|  | 	/// Note the limit is not applied (no transactions are removed) by calling this method.
 | ||||||
|  | 	fn set_limit(&mut self, limit: usize) { | ||||||
|  | 		self.limit = limit; | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| @ -253,6 +267,15 @@ pub struct TransactionQueueStatus { | |||||||
| 	pub future: usize, | 	pub future: usize, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #[derive(Debug, PartialEq)] | ||||||
|  | /// Represents the result of importing transaction.
 | ||||||
|  | pub enum TransactionImportResult { | ||||||
|  | 	/// Transaction was imported to current queue.
 | ||||||
|  | 	Current, | ||||||
|  | 	/// Transaction was imported to future queue.
 | ||||||
|  | 	Future | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /// Details of account
 | /// Details of account
 | ||||||
| pub struct AccountDetails { | pub struct AccountDetails { | ||||||
| 	/// Most recent account nonce
 | 	/// Most recent account nonce
 | ||||||
| @ -290,20 +313,21 @@ impl Default for TransactionQueue { | |||||||
| impl TransactionQueue { | impl TransactionQueue { | ||||||
| 	/// Creates new instance of this Queue
 | 	/// Creates new instance of this Queue
 | ||||||
| 	pub fn new() -> Self { | 	pub fn new() -> Self { | ||||||
| 		Self::with_limits(1024, 1024) | 		Self::with_limit(1024) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Create new instance of this Queue with specified limits
 | 	/// Create new instance of this Queue with specified limits
 | ||||||
| 	pub fn with_limits(current_limit: usize, future_limit: usize) -> Self { | 	pub fn with_limit(limit: usize) -> Self { | ||||||
| 		let current = TransactionSet { | 		let current = TransactionSet { | ||||||
| 			by_priority: BTreeSet::new(), | 			by_priority: BTreeSet::new(), | ||||||
| 			by_address: Table::new(), | 			by_address: Table::new(), | ||||||
| 			limit: current_limit, | 			limit: limit, | ||||||
| 		}; | 		}; | ||||||
|  | 
 | ||||||
| 		let future = TransactionSet { | 		let future = TransactionSet { | ||||||
| 			by_priority: BTreeSet::new(), | 			by_priority: BTreeSet::new(), | ||||||
| 			by_address: Table::new(), | 			by_address: Table::new(), | ||||||
| 			limit: future_limit, | 			limit: limit, | ||||||
| 		}; | 		}; | ||||||
| 
 | 
 | ||||||
| 		TransactionQueue { | 		TransactionQueue { | ||||||
| @ -316,6 +340,20 @@ impl TransactionQueue { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/// Set the new limit for `current` and `future` queue.
 | ||||||
|  | 	pub fn set_limit(&mut self, limit: usize) { | ||||||
|  | 		self.current.set_limit(limit); | ||||||
|  | 		self.future.set_limit(limit); | ||||||
|  | 		// And ensure the limits
 | ||||||
|  | 		self.current.enforce_limit(&mut self.by_hash); | ||||||
|  | 		self.future.enforce_limit(&mut self.by_hash); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Returns current limit of transactions in the queue.
 | ||||||
|  | 	pub fn limit(&self) -> usize { | ||||||
|  | 		self.current.limit | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/// Get the minimal gas price.
 | 	/// Get the minimal gas price.
 | ||||||
| 	pub fn minimal_gas_price(&self) -> &U256 { | 	pub fn minimal_gas_price(&self) -> &U256 { | ||||||
| 		&self.minimal_gas_price | 		&self.minimal_gas_price | ||||||
| @ -589,8 +627,8 @@ impl TransactionQueue { | |||||||
| 		// Check height
 | 		// Check height
 | ||||||
| 		if nonce > next_nonce { | 		if nonce > next_nonce { | ||||||
| 			// We have a gap - put to future
 | 			// We have a gap - put to future
 | ||||||
| 			Self::replace_transaction(tx, next_nonce, &mut self.future, &mut self.by_hash); | 			try!(check_too_cheap(Self::replace_transaction(tx, next_nonce, &mut self.future, &mut self.by_hash))); | ||||||
| 			self.future.enforce_limit(&mut self.by_hash); | 			try!(check_if_removed(&address, &nonce, self.future.enforce_limit(&mut self.by_hash))); | ||||||
| 			return Ok(TransactionImportResult::Future); | 			return Ok(TransactionImportResult::Future); | ||||||
| 		} else if nonce < state_nonce { | 		} else if nonce < state_nonce { | ||||||
| 			// Droping transaction
 | 			// Droping transaction
 | ||||||
| @ -598,7 +636,7 @@ impl TransactionQueue { | |||||||
| 			return Err(TransactionError::Old); | 			return Err(TransactionError::Old); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		Self::replace_transaction(tx, state_nonce, &mut self.current, &mut self.by_hash); | 		try!(check_too_cheap(Self::replace_transaction(tx, state_nonce, &mut self.current, &mut self.by_hash))); | ||||||
| 		// Keep track of highest nonce stored in current
 | 		// Keep track of highest nonce stored in current
 | ||||||
| 		self.last_nonces.insert(address, nonce); | 		self.last_nonces.insert(address, nonce); | ||||||
| 		// Update nonces of transactions in future
 | 		// Update nonces of transactions in future
 | ||||||
| @ -610,26 +648,48 @@ impl TransactionQueue { | |||||||
| 		if let Some(order) = self.future.drop(&address, &nonce) { | 		if let Some(order) = self.future.drop(&address, &nonce) { | ||||||
| 			// Let's insert that transaction to current (if it has higher gas_price)
 | 			// Let's insert that transaction to current (if it has higher gas_price)
 | ||||||
| 			let future_tx = self.by_hash.remove(&order.hash).unwrap(); | 			let future_tx = self.by_hash.remove(&order.hash).unwrap(); | ||||||
| 			Self::replace_transaction(future_tx, state_nonce, &mut self.current, &mut self.by_hash); | 			try!(check_too_cheap(Self::replace_transaction(future_tx, state_nonce, &mut self.current, &mut self.by_hash))); | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
| 		// Also enforce the limit
 | 		// Also enforce the limit
 | ||||||
| 		self.current.enforce_limit(&mut self.by_hash); | 		let removed = self.current.enforce_limit(&mut self.by_hash); | ||||||
|  | 		// If some transaction were removed because of limit we need to update last_nonces also.
 | ||||||
|  | 		self.update_last_nonces(&removed); | ||||||
|  | 		// Trigger error if we were removed.
 | ||||||
|  | 		try!(check_if_removed(&address, &nonce, removed)); | ||||||
| 
 | 
 | ||||||
| 		trace!(target: "miner", "status: {:?}", self.status()); | 		trace!(target: "miner", "status: {:?}", self.status()); | ||||||
| 		Ok(TransactionImportResult::Current) | 		Ok(TransactionImportResult::Current) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/// Updates
 | ||||||
|  | 	fn update_last_nonces(&mut self, removed_max_nonces: &Option<HashMap<Address, U256>>) { | ||||||
|  | 		if let Some(ref max_nonces) = *removed_max_nonces { | ||||||
|  | 			for (sender, nonce) in max_nonces.iter() { | ||||||
|  | 				if *nonce == U256::zero() { | ||||||
|  | 					self.last_nonces.remove(sender); | ||||||
|  | 				} else { | ||||||
|  | 					self.last_nonces.insert(*sender, *nonce - U256::one()); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/// Replaces transaction in given set (could be `future` or `current`).
 | 	/// Replaces transaction in given set (could be `future` or `current`).
 | ||||||
| 	///
 | 	///
 | ||||||
| 	/// If there is already transaction with same `(sender, nonce)` it will be replaced iff `gas_price` is higher.
 | 	/// If there is already transaction with same `(sender, nonce)` it will be replaced iff `gas_price` is higher.
 | ||||||
| 	/// One of the transactions is dropped from set and also removed from queue entirely (from `by_hash`).
 | 	/// One of the transactions is dropped from set and also removed from queue entirely (from `by_hash`).
 | ||||||
| 	fn replace_transaction(tx: VerifiedTransaction, base_nonce: U256, set: &mut TransactionSet, by_hash: &mut HashMap<H256, VerifiedTransaction>) { | 	///
 | ||||||
|  | 	/// Returns `true` if transaction actually got to the queue (`false` if there was already a transaction with higher
 | ||||||
|  | 	/// gas_price)
 | ||||||
|  | 	fn replace_transaction(tx: VerifiedTransaction, base_nonce: U256, set: &mut TransactionSet, by_hash: &mut HashMap<H256, VerifiedTransaction>) -> bool { | ||||||
| 		let order = TransactionOrder::for_transaction(&tx, base_nonce); | 		let order = TransactionOrder::for_transaction(&tx, base_nonce); | ||||||
| 		let hash = tx.hash(); | 		let hash = tx.hash(); | ||||||
| 		let address = tx.sender(); | 		let address = tx.sender(); | ||||||
| 		let nonce = tx.nonce(); | 		let nonce = tx.nonce(); | ||||||
| 
 | 
 | ||||||
| 		by_hash.insert(hash, tx); | 		by_hash.insert(hash, tx); | ||||||
|  | 
 | ||||||
| 		if let Some(old) = set.insert(address, nonce, order.clone()) { | 		if let Some(old) = set.insert(address, nonce, order.clone()) { | ||||||
| 			// 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_fee = old.gas_price; | 			let old_fee = old.gas_price; | ||||||
| @ -639,11 +699,35 @@ impl TransactionQueue { | |||||||
| 				set.insert(address, nonce, old); | 				set.insert(address, nonce, old); | ||||||
| 				// and remove new one
 | 				// and remove new one
 | ||||||
| 				by_hash.remove(&hash); | 				by_hash.remove(&hash); | ||||||
|  | 				false | ||||||
| 			} else { | 			} else { | ||||||
| 				// Make sure we remove old transaction entirely
 | 				// Make sure we remove old transaction entirely
 | ||||||
| 				by_hash.remove(&old.hash); | 				by_hash.remove(&old.hash); | ||||||
|  | 				true | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			true | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn check_too_cheap(is_in: bool) -> Result<(), TransactionError> { | ||||||
|  | 	if !is_in { | ||||||
|  | 		Err(TransactionError::TooCheapToReplace) | ||||||
|  | 	} else { | ||||||
|  | 		Ok(()) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn check_if_removed(sender: &Address, nonce: &U256, dropped: Option<HashMap<Address, U256>>) -> Result<(), TransactionError> { | ||||||
|  | 	match dropped { | ||||||
|  | 		Some(ref dropped) => match dropped.get(sender) { | ||||||
|  | 			Some(max) if nonce <= max => { | ||||||
|  | 				Err(TransactionError::LimitReached) | ||||||
|  | 			}, | ||||||
|  | 			_ => Ok(()), | ||||||
|  | 		}, | ||||||
|  | 		_ => Ok(()), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -654,9 +738,17 @@ mod test { | |||||||
| 	use util::table::*; | 	use util::table::*; | ||||||
| 	use util::*; | 	use util::*; | ||||||
| 	use ethcore::transaction::*; | 	use ethcore::transaction::*; | ||||||
|  | 	use ethcore::error::{Error, TransactionError}; | ||||||
| 	use super::*; | 	use super::*; | ||||||
| 	use super::{TransactionSet, TransactionOrder, VerifiedTransaction}; | 	use super::{TransactionSet, TransactionOrder, VerifiedTransaction}; | ||||||
| 
 | 
 | ||||||
|  | 	fn unwrap_tx_err(err: Result<TransactionImportResult, Error>) -> TransactionError { | ||||||
|  | 		match err.unwrap_err() { | ||||||
|  | 			Error::Transaction(e) => e, | ||||||
|  | 			_ => panic!("Expected transaction error!"), | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	fn new_unsigned_tx(nonce: U256) -> Transaction { | 	fn new_unsigned_tx(nonce: U256) -> Transaction { | ||||||
| 		Transaction { | 		Transaction { | ||||||
| 			action: Action::Create, | 			action: Action::Create, | ||||||
| @ -698,11 +790,16 @@ mod test { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	fn new_txs(second_nonce: U256) -> (SignedTransaction, SignedTransaction) { | 	fn new_txs(second_nonce: U256) -> (SignedTransaction, SignedTransaction) { | ||||||
|  | 		new_txs_with_gas_price_diff(second_nonce, U256::zero()) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn new_txs_with_gas_price_diff(second_nonce: U256, gas_price: U256) -> (SignedTransaction, SignedTransaction) { | ||||||
| 		let keypair = KeyPair::create().unwrap(); | 		let keypair = KeyPair::create().unwrap(); | ||||||
| 		let secret = &keypair.secret(); | 		let secret = &keypair.secret(); | ||||||
| 		let nonce = U256::from(123); | 		let nonce = U256::from(123); | ||||||
| 		let tx = new_unsigned_tx(nonce); | 		let tx = new_unsigned_tx(nonce); | ||||||
| 		let tx2 = new_unsigned_tx(nonce + second_nonce); | 		let mut tx2 = new_unsigned_tx(nonce + second_nonce); | ||||||
|  | 		tx2.gas_price = tx2.gas_price + gas_price; | ||||||
| 
 | 
 | ||||||
| 		(tx.sign(secret), tx2.sign(secret)) | 		(tx.sign(secret), tx2.sign(secret)) | ||||||
| 	} | 	} | ||||||
| @ -795,14 +892,14 @@ mod test { | |||||||
| 
 | 
 | ||||||
| 		// First insert one transaction to future
 | 		// First insert one transaction to future
 | ||||||
| 		let res = txq.add(tx, &prev_nonce); | 		let res = txq.add(tx, &prev_nonce); | ||||||
| 		assert!(res.is_ok()); | 		assert_eq!(res.unwrap(), TransactionImportResult::Future); | ||||||
| 		assert_eq!(txq.status().future, 1); | 		assert_eq!(txq.status().future, 1); | ||||||
| 
 | 
 | ||||||
| 		// now import second transaction to current
 | 		// now import second transaction to current
 | ||||||
| 		let res = txq.add(tx2.clone(), &default_nonce); | 		let res = txq.add(tx2.clone(), &default_nonce); | ||||||
| 
 | 
 | ||||||
| 		// and then there should be only one transaction in current (the one with higher gas_price)
 | 		// and then there should be only one transaction in current (the one with higher gas_price)
 | ||||||
| 		assert!(res.is_ok()); | 		assert_eq!(unwrap_tx_err(res), TransactionError::TooCheapToReplace); | ||||||
| 		assert_eq!(txq.status().pending, 1); | 		assert_eq!(txq.status().pending, 1); | ||||||
| 		assert_eq!(txq.status().future, 0); | 		assert_eq!(txq.status().future, 0); | ||||||
| 		assert_eq!(txq.current.by_priority.len(), 1); | 		assert_eq!(txq.current.by_priority.len(), 1); | ||||||
| @ -821,7 +918,7 @@ mod test { | |||||||
| 		let res = txq.add(tx, &default_nonce); | 		let res = txq.add(tx, &default_nonce); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
| 		assert!(res.is_ok()); | 		assert_eq!(res.unwrap(), TransactionImportResult::Current); | ||||||
| 		let stats = txq.status(); | 		let stats = txq.status(); | ||||||
| 		assert_eq!(stats.pending, 1); | 		assert_eq!(stats.pending, 1); | ||||||
| 	} | 	} | ||||||
| @ -845,12 +942,18 @@ mod test { | |||||||
| 		// given
 | 		// given
 | ||||||
| 		let mut txq = TransactionQueue::new(); | 		let mut txq = TransactionQueue::new(); | ||||||
| 		let tx = new_tx(); | 		let tx = new_tx(); | ||||||
| 		txq.set_gas_limit(tx.gas / U256::from(2)); | 		let gas = tx.gas; | ||||||
|  | 		let limit = gas / U256::from(2); | ||||||
|  | 		txq.set_gas_limit(limit); | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		txq.add(tx, &default_nonce).unwrap_err(); | 		let res = txq.add(tx, &default_nonce); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
|  | 		assert_eq!(unwrap_tx_err(res), TransactionError::GasLimitExceeded { | ||||||
|  | 			limit: U256::from(55_000), // Should be 110% of set_gas_limit
 | ||||||
|  | 			got: gas, | ||||||
|  | 		}); | ||||||
| 		let stats = txq.status(); | 		let stats = txq.status(); | ||||||
| 		assert_eq!(stats.pending, 0); | 		assert_eq!(stats.pending, 0); | ||||||
| 		assert_eq!(stats.future, 0); | 		assert_eq!(stats.future, 0); | ||||||
| @ -868,9 +971,13 @@ mod test { | |||||||
| 		}; | 		}; | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		txq.add(tx, &account).unwrap_err(); | 		let res = txq.add(tx, &account); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
|  | 		assert_eq!(unwrap_tx_err(res), TransactionError::InsufficientBalance { | ||||||
|  | 			balance: U256::from(1), | ||||||
|  | 			cost: U256::from(100_100), | ||||||
|  | 		}); | ||||||
| 		let stats = txq.status(); | 		let stats = txq.status(); | ||||||
| 		assert_eq!(stats.pending, 0); | 		assert_eq!(stats.pending, 0); | ||||||
| 		assert_eq!(stats.future, 0); | 		assert_eq!(stats.future, 0); | ||||||
| @ -884,9 +991,13 @@ mod test { | |||||||
| 		txq.set_minimal_gas_price(tx.gas_price + U256::one()); | 		txq.set_minimal_gas_price(tx.gas_price + U256::one()); | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		txq.add(tx, &default_nonce).unwrap_err(); | 		let res = txq.add(tx, &default_nonce); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
|  | 		assert_eq!(unwrap_tx_err(res), TransactionError::InsufficientGasPrice { | ||||||
|  | 			minimal: U256::from(2), | ||||||
|  | 			got: U256::from(1), | ||||||
|  | 		}); | ||||||
| 		let stats = txq.status(); | 		let stats = txq.status(); | ||||||
| 		assert_eq!(stats.pending, 0); | 		assert_eq!(stats.pending, 0); | ||||||
| 		assert_eq!(stats.future, 0); | 		assert_eq!(stats.future, 0); | ||||||
| @ -961,10 +1072,12 @@ mod test { | |||||||
| 		let (tx, tx2) = new_txs(U256::from(2)); | 		let (tx, tx2) = new_txs(U256::from(2)); | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		txq.add(tx.clone(), &default_nonce).unwrap(); | 		let res1 = txq.add(tx.clone(), &default_nonce).unwrap(); | ||||||
| 		txq.add(tx2.clone(), &default_nonce).unwrap(); | 		let res2 = txq.add(tx2.clone(), &default_nonce).unwrap(); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
|  | 		assert_eq!(res1, TransactionImportResult::Current); | ||||||
|  | 		assert_eq!(res2, TransactionImportResult::Future); | ||||||
| 		let stats = txq.status(); | 		let stats = txq.status(); | ||||||
| 		assert_eq!(stats.pending, 1); | 		assert_eq!(stats.pending, 1); | ||||||
| 		assert_eq!(stats.future, 1); | 		assert_eq!(stats.future, 1); | ||||||
| @ -1085,26 +1198,53 @@ mod test { | |||||||
| 	#[test] | 	#[test] | ||||||
| 	fn should_drop_old_transactions_when_hitting_the_limit() { | 	fn should_drop_old_transactions_when_hitting_the_limit() { | ||||||
| 		// given
 | 		// given
 | ||||||
| 		let mut txq = TransactionQueue::with_limits(1, 1); | 		let mut txq = TransactionQueue::with_limit(1); | ||||||
| 		let (tx, tx2) = new_txs(U256::one()); | 		let (tx, tx2) = new_txs(U256::one()); | ||||||
|  | 		let sender = tx.sender().unwrap(); | ||||||
|  | 		let nonce = tx.nonce; | ||||||
| 		txq.add(tx.clone(), &default_nonce).unwrap(); | 		txq.add(tx.clone(), &default_nonce).unwrap(); | ||||||
| 		assert_eq!(txq.status().pending, 1); | 		assert_eq!(txq.status().pending, 1); | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		txq.add(tx2.clone(), &default_nonce).unwrap(); | 		let res = txq.add(tx2.clone(), &default_nonce); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
| 		let t = txq.top_transactions(); | 		let t = txq.top_transactions(); | ||||||
|  | 		assert_eq!(unwrap_tx_err(res), TransactionError::LimitReached); | ||||||
| 		assert_eq!(txq.status().pending, 1); | 		assert_eq!(txq.status().pending, 1); | ||||||
| 		assert_eq!(t.len(), 1); | 		assert_eq!(t.len(), 1); | ||||||
| 		assert_eq!(t[0], tx); | 		assert_eq!(t[0], tx); | ||||||
|  | 		assert_eq!(txq.last_nonce(&sender), Some(nonce)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn should_return_correct_nonces_when_dropped_because_of_limit() { | ||||||
|  | 		// given
 | ||||||
|  | 		let mut txq = TransactionQueue::with_limit(2); | ||||||
|  | 		let tx = new_tx(); | ||||||
|  | 		let (tx1, tx2) = new_txs(U256::one()); | ||||||
|  | 		let sender = tx1.sender().unwrap(); | ||||||
|  | 		let nonce = tx1.nonce; | ||||||
|  | 		txq.add(tx1.clone(), &default_nonce).unwrap(); | ||||||
|  | 		txq.add(tx2.clone(), &default_nonce).unwrap(); | ||||||
|  | 		assert_eq!(txq.status().pending, 2); | ||||||
|  | 		assert_eq!(txq.last_nonce(&sender), Some(nonce + U256::one())); | ||||||
|  | 
 | ||||||
|  | 		// when
 | ||||||
|  | 		let res = txq.add(tx.clone(), &default_nonce); | ||||||
|  | 
 | ||||||
|  | 		// then
 | ||||||
|  | 		assert_eq!(res.unwrap(), TransactionImportResult::Current); | ||||||
|  | 		assert_eq!(txq.status().pending, 2); | ||||||
|  | 		assert_eq!(txq.last_nonce(&sender), Some(nonce)); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	#[test] | 	#[test] | ||||||
| 	fn should_limit_future_transactions() { | 	fn should_limit_future_transactions() { | ||||||
| 		let mut txq = TransactionQueue::with_limits(10, 1); | 		let mut txq = TransactionQueue::with_limit(1); | ||||||
| 		let (tx1, tx2) = new_txs(U256::from(4)); | 		txq.current.set_limit(10); | ||||||
| 		let (tx3, tx4) = new_txs(U256::from(4)); | 		let (tx1, tx2) = new_txs_with_gas_price_diff(U256::from(4), U256::from(1)); | ||||||
|  | 		let (tx3, tx4) = new_txs_with_gas_price_diff(U256::from(4), U256::from(2)); | ||||||
| 		txq.add(tx1.clone(), &default_nonce).unwrap(); | 		txq.add(tx1.clone(), &default_nonce).unwrap(); | ||||||
| 		txq.add(tx3.clone(), &default_nonce).unwrap(); | 		txq.add(tx3.clone(), &default_nonce).unwrap(); | ||||||
| 		assert_eq!(txq.status().pending, 2); | 		assert_eq!(txq.status().pending, 2); | ||||||
| @ -1126,9 +1266,10 @@ mod test { | |||||||
| 		let fetch_last_nonce = |_a: &Address| AccountDetails{ nonce: last_nonce, balance: !U256::zero() }; | 		let fetch_last_nonce = |_a: &Address| AccountDetails{ nonce: last_nonce, balance: !U256::zero() }; | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		txq.add(tx, &fetch_last_nonce).unwrap_err(); | 		let res = txq.add(tx, &fetch_last_nonce); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
|  | 		assert_eq!(unwrap_tx_err(res), TransactionError::Old); | ||||||
| 		let stats = txq.status(); | 		let stats = txq.status(); | ||||||
| 		assert_eq!(stats.pending, 0); | 		assert_eq!(stats.pending, 0); | ||||||
| 		assert_eq!(stats.future, 0); | 		assert_eq!(stats.future, 0); | ||||||
| @ -1146,9 +1287,10 @@ mod test { | |||||||
| 		assert_eq!(txq.status().pending, 0); | 		assert_eq!(txq.status().pending, 0); | ||||||
| 
 | 
 | ||||||
| 		// when
 | 		// when
 | ||||||
| 		txq.add(tx2.clone(), &nonce).unwrap_err(); | 		let res = txq.add(tx2.clone(), &nonce); | ||||||
| 
 | 
 | ||||||
| 		// then
 | 		// then
 | ||||||
|  | 		assert_eq!(unwrap_tx_err(res), TransactionError::AlreadyImported); | ||||||
| 		let stats = txq.status(); | 		let stats = txq.status(); | ||||||
| 		assert_eq!(stats.future, 1); | 		assert_eq!(stats.future, 1); | ||||||
| 		assert_eq!(stats.pending, 0); | 		assert_eq!(stats.pending, 0); | ||||||
|  | |||||||
| @ -172,6 +172,8 @@ Sealing/Mining Options: | |||||||
|                            [default: 0037a6b811ffeb6e072da21179d11b1406371c63]. |                            [default: 0037a6b811ffeb6e072da21179d11b1406371c63]. | ||||||
|   --extra-data STRING      Specify a custom extra-data for authored blocks, no |   --extra-data STRING      Specify a custom extra-data for authored blocks, no | ||||||
|                            more than 32 characters. |                            more than 32 characters. | ||||||
|  |   --tx-limit LIMIT         Limit of transactions kept in the queue (waiting to | ||||||
|  |                            be included in next block) [default: 1024]. | ||||||
| 
 | 
 | ||||||
| Footprint Options: | Footprint Options: | ||||||
|   --pruning METHOD         Configure pruning of the state/storage trie. METHOD |   --pruning METHOD         Configure pruning of the state/storage trie. METHOD | ||||||
| @ -259,6 +261,7 @@ struct Args { | |||||||
| 	flag_usd_per_eth: String, | 	flag_usd_per_eth: String, | ||||||
| 	flag_gas_floor_target: String, | 	flag_gas_floor_target: String, | ||||||
| 	flag_extra_data: Option<String>, | 	flag_extra_data: Option<String>, | ||||||
|  | 	flag_tx_limit: usize, | ||||||
| 	flag_logging: Option<String>, | 	flag_logging: Option<String>, | ||||||
| 	flag_version: bool, | 	flag_version: bool, | ||||||
| 	// geth-compatibility...
 | 	// geth-compatibility...
 | ||||||
| @ -713,6 +716,7 @@ impl Configuration { | |||||||
| 		miner.set_gas_floor_target(self.gas_floor_target()); | 		miner.set_gas_floor_target(self.gas_floor_target()); | ||||||
| 		miner.set_extra_data(self.extra_data()); | 		miner.set_extra_data(self.extra_data()); | ||||||
| 		miner.set_minimal_gas_price(self.gas_price()); | 		miner.set_minimal_gas_price(self.gas_price()); | ||||||
|  | 		miner.set_transactions_limit(self.args.flag_tx_limit); | ||||||
| 
 | 
 | ||||||
| 		// Sync
 | 		// Sync
 | ||||||
| 		let sync = EthSync::register(service.network(), sync_config, client.clone(), miner.clone()); | 		let sync = EthSync::register(service.network(), sync_config, client.clone(), miner.clone()); | ||||||
|  | |||||||
| @ -67,6 +67,17 @@ impl<M> Ethcore for EthcoreClient<M> where M: MinerService + 'static { | |||||||
| 		}) | 		}) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	fn set_transactions_limit(&self, params: Params) -> Result<Value, Error> { | ||||||
|  | 		from_params::<(usize,)>(params).and_then(|(limit,)| { | ||||||
|  | 			take_weak!(self.miner).set_transactions_limit(limit); | ||||||
|  | 			to_value(&true) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn transactions_limit(&self, _: Params) -> Result<Value, Error> { | ||||||
|  | 		to_value(&take_weak!(self.miner).transactions_limit()) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	fn min_gas_price(&self, _: Params) -> Result<Value, Error> { | 	fn min_gas_price(&self, _: Params) -> Result<Value, Error> { | ||||||
| 		to_value(&take_weak!(self.miner).minimal_gas_price()) | 		to_value(&take_weak!(self.miner).minimal_gas_price()) | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -123,3 +123,31 @@ fn rpc_ethcore_set_author() { | |||||||
| 	assert_eq!(io.handle_request(request), Some(response.to_owned())); | 	assert_eq!(io.handle_request(request), Some(response.to_owned())); | ||||||
| 	assert_eq!(miner.author(), Address::from_str("cd1722f3947def4cf144679da39c4c32bdc35681").unwrap()); | 	assert_eq!(miner.author(), Address::from_str("cd1722f3947def4cf144679da39c4c32bdc35681").unwrap()); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn rpc_ethcore_set_transactions_limit() { | ||||||
|  | 	let miner = miner_service(); | ||||||
|  | 	let ethcore = EthcoreClient::new(&miner).to_delegate(); | ||||||
|  | 	let io = IoHandler::new(); | ||||||
|  | 	io.add_delegate(ethcore); | ||||||
|  | 
 | ||||||
|  | 	let request = r#"{"jsonrpc": "2.0", "method": "ethcore_setTransactionsLimit", "params":[10240240], "id": 1}"#; | ||||||
|  | 	let response = r#"{"jsonrpc":"2.0","result":true,"id":1}"#; | ||||||
|  | 
 | ||||||
|  | 	assert_eq!(io.handle_request(request), Some(response.to_owned())); | ||||||
|  | 	assert_eq!(miner.transactions_limit(), 10_240_240); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn rpc_ethcore_transactions_limit() { | ||||||
|  | 	let miner = miner_service(); | ||||||
|  | 	let ethcore = EthcoreClient::new(&miner).to_delegate(); | ||||||
|  | 	let io = IoHandler::new(); | ||||||
|  | 	io.add_delegate(ethcore); | ||||||
|  | 
 | ||||||
|  | 	let request = r#"{"jsonrpc": "2.0", "method": "ethcore_transactionsLimit", "params":[], "id": 1}"#; | ||||||
|  | 	let response = r#"{"jsonrpc":"2.0","result":1024,"id":1}"#; | ||||||
|  | 
 | ||||||
|  | 	assert_eq!(io.handle_request(request), Some(response.to_owned())); | ||||||
|  | } | ||||||
|  | |||||||
| @ -39,6 +39,7 @@ pub struct TestMinerService { | |||||||
| 	gas_floor_target: RwLock<U256>, | 	gas_floor_target: RwLock<U256>, | ||||||
| 	author: RwLock<Address>, | 	author: RwLock<Address>, | ||||||
| 	extra_data: RwLock<Bytes>, | 	extra_data: RwLock<Bytes>, | ||||||
|  | 	limit: RwLock<usize>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Default for TestMinerService { | impl Default for TestMinerService { | ||||||
| @ -52,6 +53,7 @@ impl Default for TestMinerService { | |||||||
| 			gas_floor_target: RwLock::new(U256::from(12345)), | 			gas_floor_target: RwLock::new(U256::from(12345)), | ||||||
| 			author: RwLock::new(Address::zero()), | 			author: RwLock::new(Address::zero()), | ||||||
| 			extra_data: RwLock::new(vec![1, 2, 3, 4]), | 			extra_data: RwLock::new(vec![1, 2, 3, 4]), | ||||||
|  | 			limit: RwLock::new(1024), | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -84,6 +86,14 @@ impl MinerService for TestMinerService { | |||||||
| 		*self.min_gas_price.write().unwrap() = min_gas_price; | 		*self.min_gas_price.write().unwrap() = min_gas_price; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	fn set_transactions_limit(&self, limit: usize) { | ||||||
|  | 		*self.limit.write().unwrap() = limit; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn transactions_limit(&self) -> usize { | ||||||
|  | 		*self.limit.read().unwrap() | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	fn author(&self) -> Address { | 	fn author(&self) -> Address { | ||||||
| 		*self.author.read().unwrap() | 		*self.author.read().unwrap() | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -33,6 +33,12 @@ pub trait Ethcore: Sized + Send + Sync + 'static { | |||||||
| 	/// Sets new author for mined block.
 | 	/// Sets new author for mined block.
 | ||||||
| 	fn set_author(&self, _: Params) -> Result<Value, Error> { rpc_unimplemented!() } | 	fn set_author(&self, _: Params) -> Result<Value, Error> { rpc_unimplemented!() } | ||||||
| 
 | 
 | ||||||
|  | 	/// Sets the limits for transaction queue.
 | ||||||
|  | 	fn set_transactions_limit(&self, _: Params) -> Result<Value, Error> { rpc_unimplemented!() } | ||||||
|  | 
 | ||||||
|  | 	/// Returns current transactions limit.
 | ||||||
|  | 	fn transactions_limit(&self, _: Params) -> Result<Value, Error> { rpc_unimplemented!() } | ||||||
|  | 
 | ||||||
| 	/// Returns mining extra data.
 | 	/// Returns mining extra data.
 | ||||||
| 	fn extra_data(&self, _: Params) -> Result<Value, Error> { rpc_unimplemented!() } | 	fn extra_data(&self, _: Params) -> Result<Value, Error> { rpc_unimplemented!() } | ||||||
| 
 | 
 | ||||||
| @ -49,10 +55,12 @@ pub trait Ethcore: Sized + Send + Sync + 'static { | |||||||
| 		delegate.add_method("ethcore_setGasFloorTarget", Ethcore::set_gas_floor_target); | 		delegate.add_method("ethcore_setGasFloorTarget", Ethcore::set_gas_floor_target); | ||||||
| 		delegate.add_method("ethcore_setExtraData", Ethcore::set_extra_data); | 		delegate.add_method("ethcore_setExtraData", Ethcore::set_extra_data); | ||||||
| 		delegate.add_method("ethcore_setAuthor", Ethcore::set_author); | 		delegate.add_method("ethcore_setAuthor", Ethcore::set_author); | ||||||
|  | 		delegate.add_method("ethcore_setTransactionsLimit", Ethcore::set_transactions_limit); | ||||||
| 
 | 
 | ||||||
| 		delegate.add_method("ethcore_extraData", Ethcore::extra_data); | 		delegate.add_method("ethcore_extraData", Ethcore::extra_data); | ||||||
| 		delegate.add_method("ethcore_gasFloorTarget", Ethcore::gas_floor_target); | 		delegate.add_method("ethcore_gasFloorTarget", Ethcore::gas_floor_target); | ||||||
| 		delegate.add_method("ethcore_minGasPrice", Ethcore::min_gas_price); | 		delegate.add_method("ethcore_minGasPrice", Ethcore::min_gas_price); | ||||||
|  | 		delegate.add_method("ethcore_transactionsLimit", Ethcore::transactions_limit); | ||||||
| 		delegate | 		delegate | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -44,7 +44,7 @@ | |||||||
| //! fn main() {
 | //! fn main() {
 | ||||||
| //! 	let mut service = NetworkService::start(NetworkConfiguration::new()).unwrap();
 | //! 	let mut service = NetworkService::start(NetworkConfiguration::new()).unwrap();
 | ||||||
| //! 	let dir = env::temp_dir();
 | //! 	let dir = env::temp_dir();
 | ||||||
| //! 	let client = Client::new(ClientConfig::default(), ethereum::new_frontier(), &dir, service.io().channel()).unwrap();
 | //! 	let client = Client::new(ClientConfig::default(), ethereum::new_frontier(), &dir, service.io().channel());
 | ||||||
| //! 	let miner = Miner::new(false);
 | //! 	let miner = Miner::new(false);
 | ||||||
| //! 	EthSync::register(&mut service, SyncConfig::default(), client, miner);
 | //! 	EthSync::register(&mut service, SyncConfig::default(), client, miner);
 | ||||||
| //! }
 | //! }
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user