Add UsingQueue.
This commit is contained in:
		
							parent
							
								
									038b67cbd4
								
							
						
					
					
						commit
						1700b6a087
					
				| @ -39,7 +39,6 @@ pub struct Miner { | |||||||
| 	gas_floor_target: RwLock<U256>, | 	gas_floor_target: RwLock<U256>, | ||||||
| 	author: RwLock<Address>, | 	author: RwLock<Address>, | ||||||
| 	extra_data: RwLock<Bytes>, | 	extra_data: RwLock<Bytes>, | ||||||
| 
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Default for Miner { | impl Default for Miner { | ||||||
|  | |||||||
| @ -124,6 +124,7 @@ pub mod hash; | |||||||
| pub mod bytes; | pub mod bytes; | ||||||
| pub mod rlp; | pub mod rlp; | ||||||
| pub mod misc; | pub mod misc; | ||||||
|  | pub mod using_queue; | ||||||
| mod json_aid; | mod json_aid; | ||||||
| pub mod vector; | pub mod vector; | ||||||
| pub mod sha3; | pub mod sha3; | ||||||
| @ -149,6 +150,7 @@ pub mod table; | |||||||
| 
 | 
 | ||||||
| pub use common::*; | pub use common::*; | ||||||
| pub use misc::*; | pub use misc::*; | ||||||
|  | pub use using_queue::*; | ||||||
| pub use json_aid::*; | pub use json_aid::*; | ||||||
| pub use rlp::*; | pub use rlp::*; | ||||||
| pub use hashdb::*; | pub use hashdb::*; | ||||||
|  | |||||||
							
								
								
									
										207
									
								
								util/src/using_queue.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										207
									
								
								util/src/using_queue.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,207 @@ | |||||||
|  | // 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/>.
 | ||||||
|  | //! Queue-like datastructure including notion of usage.
 | ||||||
|  | 
 | ||||||
|  | /// Special queue-like datastructure that includes the notion of
 | ||||||
|  | /// usage to avoid items that were queued but never used from making it into
 | ||||||
|  | /// the queue.
 | ||||||
|  | pub struct UsingQueue<T> where T: Clone { | ||||||
|  | 	/// Not yet being sealed by a miner, but if one asks for work, we'd prefer they do this.
 | ||||||
|  | 	pending: Option<T>, | ||||||
|  | 	/// Currently being sealed by miners.
 | ||||||
|  | 	in_use: Vec<T>, | ||||||
|  | 	/// The maximum allowable number of items in_use.
 | ||||||
|  | 	max_size: usize, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<T> UsingQueue<T> where T: Clone { | ||||||
|  | 	/// Create a new struct with a maximum size of `max_size`.
 | ||||||
|  | 	pub fn new(max_size: usize) -> UsingQueue<T> { | ||||||
|  | 		UsingQueue { | ||||||
|  | 			pending: None, | ||||||
|  | 			in_use: vec![], | ||||||
|  | 			max_size: max_size, | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Return a reference to the item at the top of the queue (or `None` if the queue is empty);
 | ||||||
|  | 	/// it doesn't constitute noting that the item is used.
 | ||||||
|  | 	pub fn peek_last_ref(&self) -> Option<&T> { | ||||||
|  | 		self.pending.as_ref().or(self.in_use.last()) | ||||||
|  | 	} | ||||||
|  | 	
 | ||||||
|  | 	/// Return a reference to the item at the top of the queue (or `None` if the queue is empty);
 | ||||||
|  | 	/// this constitutes using the item and will remain in the queue for at least another
 | ||||||
|  | 	/// `max_size` invocations of `push()`.
 | ||||||
|  | 	pub fn use_last_ref(&mut self) -> Option<&T> { | ||||||
|  | 		if let Some(x) = self.pending.take() { | ||||||
|  | 			self.in_use.push(x); | ||||||
|  | 			if self.in_use.len() > self.max_size { | ||||||
|  | 				self.in_use.remove(0); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		self.in_use.last() | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Place an item on the end of the queue. The previously `push()`ed item will be removed
 | ||||||
|  | 	/// if `use_last_ref()` since it was `push()`ed.
 | ||||||
|  | 	pub fn push(&mut self, b: T) { | ||||||
|  | 		self.pending = Some(b); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Clears everything; the queue is entirely reset.
 | ||||||
|  | 	pub fn reset(&mut self) { | ||||||
|  | 		self.pending = None; | ||||||
|  | 		self.in_use.clear(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Returns `Some` reference to first block that `f` returns `true` with it as a parameter
 | ||||||
|  | 	/// or `None` if no such block exists in the queue.
 | ||||||
|  | 	pub fn find_if<P>(&self, predicate: P) -> Option<&T> where P: Fn(&T) -> bool { | ||||||
|  | 		if self.pending.as_ref().map(|r| predicate(r)).unwrap_or(false) { | ||||||
|  | 			self.pending.as_ref() | ||||||
|  | 		} else { | ||||||
|  | 			self.in_use.iter().find(|r| predicate(r)) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/// Returns the most recently pushed block if `f` returns `true` with a reference to it as
 | ||||||
|  | 	/// a parameter, otherwise `None`.
 | ||||||
|  | 	/// Will not destroy a block if a reference to it has previously been returned by `use_last_ref`,
 | ||||||
|  | 	/// but rather clone it.
 | ||||||
|  | 	pub fn pop_if<P>(&mut self, predicate: P) -> Option<T> where P: Fn(&T) -> bool { | ||||||
|  | 		// a bit clumsy - TODO: think about a nicer way of expressing this.
 | ||||||
|  | 		if let Some(x) = self.pending.take() { | ||||||
|  | 			if predicate(&x) { | ||||||
|  | 				Some(x) | ||||||
|  | 			} else { | ||||||
|  | 				self.pending = Some(x); | ||||||
|  | 				None | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			self.in_use.last().into_iter().filter(|x| predicate(x)).next().cloned() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_find_when_pushed() { | ||||||
|  | 	let mut q = UsingQueue::new(2); | ||||||
|  | 	q.push(1); | ||||||
|  | 	assert!(q.find_if(|i| i == &1).is_some()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_find_when_pushed_and_used() { | ||||||
|  | 	let mut q = UsingQueue::new(2); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	assert!(q.find_if(|i| i == &1).is_some()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_find_when_others_used() { | ||||||
|  | 	let mut q = UsingQueue::new(2); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	q.push(2); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	assert!(q.find_if(|i| i == &1).is_some()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_not_find_when_too_many_used() { | ||||||
|  | 	let mut q = UsingQueue::new(1); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	q.push(2); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	assert!(q.find_if(|i| i == &1).is_none()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_not_find_when_not_used_and_then_pushed() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.push(2); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	assert!(q.find_if(|i| i == &1).is_none()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_peek_correctly_after_push() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	assert_eq!(q.peek_last_ref(), Some(&1)); | ||||||
|  | 	q.push(2); | ||||||
|  | 	assert_eq!(q.peek_last_ref(), Some(&2)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_inspect_correctly() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	assert_eq!(q.use_last_ref(), Some(&1)); | ||||||
|  | 	assert_eq!(q.peek_last_ref(), Some(&1)); | ||||||
|  | 	q.push(2); | ||||||
|  | 	assert_eq!(q.use_last_ref(), Some(&2)); | ||||||
|  | 	assert_eq!(q.peek_last_ref(), Some(&2)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_not_find_when_not_used_peeked_and_then_pushed() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.peek_last_ref(); | ||||||
|  | 	q.push(2); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	assert!(q.find_if(|i| i == &1).is_none()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_pop_used() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	let popped = q.pop_if(|i| i == &1); | ||||||
|  | 	assert_eq!(popped, Some(1)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_pop_unused() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	assert_eq!(q.pop_if(|i| i == &1), Some(1)); | ||||||
|  | 	assert_eq!(q.pop_if(|i| i == &1), None); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_not_pop_unused_before_used() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.push(2); | ||||||
|  | 	let popped = q.pop_if(|i| i == &1); | ||||||
|  | 	assert_eq!(popped, None); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_not_remove_used_popped() { | ||||||
|  | 	let mut q = UsingQueue::new(3); | ||||||
|  | 	q.push(1); | ||||||
|  | 	q.use_last_ref(); | ||||||
|  | 	assert_eq!(q.pop_if(|i| i == &1), Some(1)); | ||||||
|  | 	assert_eq!(q.pop_if(|i| i == &1), Some(1)); | ||||||
|  | } | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user