Split internal sealing from work preparation (#2071)
* separate block preparation methods * Split internal sealing from work sealing, add cli option * replace cli with engine method, simplify * More docs about sealing types. Bypass work in external txs. * split requires_reseal, add test and new test miner
This commit is contained in:
		
							parent
							
								
									83ddce011d
								
							
						
					
					
						commit
						341e06481e
					
				| @ -99,6 +99,8 @@ impl Engine for BasicAuthority { | |||||||
| 	/// This assumes that all uncles are valid uncles (i.e. of at least one generation before the current).
 | 	/// This assumes that all uncles are valid uncles (i.e. of at least one generation before the current).
 | ||||||
| 	fn on_close_block(&self, _block: &mut ExecutedBlock) {} | 	fn on_close_block(&self, _block: &mut ExecutedBlock) {} | ||||||
| 
 | 
 | ||||||
|  | 	fn seals_internally(&self) -> bool { true } | ||||||
|  | 
 | ||||||
| 	/// Attempt to seal the block internally.
 | 	/// Attempt to seal the block internally.
 | ||||||
| 	///
 | 	///
 | ||||||
| 	/// This operation is synchronous and may (quite reasonably) not be available, in which `false` will
 | 	/// This operation is synchronous and may (quite reasonably) not be available, in which `false` will
 | ||||||
|  | |||||||
| @ -58,6 +58,8 @@ impl Engine for InstantSeal { | |||||||
| 		Schedule::new_homestead() | 		Schedule::new_homestead() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	fn seals_internally(&self) -> bool { true } | ||||||
|  | 
 | ||||||
| 	fn generate_seal(&self, _block: &ExecutedBlock, _accounts: Option<&AccountProvider>) -> Option<Vec<Bytes>> { | 	fn generate_seal(&self, _block: &ExecutedBlock, _accounts: Option<&AccountProvider>) -> Option<Vec<Bytes>> { | ||||||
| 		Some(Vec::new()) | 		Some(Vec::new()) | ||||||
| 	} | 	} | ||||||
| @ -71,18 +73,12 @@ mod tests { | |||||||
| 	use spec::Spec; | 	use spec::Spec; | ||||||
| 	use block::*; | 	use block::*; | ||||||
| 
 | 
 | ||||||
| 	/// Create a new test chain spec with `BasicAuthority` consensus engine.
 |  | ||||||
| 	fn new_test_instant() -> Spec { |  | ||||||
| 		let bytes: &[u8] = include_bytes!("../../res/instant_seal.json"); |  | ||||||
| 		Spec::load(bytes).expect("invalid chain spec") |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	#[test] | 	#[test] | ||||||
| 	fn instant_can_seal() { | 	fn instant_can_seal() { | ||||||
| 		let tap = AccountProvider::transient_provider(); | 		let tap = AccountProvider::transient_provider(); | ||||||
| 		let addr = tap.insert_account("".sha3(), "").unwrap(); | 		let addr = tap.insert_account("".sha3(), "").unwrap(); | ||||||
| 
 | 
 | ||||||
| 		let spec = new_test_instant(); | 		let spec = Spec::new_test_instant(); | ||||||
| 		let engine = &*spec.engine; | 		let engine = &*spec.engine; | ||||||
| 		let genesis_header = spec.genesis_header(); | 		let genesis_header = spec.genesis_header(); | ||||||
| 		let mut db_result = get_temp_journal_db(); | 		let mut db_result = get_temp_journal_db(); | ||||||
| @ -98,7 +94,7 @@ mod tests { | |||||||
| 
 | 
 | ||||||
| 	#[test] | 	#[test] | ||||||
| 	fn instant_cant_verify() { | 	fn instant_cant_verify() { | ||||||
| 		let engine = new_test_instant().engine; | 		let engine = Spec::new_test_instant().engine; | ||||||
| 		let mut header: Header = Header::default(); | 		let mut header: Header = Header::default(); | ||||||
| 
 | 
 | ||||||
| 		assert!(engine.verify_block_basic(&header, None).is_ok()); | 		assert!(engine.verify_block_basic(&header, None).is_ok()); | ||||||
|  | |||||||
| @ -71,6 +71,8 @@ pub trait Engine : Sync + Send { | |||||||
| 	/// Block transformation functions, after the transactions.
 | 	/// Block transformation functions, after the transactions.
 | ||||||
| 	fn on_close_block(&self, _block: &mut ExecutedBlock) {} | 	fn on_close_block(&self, _block: &mut ExecutedBlock) {} | ||||||
| 
 | 
 | ||||||
|  | 	/// If true, generate_seal has to be implemented.
 | ||||||
|  | 	fn seals_internally(&self) -> bool { false } | ||||||
| 	/// Attempt to seal the block internally.
 | 	/// Attempt to seal the block internally.
 | ||||||
| 	///
 | 	///
 | ||||||
| 	/// If `Some` is returned, then you get a valid seal.
 | 	/// If `Some` is returned, then you get a valid seal.
 | ||||||
|  | |||||||
| @ -24,7 +24,7 @@ use views::{BlockView, HeaderView}; | |||||||
| use state::State; | use state::State; | ||||||
| use client::{MiningBlockChainClient, Executive, Executed, EnvInfo, TransactOptions, BlockID, CallAnalytics}; | use client::{MiningBlockChainClient, Executive, Executed, EnvInfo, TransactOptions, BlockID, CallAnalytics}; | ||||||
| use executive::contract_address; | use executive::contract_address; | ||||||
| use block::{ClosedBlock, IsBlock, Block}; | use block::{ClosedBlock, SealedBlock, IsBlock, Block}; | ||||||
| use error::*; | use error::*; | ||||||
| use transaction::{Action, SignedTransaction}; | use transaction::{Action, SignedTransaction}; | ||||||
| use receipt::{Receipt, RichReceipt}; | use receipt::{Receipt, RichReceipt}; | ||||||
| @ -34,6 +34,7 @@ use miner::{MinerService, MinerStatus, TransactionQueue, AccountDetails, Transac | |||||||
| use miner::work_notify::WorkPoster; | use miner::work_notify::WorkPoster; | ||||||
| use client::TransactionImportResult; | use client::TransactionImportResult; | ||||||
| use miner::price_info::PriceInfo; | use miner::price_info::PriceInfo; | ||||||
|  | use header::BlockNumber; | ||||||
| 
 | 
 | ||||||
| /// Different possible definitions for pending transaction set.
 | /// Different possible definitions for pending transaction set.
 | ||||||
| #[derive(Debug, PartialEq)] | #[derive(Debug, PartialEq)] | ||||||
| @ -165,6 +166,7 @@ struct SealingWork { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Keeps track of transactions using priority queue and holds currently mined block.
 | /// Keeps track of transactions using priority queue and holds currently mined block.
 | ||||||
|  | /// Handles preparing work for "work sealing" or seals "internally" if Engine does not require work.
 | ||||||
| pub struct Miner { | pub struct Miner { | ||||||
| 	// NOTE [ToDr]  When locking always lock in this order!
 | 	// NOTE [ToDr]  When locking always lock in this order!
 | ||||||
| 	transaction_queue: Arc<Mutex<TransactionQueue>>, | 	transaction_queue: Arc<Mutex<TransactionQueue>>, | ||||||
| @ -243,19 +245,15 @@ impl Miner { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Prepares new block for sealing including top transactions from queue.
 | 	/// Prepares new block for sealing including top transactions from queue.
 | ||||||
| 	#[cfg_attr(feature="dev", allow(match_same_arms))] | 	fn prepare_block(&self, chain: &MiningBlockChainClient) -> (ClosedBlock, Option<H256>) { | ||||||
| 	#[cfg_attr(feature="dev", allow(cyclomatic_complexity))] |  | ||||||
| 	fn prepare_sealing(&self, chain: &MiningBlockChainClient) { |  | ||||||
| 		trace!(target: "miner", "prepare_sealing: entering"); |  | ||||||
| 
 |  | ||||||
| 		{ | 		{ | ||||||
| 			trace!(target: "miner", "recalibrating..."); | 			trace!(target: "miner", "prepare_block: recalibrating..."); | ||||||
| 			let txq = self.transaction_queue.clone(); | 			let txq = self.transaction_queue.clone(); | ||||||
| 			self.gas_pricer.lock().recalibrate(move |price| { | 			self.gas_pricer.lock().recalibrate(move |price| { | ||||||
| 				trace!(target: "miner", "Got gas price! {}", price); | 				trace!(target: "miner", "prepare_block: Got gas price! {}", price); | ||||||
| 				txq.lock().set_minimal_gas_price(price); | 				txq.lock().set_minimal_gas_price(price); | ||||||
| 			}); | 			}); | ||||||
| 			trace!(target: "miner", "done recalibration."); | 			trace!(target: "miner", "prepare_block: done recalibration."); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		let (transactions, mut open_block, original_work_hash) = { | 		let (transactions, mut open_block, original_work_hash) = { | ||||||
| @ -273,13 +271,13 @@ impl Miner { | |||||||
| */ | */ | ||||||
| 			let open_block = match sealing_work.queue.pop_if(|b| b.block().fields().header.parent_hash() == &best_hash) { | 			let open_block = match sealing_work.queue.pop_if(|b| b.block().fields().header.parent_hash() == &best_hash) { | ||||||
| 				Some(old_block) => { | 				Some(old_block) => { | ||||||
| 					trace!(target: "miner", "Already have previous work; updating and returning"); | 					trace!(target: "miner", "prepare_block: Already have previous work; updating and returning"); | ||||||
| 					// add transactions to old_block
 | 					// add transactions to old_block
 | ||||||
| 					old_block.reopen(&*self.engine) | 					old_block.reopen(&*self.engine) | ||||||
| 				} | 				} | ||||||
| 				None => { | 				None => { | ||||||
| 					// block not found - create it.
 | 					// block not found - create it.
 | ||||||
| 					trace!(target: "miner", "No existing work - making new block"); | 					trace!(target: "miner", "prepare_block: No existing work - making new block"); | ||||||
| 					chain.prepare_open_block( | 					chain.prepare_open_block( | ||||||
| 						self.author(), | 						self.author(), | ||||||
| 						(self.gas_floor_target(), self.gas_ceil_target()), | 						(self.gas_floor_target(), self.gas_ceil_target()), | ||||||
| @ -334,37 +332,79 @@ impl Miner { | |||||||
| 				queue.remove_invalid(&hash, &fetch_account); | 				queue.remove_invalid(&hash, &fetch_account); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 		(block, original_work_hash) | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 		if !block.transactions().is_empty() { | 	/// Check is reseal is allowed and necessary.
 | ||||||
| 			trace!(target: "miner", "prepare_sealing: block has transaction - attempting internal seal."); | 	fn requires_reseal(&self, best_block: BlockNumber) -> bool { | ||||||
| 			// block with transactions - see if we can seal immediately.
 | 		let has_local_transactions = self.transaction_queue.lock().has_local_pending_transactions(); | ||||||
|  | 		let mut sealing_work = self.sealing_work.lock(); | ||||||
|  | 		if sealing_work.enabled { | ||||||
|  | 			trace!(target: "miner", "requires_reseal: sealing enabled"); | ||||||
|  | 			let last_request = *self.sealing_block_last_request.lock(); | ||||||
|  | 			let should_disable_sealing = !self.forced_sealing() | ||||||
|  | 				&& !has_local_transactions | ||||||
|  | 				&& best_block > last_request | ||||||
|  | 				&& best_block - last_request > SEALING_TIMEOUT_IN_BLOCKS; | ||||||
|  | 
 | ||||||
|  | 			trace!(target: "miner", "requires_reseal: should_disable_sealing={}; best_block={}, last_request={}", should_disable_sealing, best_block, last_request); | ||||||
|  | 
 | ||||||
|  | 			if should_disable_sealing { | ||||||
|  | 				trace!(target: "miner", "Miner sleeping (current {}, last {})", best_block, last_request); | ||||||
|  | 				sealing_work.enabled = false; | ||||||
|  | 				sealing_work.queue.reset(); | ||||||
|  | 				false | ||||||
|  | 			} else { | ||||||
|  | 				// sealing enabled and we don't want to sleep.
 | ||||||
|  | 				*self.next_allowed_reseal.lock() = Instant::now() + self.options.reseal_min_period; | ||||||
|  | 				true | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			// sealing is disabled.
 | ||||||
|  | 			false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Attempts to perform internal sealing (one that does not require work) to return Ok(sealed),
 | ||||||
|  | 	/// Err(Some(block)) returns for unsuccesful sealing while Err(None) indicates misspecified engine.
 | ||||||
|  | 	fn seal_block_internally(&self, block: ClosedBlock) -> Result<SealedBlock, Option<ClosedBlock>> { 
 | ||||||
|  | 		trace!(target: "miner", "seal_block_internally: block has transaction - attempting internal seal."); | ||||||
| 		let s = self.engine.generate_seal(block.block(), match self.accounts { | 		let s = self.engine.generate_seal(block.block(), match self.accounts { | ||||||
| 			Some(ref x) => Some(&**x), | 			Some(ref x) => Some(&**x), | ||||||
| 			None => None, | 			None => None, | ||||||
| 		}); | 		}); | ||||||
| 		if let Some(seal) = s { | 		if let Some(seal) = s { | ||||||
| 				trace!(target: "miner", "prepare_sealing: managed internal seal. importing..."); | 			trace!(target: "miner", "seal_block_internally: managed internal seal. importing..."); | ||||||
| 				if let Ok(sealed) = block.lock().try_seal(&*self.engine, seal) { | 			block.lock().try_seal(&*self.engine, seal).or_else(|_| { | ||||||
| 					if let Ok(_) = chain.import_block(sealed.rlp_bytes()) { |  | ||||||
| 						trace!(target: "miner", "prepare_sealing: sealed internally and imported. leaving."); |  | ||||||
| 					} else { |  | ||||||
| 						warn!("prepare_sealing: ERROR: could not import internally sealed block. WTF?"); |  | ||||||
| 					} |  | ||||||
| 				} else { |  | ||||||
| 				warn!("prepare_sealing: ERROR: try_seal failed when given internally generated seal. WTF?"); | 				warn!("prepare_sealing: ERROR: try_seal failed when given internally generated seal. WTF?"); | ||||||
| 				} | 				Err(None) | ||||||
| 				return; | 			}) | ||||||
| 		} else { | 		} else { | ||||||
| 				trace!(target: "miner", "prepare_sealing: unable to generate seal internally"); | 			trace!(target: "miner", "seal_block_internally: unable to generate seal internally"); | ||||||
|  | 			Err(Some(block)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/// Uses Engine to seal the block internally and then imports it to chain.
 | ||||||
|  | 	fn seal_and_import_block_internally(&self, chain: &MiningBlockChainClient, block: ClosedBlock) -> bool { | ||||||
|  | 		if !block.transactions().is_empty() { | ||||||
|  | 			if let Ok(sealed) = self.seal_block_internally(block) { | ||||||
|  | 				if chain.import_block(sealed.rlp_bytes()).is_ok() { | ||||||
|  | 					return true | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		false | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Prepares work which has to be done to seal.
 | ||||||
|  | 	fn prepare_work(&self, block: ClosedBlock, original_work_hash: Option<H256>) { | ||||||
| 		let (work, is_new) = { | 		let (work, is_new) = { | ||||||
| 			let mut sealing_work = self.sealing_work.lock(); | 			let mut sealing_work = self.sealing_work.lock(); | ||||||
| 			let last_work_hash = sealing_work.queue.peek_last_ref().map(|pb| pb.block().fields().header.hash()); | 			let last_work_hash = sealing_work.queue.peek_last_ref().map(|pb| pb.block().fields().header.hash()); | ||||||
| 			trace!(target: "miner", "Checking whether we need to reseal: orig={:?} last={:?}, this={:?}", original_work_hash, last_work_hash, block.block().fields().header.hash()); | 			trace!(target: "miner", "prepare_work: Checking whether we need to reseal: orig={:?} last={:?}, this={:?}", original_work_hash, last_work_hash, block.block().fields().header.hash()); | ||||||
| 			let (work, is_new) = if last_work_hash.map_or(true, |h| h != block.block().fields().header.hash()) { | 			let (work, is_new) = if last_work_hash.map_or(true, |h| h != block.block().fields().header.hash()) { | ||||||
| 				trace!(target: "miner", "Pushing a new, refreshed or borrowed pending {}...", block.block().fields().header.hash()); | 				trace!(target: "miner", "prepare_work: Pushing a new, refreshed or borrowed pending {}...", block.block().fields().header.hash()); | ||||||
| 				let pow_hash = block.block().fields().header.hash(); | 				let pow_hash = block.block().fields().header.hash(); | ||||||
| 				let number = block.block().fields().header.number(); | 				let number = block.block().fields().header.number(); | ||||||
| 				let difficulty = *block.block().fields().header.difficulty(); | 				let difficulty = *block.block().fields().header.difficulty(); | ||||||
| @ -378,7 +418,7 @@ impl Miner { | |||||||
| 			} else { | 			} else { | ||||||
| 				(None, false) | 				(None, false) | ||||||
| 			}; | 			}; | ||||||
| 			trace!(target: "miner", "prepare_sealing: leaving (last={:?})", sealing_work.queue.peek_last_ref().map(|b| b.block().fields().header.hash())); | 			trace!(target: "miner", "prepare_work: leaving (last={:?})", sealing_work.queue.peek_last_ref().map(|b| b.block().fields().header.hash())); | ||||||
| 			(work, is_new) | 			(work, is_new) | ||||||
| 		}; | 		}; | ||||||
| 		if is_new { | 		if is_new { | ||||||
| @ -392,13 +432,13 @@ impl Miner { | |||||||
| 		queue.set_gas_limit(gas_limit); | 		queue.set_gas_limit(gas_limit); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Returns true if we had to prepare new pending block
 | 	/// Returns true if we had to prepare new pending block.
 | ||||||
| 	fn enable_and_prepare_sealing(&self, chain: &MiningBlockChainClient) -> bool { | 	fn prepare_work_sealing(&self, chain: &MiningBlockChainClient) -> bool { | ||||||
| 		trace!(target: "miner", "enable_and_prepare_sealing: entering"); | 		trace!(target: "miner", "prepare_work_sealing: entering"); | ||||||
| 		let prepare_new = { | 		let prepare_new = { | ||||||
| 			let mut sealing_work = self.sealing_work.lock(); | 			let mut sealing_work = self.sealing_work.lock(); | ||||||
| 			let have_work = sealing_work.queue.peek_last_ref().is_some(); | 			let have_work = sealing_work.queue.peek_last_ref().is_some(); | ||||||
| 			trace!(target: "miner", "enable_and_prepare_sealing: have_work={}", have_work); | 			trace!(target: "miner", "prepare_work_sealing: have_work={}", have_work); | ||||||
| 			if !have_work { | 			if !have_work { | ||||||
| 				sealing_work.enabled = true; | 				sealing_work.enabled = true; | ||||||
| 				true | 				true | ||||||
| @ -411,12 +451,13 @@ impl Miner { | |||||||
| 			// | NOTE Code below requires transaction_queue and sealing_work locks.     |
 | 			// | NOTE Code below requires transaction_queue and sealing_work locks.     |
 | ||||||
| 			// | Make sure to release the locks before calling that method.             |
 | 			// | Make sure to release the locks before calling that method.             |
 | ||||||
| 			// --------------------------------------------------------------------------
 | 			// --------------------------------------------------------------------------
 | ||||||
| 			self.prepare_sealing(chain); | 			let (block, original_work_hash) = self.prepare_block(chain); | ||||||
|  | 			self.prepare_work(block, original_work_hash); | ||||||
| 		} | 		} | ||||||
| 		let mut sealing_block_last_request = self.sealing_block_last_request.lock(); | 		let mut sealing_block_last_request = self.sealing_block_last_request.lock(); | ||||||
| 		let best_number = chain.chain_info().best_block_number; | 		let best_number = chain.chain_info().best_block_number; | ||||||
| 		if *sealing_block_last_request != best_number { | 		if *sealing_block_last_request != best_number { | ||||||
| 			trace!(target: "miner", "enable_and_prepare_sealing: Miner received request (was {}, now {}) - waking up.", *sealing_block_last_request, best_number); | 			trace!(target: "miner", "prepare_work_sealing: Miner received request (was {}, now {}) - waking up.", *sealing_block_last_request, best_number); | ||||||
| 			*sealing_block_last_request = best_number; | 			*sealing_block_last_request = best_number; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -635,7 +676,7 @@ impl MinerService for Miner { | |||||||
| 		trace!(target: "own_tx", "Importing transaction: {:?}", transaction); | 		trace!(target: "own_tx", "Importing transaction: {:?}", transaction); | ||||||
| 
 | 
 | ||||||
| 		let imported = { | 		let imported = { | ||||||
| 			// Be sure to release the lock before we call enable_and_prepare_sealing
 | 			// Be sure to release the lock before we call prepare_work_sealing
 | ||||||
| 			let mut transaction_queue = self.transaction_queue.lock(); | 			let mut transaction_queue = self.transaction_queue.lock(); | ||||||
| 			let import = self.add_transactions_to_queue( | 			let import = self.add_transactions_to_queue( | ||||||
| 				chain, vec![transaction], TransactionOrigin::Local, &mut transaction_queue | 				chain, vec![transaction], TransactionOrigin::Local, &mut transaction_queue | ||||||
| @ -661,11 +702,11 @@ impl MinerService for Miner { | |||||||
| 		// --------------------------------------------------------------------------
 | 		// --------------------------------------------------------------------------
 | ||||||
| 		if imported.is_ok() && self.options.reseal_on_own_tx && self.tx_reseal_allowed() { | 		if imported.is_ok() && self.options.reseal_on_own_tx && self.tx_reseal_allowed() { | ||||||
| 			// Make sure to do it after transaction is imported and lock is droped.
 | 			// Make sure to do it after transaction is imported and lock is droped.
 | ||||||
| 			// We need to create pending block and enable sealing
 | 			// We need to create pending block and enable sealing.
 | ||||||
| 			let prepared = self.enable_and_prepare_sealing(chain); | 			if self.engine.seals_internally() || !self.prepare_work_sealing(chain) { | ||||||
| 				// If new block has not been prepared (means we already had one)
 | 				// If new block has not been prepared (means we already had one)
 | ||||||
| 			// we need to update sealing
 | 				// or Engine might be able to seal internally,
 | ||||||
| 			if !prepared { | 				// we need to update sealing.
 | ||||||
| 				self.update_sealing(chain); | 				self.update_sealing(chain); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| @ -767,44 +808,26 @@ impl MinerService for Miner { | |||||||
| 		self.transaction_queue.lock().last_nonce(address) | 		self.transaction_queue.lock().last_nonce(address) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 	/// Update sealing if required.
 | ||||||
|  | 	/// Prepare the block and work if the Engine does not seal internally.
 | ||||||
| 	fn update_sealing(&self, chain: &MiningBlockChainClient) { | 	fn update_sealing(&self, chain: &MiningBlockChainClient) { | ||||||
| 		trace!(target: "miner", "update_sealing"); | 		trace!(target: "miner", "update_sealing"); | ||||||
| 		let requires_reseal = { |  | ||||||
| 			let has_local_transactions = self.transaction_queue.lock().has_local_pending_transactions(); |  | ||||||
| 			let mut sealing_work = self.sealing_work.lock(); |  | ||||||
| 			if sealing_work.enabled { |  | ||||||
| 				trace!(target: "miner", "update_sealing: sealing enabled"); |  | ||||||
| 				let current_no = chain.chain_info().best_block_number; |  | ||||||
| 				let last_request = *self.sealing_block_last_request.lock(); |  | ||||||
| 				let should_disable_sealing = !self.forced_sealing() |  | ||||||
| 					&& !has_local_transactions |  | ||||||
| 					&& current_no > last_request |  | ||||||
| 					&& current_no - last_request > SEALING_TIMEOUT_IN_BLOCKS; |  | ||||||
| 
 | 
 | ||||||
| 				trace!(target: "miner", "update_sealing: should_disable_sealing={}; current_no={}, last_request={}", should_disable_sealing, current_no, last_request); | 		if self.requires_reseal(chain.chain_info().best_block_number) { | ||||||
| 
 |  | ||||||
| 				if should_disable_sealing { |  | ||||||
| 					trace!(target: "miner", "Miner sleeping (current {}, last {})", current_no, last_request); |  | ||||||
| 					sealing_work.enabled = false; |  | ||||||
| 					sealing_work.queue.reset(); |  | ||||||
| 					false |  | ||||||
| 				} else { |  | ||||||
| 					// sealing enabled and we don't want to sleep.
 |  | ||||||
| 					*self.next_allowed_reseal.lock() = Instant::now() + self.options.reseal_min_period; |  | ||||||
| 					true |  | ||||||
| 				} |  | ||||||
| 			} else { |  | ||||||
| 				// sealing is disabled.
 |  | ||||||
| 				false |  | ||||||
| 			} |  | ||||||
| 		}; |  | ||||||
| 
 |  | ||||||
| 		if requires_reseal { |  | ||||||
| 			// --------------------------------------------------------------------------
 | 			// --------------------------------------------------------------------------
 | ||||||
| 			// | NOTE Code below requires transaction_queue and sealing_work locks.     |
 | 			// | NOTE Code below requires transaction_queue and sealing_work locks.     |
 | ||||||
| 			// | Make sure to release the locks before calling that method.             |
 | 			// | Make sure to release the locks before calling that method.             |
 | ||||||
| 			// --------------------------------------------------------------------------
 | 			// --------------------------------------------------------------------------
 | ||||||
| 			self.prepare_sealing(chain); | 			trace!(target: "miner", "update_sealing: preparing a block"); | ||||||
|  | 			let (block, original_work_hash) = self.prepare_block(chain); | ||||||
|  | 			if self.engine.seals_internally() { | ||||||
|  | 				trace!(target: "miner", "update_sealing: engine indicates internal sealing"); | ||||||
|  | 				self.seal_and_import_block_internally(chain, block); | ||||||
|  | 			} else { | ||||||
|  | 				trace!(target: "miner", "update_sealing: engine does not seal internally, preparing work"); | ||||||
|  | 				self.prepare_work(block, original_work_hash); | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -814,7 +837,7 @@ impl MinerService for Miner { | |||||||
| 
 | 
 | ||||||
| 	fn map_sealing_work<F, T>(&self, chain: &MiningBlockChainClient, f: F) -> Option<T> where F: FnOnce(&ClosedBlock) -> T { | 	fn map_sealing_work<F, T>(&self, chain: &MiningBlockChainClient, f: F) -> Option<T> where F: FnOnce(&ClosedBlock) -> T { | ||||||
| 		trace!(target: "miner", "map_sealing_work: entering"); | 		trace!(target: "miner", "map_sealing_work: entering"); | ||||||
| 		self.enable_and_prepare_sealing(chain); | 		self.prepare_work_sealing(chain); | ||||||
| 		trace!(target: "miner", "map_sealing_work: sealing prepared"); | 		trace!(target: "miner", "map_sealing_work: sealing prepared"); | ||||||
| 		let mut sealing_work = self.sealing_work.lock(); | 		let mut sealing_work = self.sealing_work.lock(); | ||||||
| 		let ret = sealing_work.queue.use_last_ref(); | 		let ret = sealing_work.queue.use_last_ref(); | ||||||
| @ -917,11 +940,12 @@ mod tests { | |||||||
| 	use super::*; | 	use super::*; | ||||||
| 	use util::*; | 	use util::*; | ||||||
| 	use ethkey::{Generator, Random}; | 	use ethkey::{Generator, Random}; | ||||||
| 	use client::{TestBlockChainClient, EachBlockWith}; | 	use client::{BlockChainClient, TestBlockChainClient, EachBlockWith, TransactionImportResult}; | ||||||
| 	use client::{TransactionImportResult}; | 	use header::BlockNumber; | ||||||
| 	use types::transaction::{Transaction, Action}; | 	use types::transaction::{Transaction, SignedTransaction, Action}; | ||||||
| 	use block::*; | 	use block::*; | ||||||
| 	use spec::Spec; | 	use spec::Spec; | ||||||
|  | 	use tests::helpers::{generate_dummy_client}; | ||||||
| 
 | 
 | ||||||
| 	#[test] | 	#[test] | ||||||
| 	fn should_prepare_block_to_seal() { | 	fn should_prepare_block_to_seal() { | ||||||
| @ -975,12 +999,7 @@ mod tests { | |||||||
| 		)).ok().expect("Miner was just created.") | 		)).ok().expect("Miner was just created.") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	#[test] | 	fn transaction() -> SignedTransaction { | ||||||
| 	fn should_make_pending_block_when_importing_own_transaction() { |  | ||||||
| 		// given
 |  | ||||||
| 		let client = TestBlockChainClient::default(); |  | ||||||
| 		let miner = miner(); |  | ||||||
| 		let transaction = { |  | ||||||
| 		let keypair = Random.generate().unwrap(); | 		let keypair = Random.generate().unwrap(); | ||||||
| 		Transaction { | 		Transaction { | ||||||
| 			action: Action::Create, | 			action: Action::Create, | ||||||
| @ -990,8 +1009,14 @@ mod tests { | |||||||
| 			gas_price: U256::zero(), | 			gas_price: U256::zero(), | ||||||
| 			nonce: U256::zero(), | 			nonce: U256::zero(), | ||||||
| 		}.sign(keypair.secret()) | 		}.sign(keypair.secret()) | ||||||
| 		}; | 	} | ||||||
| 
 | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn should_make_pending_block_when_importing_own_transaction() { | ||||||
|  | 		// given
 | ||||||
|  | 		let client = TestBlockChainClient::default(); | ||||||
|  | 		let miner = miner(); | ||||||
|  | 		let transaction = transaction(); | ||||||
| 		// when
 | 		// when
 | ||||||
| 		let res = miner.import_own_transaction(&client, transaction); | 		let res = miner.import_own_transaction(&client, transaction); | ||||||
| 
 | 
 | ||||||
| @ -1002,7 +1027,7 @@ mod tests { | |||||||
| 		assert_eq!(miner.pending_transactions_hashes().len(), 1); | 		assert_eq!(miner.pending_transactions_hashes().len(), 1); | ||||||
| 		assert_eq!(miner.pending_receipts().len(), 1); | 		assert_eq!(miner.pending_receipts().len(), 1); | ||||||
| 		// This method will let us know if pending block was created (before calling that method)
 | 		// This method will let us know if pending block was created (before calling that method)
 | ||||||
| 		assert_eq!(miner.enable_and_prepare_sealing(&client), false); | 		assert_eq!(miner.prepare_work_sealing(&client), false); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	#[test] | 	#[test] | ||||||
| @ -1010,18 +1035,7 @@ mod tests { | |||||||
| 		// given
 | 		// given
 | ||||||
| 		let client = TestBlockChainClient::default(); | 		let client = TestBlockChainClient::default(); | ||||||
| 		let miner = miner(); | 		let miner = miner(); | ||||||
| 		let transaction = { | 		let transaction = transaction(); | ||||||
| 			let keypair = Random.generate().unwrap(); |  | ||||||
| 			Transaction { |  | ||||||
| 				action: Action::Create, |  | ||||||
| 				value: U256::zero(), |  | ||||||
| 				data: "3331600055".from_hex().unwrap(), |  | ||||||
| 				gas: U256::from(100_000), |  | ||||||
| 				gas_price: U256::zero(), |  | ||||||
| 				nonce: U256::zero(), |  | ||||||
| 			}.sign(keypair.secret()) |  | ||||||
| 		}; |  | ||||||
| 
 |  | ||||||
| 		// when
 | 		// when
 | ||||||
| 		let res = miner.import_external_transactions(&client, vec![transaction]).pop().unwrap(); | 		let res = miner.import_external_transactions(&client, vec![transaction]).pop().unwrap(); | ||||||
| 
 | 
 | ||||||
| @ -1032,6 +1046,31 @@ mod tests { | |||||||
| 		assert_eq!(miner.pending_transactions().len(), 0); | 		assert_eq!(miner.pending_transactions().len(), 0); | ||||||
| 		assert_eq!(miner.pending_receipts().len(), 0); | 		assert_eq!(miner.pending_receipts().len(), 0); | ||||||
| 		// This method will let us know if pending block was created (before calling that method)
 | 		// This method will let us know if pending block was created (before calling that method)
 | ||||||
| 		assert_eq!(miner.enable_and_prepare_sealing(&client), true); | 		assert_eq!(miner.prepare_work_sealing(&client), true); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn internal_seals_without_work() { | ||||||
|  | 		let miner = Miner::with_spec(&Spec::new_test_instant()); | ||||||
|  | 		{ | ||||||
|  | 			let mut sealing_work = miner.sealing_work.lock(); | ||||||
|  | 			sealing_work.enabled = true; | ||||||
|  | 		} | ||||||
|  | 		let c = generate_dummy_client(2); | ||||||
|  | 		let client = c.reference().as_ref(); | ||||||
|  | 
 | ||||||
|  | 		assert_eq!(miner.import_external_transactions(client, vec![transaction()]).pop().unwrap().unwrap(), TransactionImportResult::Current); | ||||||
|  | 
 | ||||||
|  | 		miner.update_sealing(client); | ||||||
|  | 		client.flush_queue(); | ||||||
|  | 		assert!(miner.pending_block().is_none()); | ||||||
|  | 		assert_eq!(client.chain_info().best_block_number, 3 as BlockNumber); | ||||||
|  | 
 | ||||||
|  | 		assert_eq!(miner.import_own_transaction(client, transaction()).unwrap(), TransactionImportResult::Current); | ||||||
|  | 
 | ||||||
|  | 		miner.update_sealing(client); | ||||||
|  | 		client.flush_queue(); | ||||||
|  | 		assert!(miner.pending_block().is_none()); | ||||||
|  | 		assert_eq!(client.chain_info().best_block_number, 4 as BlockNumber); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -260,6 +260,11 @@ impl Spec { | |||||||
| 	pub fn new_null() -> Self { | 	pub fn new_null() -> Self { | ||||||
| 		Spec::load(include_bytes!("../../res/null.json") as &[u8]).expect("null.json is invalid") | 		Spec::load(include_bytes!("../../res/null.json") as &[u8]).expect("null.json is invalid") | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Create a new Spec with InstantSeal consensus which does internal sealing (not requiring work).
 | ||||||
|  | 	pub fn new_test_instant() -> Self { | ||||||
|  | 		Spec::load(include_bytes!("../../res/instant_seal.json") as &[u8]).expect("instant_seal.json is invalid") | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user