Merge branch 'master' into finduncles
This commit is contained in:
		
						commit
						52dceff3ea
					
				
							
								
								
									
										9
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @ -227,6 +227,7 @@ dependencies = [ | ||||
|  "serde_codegen 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
|  "serde_json 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
|  "syntex 0.29.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
|  "transient-hashmap 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| @ -821,6 +822,14 @@ name = "traitobject" | ||||
| version = "0.0.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| 
 | ||||
| [[package]] | ||||
| name = "transient-hashmap" | ||||
| version = "0.1.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| dependencies = [ | ||||
|  "time 0.1.34 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| name = "typeable" | ||||
| version = "0.1.2" | ||||
|  | ||||
| @ -466,7 +466,8 @@ impl BlockChain { | ||||
| 		let mut write_details = self.block_details.write().unwrap(); | ||||
| 		for (hash, details) in update.block_details.into_iter() { | ||||
| 			batch.put_extras(&hash, &details); | ||||
| 			write_details.insert(hash, details); | ||||
| 			write_details.insert(hash.clone(), details); | ||||
| 			self.note_used(CacheID::Extras(ExtrasIndex::BlockDetails, hash)); | ||||
| 		} | ||||
| 
 | ||||
| 		let mut write_receipts = self.block_receipts.write().unwrap(); | ||||
| @ -802,6 +803,14 @@ impl BlockChain { | ||||
| 
 | ||||
| 				// TODO: handle block_hashes properly.
 | ||||
| 				block_hashes.clear(); | ||||
| 
 | ||||
| 				blocks.shrink_to_fit(); | ||||
| 				block_details.shrink_to_fit(); | ||||
|  				block_hashes.shrink_to_fit(); | ||||
|  				transaction_addresses.shrink_to_fit(); | ||||
|  				block_logs.shrink_to_fit(); | ||||
|  				blocks_blooms.shrink_to_fit(); | ||||
|  				block_receipts.shrink_to_fit(); | ||||
| 			} | ||||
| 			if self.cache_size().total() < self.max_cache_size { break; } | ||||
| 		} | ||||
|  | ||||
| @ -123,6 +123,9 @@ pub trait BlockChainClient : Sync + Send { | ||||
| 	/// Get block total difficulty.
 | ||||
| 	fn block_total_difficulty(&self, id: BlockId) -> Option<U256>; | ||||
| 
 | ||||
| 	/// Get block hash.
 | ||||
| 	fn block_hash(&self, id: BlockId) -> Option<H256>; | ||||
| 
 | ||||
| 	/// Get address code.
 | ||||
| 	fn code(&self, address: &Address) -> Option<Bytes>; | ||||
| 
 | ||||
| @ -540,6 +543,11 @@ impl BlockChainClient for Client { | ||||
| 		Self::block_hash(&chain, id).and_then(|hash| chain.block_details(&hash)).map(|d| d.total_difficulty) | ||||
| 	} | ||||
| 
 | ||||
| 	fn block_hash(&self, id: BlockId) -> Option<H256> { | ||||
| 		let chain = self.chain.read().unwrap(); | ||||
| 		Self::block_hash(&chain, id) | ||||
| 	} | ||||
| 
 | ||||
| 	fn code(&self, address: &Address) -> Option<Bytes> { | ||||
| 		self.state().code(address) | ||||
| 	} | ||||
|  | ||||
| @ -42,6 +42,22 @@ pub struct Filter { | ||||
| 	pub topics: [Option<Vec<H256>>; 4], | ||||
| } | ||||
| 
 | ||||
| impl Clone for Filter { | ||||
| 	fn clone(&self) -> Self { | ||||
| 		let mut topics = [None, None, None, None]; | ||||
| 		for i in 0..4 { | ||||
| 			topics[i] = self.topics[i].clone(); | ||||
| 		} | ||||
| 
 | ||||
| 		Filter { | ||||
| 			from_block: self.from_block.clone(), | ||||
| 			to_block: self.to_block.clone(), | ||||
| 			address: self.address.clone(), | ||||
| 			topics: topics | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| impl Filter { | ||||
| 	/// Returns combinations of each address and topic.	
 | ||||
| 	pub fn bloom_possibilities(&self) -> Vec<H2048> { | ||||
|  | ||||
| @ -108,6 +108,25 @@ fn can_collect_garbage() { | ||||
| 	assert!(client.blockchain_cache_info().blocks < 100 * 1024); | ||||
| } | ||||
| 
 | ||||
| #[test] | ||||
| fn can_handle_long_fork() { | ||||
| 	let client_result = generate_dummy_client(1200); | ||||
| 	let client = client_result.reference(); | ||||
| 	for _ in 0..10 { | ||||
| 		client.import_verified_blocks(&IoChannel::disconnected()); | ||||
| 	} | ||||
| 	assert_eq!(1200, client.chain_info().best_block_number); | ||||
| 
 | ||||
| 	push_blocks_to_client(client, 45, 1201, 800); | ||||
| 	push_blocks_to_client(client, 49, 1201, 800); | ||||
| 	push_blocks_to_client(client, 53, 1201, 600); | ||||
| 
 | ||||
| 	for _ in 0..20 { | ||||
| 		client.import_verified_blocks(&IoChannel::disconnected()); | ||||
| 	} | ||||
| 	assert_eq!(2000, client.chain_info().best_block_number); | ||||
| } | ||||
| 
 | ||||
| #[test] | ||||
| fn can_mine() { | ||||
| 	let dummy_blocks = get_good_dummy_block_seq(2); | ||||
|  | ||||
| @ -156,10 +156,9 @@ pub fn generate_dummy_client(block_number: u32) -> GuardedTempResult<Arc<Client> | ||||
| 		rolling_block_number = rolling_block_number + 1; | ||||
| 		rolling_timestamp = rolling_timestamp + 10; | ||||
| 
 | ||||
| 		if let Err(_) = client.import_block(create_test_block(&header)) { | ||||
| 			panic!("error importing block which is valid by definition"); | ||||
| 		if let Err(e) = client.import_block(create_test_block(&header)) { | ||||
| 			panic!("error importing block which is valid by definition: {:?}", e); | ||||
| 		} | ||||
| 
 | ||||
| 	} | ||||
| 	client.flush_queue(); | ||||
| 	client.import_verified_blocks(&IoChannel::disconnected()); | ||||
| @ -170,6 +169,34 @@ pub fn generate_dummy_client(block_number: u32) -> GuardedTempResult<Arc<Client> | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| pub fn push_blocks_to_client(client: &Arc<Client>, timestamp_salt: u64, starting_number: usize, block_number: usize) { | ||||
| 	let test_spec = get_test_spec(); | ||||
| 	let test_engine = test_spec.to_engine().unwrap(); | ||||
| 	let state_root = test_engine.spec().genesis_header().state_root; | ||||
| 	let mut rolling_hash = client.chain_info().best_block_hash; | ||||
| 	let mut rolling_block_number = starting_number as u64; | ||||
| 	let mut rolling_timestamp = timestamp_salt + starting_number as u64 * 10; | ||||
| 
 | ||||
| 	for _ in 0..block_number { | ||||
| 		let mut header = Header::new(); | ||||
| 
 | ||||
| 		header.gas_limit = decode(test_engine.spec().engine_params.get("minGasLimit").unwrap()); | ||||
| 		header.difficulty = decode(test_engine.spec().engine_params.get("minimumDifficulty").unwrap()); | ||||
| 		header.timestamp = rolling_timestamp; | ||||
| 		header.number = rolling_block_number; | ||||
| 		header.parent_hash = rolling_hash; | ||||
| 		header.state_root = state_root.clone(); | ||||
| 
 | ||||
| 		rolling_hash = header.hash(); | ||||
| 		rolling_block_number = rolling_block_number + 1; | ||||
| 		rolling_timestamp = rolling_timestamp + 10; | ||||
| 
 | ||||
| 		if let Err(e) = client.import_block(create_test_block(&header)) { | ||||
| 			panic!("error importing block which is valid by definition: {:?}", e); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| pub fn get_test_client_with_blocks(blocks: Vec<Bytes>) -> GuardedTempResult<Arc<Client>> { | ||||
| 	let dir = RandomTempPath::new(); | ||||
| 	let client = Client::new(ClientConfig::default(), get_test_spec(), dir.as_path(), IoChannel::disconnected()).unwrap(); | ||||
| @ -253,18 +280,29 @@ pub fn get_temp_state_in(path: &Path) -> State { | ||||
| pub fn get_good_dummy_block_seq(count: usize) -> Vec<Bytes> { | ||||
| 	let test_spec = get_test_spec(); | ||||
| 	let test_engine = test_spec.to_engine().unwrap(); | ||||
| 	let mut parent = test_engine.spec().genesis_header().hash(); | ||||
|   	get_good_dummy_block_fork_seq(1, count, &test_engine.spec().genesis_header().hash()) | ||||
| } | ||||
| 
 | ||||
| pub fn get_good_dummy_block_fork_seq(start_number: usize, count: usize, parent_hash: &H256) -> Vec<Bytes> { | ||||
| 	let test_spec = get_test_spec(); | ||||
| 	let test_engine = test_spec.to_engine().unwrap(); | ||||
| 	let mut rolling_timestamp = start_number as u64 * 10; | ||||
| 	let mut parent = *parent_hash; | ||||
| 	let mut r = Vec::new(); | ||||
| 	for i in 1 .. count + 1 { | ||||
| 	for i in start_number .. start_number + count + 1 { | ||||
| 		let mut block_header = Header::new(); | ||||
| 		block_header.gas_limit = decode(test_engine.spec().engine_params.get("minGasLimit").unwrap()); | ||||
| 		block_header.difficulty = decode(test_engine.spec().engine_params.get("minimumDifficulty").unwrap()); | ||||
| 		block_header.timestamp = i as u64; | ||||
| 		block_header.difficulty = U256::from(i).mul(U256([0, 1, 0, 0])); | ||||
| 		block_header.timestamp = rolling_timestamp; | ||||
| 		block_header.number = i as u64; | ||||
| 		block_header.parent_hash = parent; | ||||
| 		block_header.state_root = test_engine.spec().genesis_header().state_root; | ||||
| 
 | ||||
| 		parent = block_header.hash(); | ||||
| 		rolling_timestamp = rolling_timestamp + 10; | ||||
| 
 | ||||
| 		r.push(create_test_block(&block_header)); | ||||
| 
 | ||||
| 	} | ||||
| 	r | ||||
| } | ||||
|  | ||||
| @ -20,6 +20,7 @@ ethash = { path = "../ethash" } | ||||
| ethsync = { path = "../sync" } | ||||
| clippy = { version = "0.0.44", optional = true } | ||||
| rustc-serialize = "0.3" | ||||
| transient-hashmap = "0.1" | ||||
| serde_macros = { version = "0.7.0", optional = true } | ||||
| 
 | ||||
| [build-dependencies] | ||||
|  | ||||
| @ -9,8 +9,8 @@ mod inner { | ||||
|     pub fn main() { | ||||
|         let out_dir = env::var_os("OUT_DIR").unwrap(); | ||||
| 
 | ||||
|         let src = Path::new("src/lib.rs.in"); | ||||
|         let dst = Path::new(&out_dir).join("lib.rs"); | ||||
|         let src = Path::new("src/v1/types/mod.rs.in"); | ||||
|         let dst = Path::new(&out_dir).join("mod.rs"); | ||||
| 
 | ||||
|         let mut registry = syntex::Registry::new(); | ||||
| 
 | ||||
|  | ||||
| @ -27,9 +27,35 @@ extern crate jsonrpc_http_server; | ||||
| extern crate ethcore_util as util; | ||||
| extern crate ethcore; | ||||
| extern crate ethsync; | ||||
| extern crate transient_hashmap; | ||||
| 
 | ||||
| #[cfg(feature = "serde_macros")] | ||||
| include!("lib.rs.in"); | ||||
| use self::jsonrpc_core::{IoHandler, IoDelegate}; | ||||
| 
 | ||||
| #[cfg(not(feature = "serde_macros"))] | ||||
| include!(concat!(env!("OUT_DIR"), "/lib.rs")); | ||||
| pub mod v1; | ||||
| 
 | ||||
| /// Http server.
 | ||||
| pub struct HttpServer { | ||||
| 	handler: IoHandler, | ||||
| 	threads: usize | ||||
| } | ||||
| 
 | ||||
| impl HttpServer { | ||||
| 	/// Construct new http server object with given number of threads.
 | ||||
| 	pub fn new(threads: usize) -> HttpServer { | ||||
| 		HttpServer { | ||||
| 			handler: IoHandler::new(), | ||||
| 			threads: threads | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/// Add io delegate.
 | ||||
| 	pub fn add_delegate<D>(&mut self, delegate: IoDelegate<D>) where D: Send + Sync + 'static { | ||||
| 		self.handler.add_delegate(delegate); | ||||
| 	} | ||||
| 
 | ||||
| 	/// Start server asynchronously in new thread
 | ||||
| 	pub fn start_async(self, addr: &str, cors_domain: &str) { | ||||
| 		let server = jsonrpc_http_server::Server::new(self.handler, self.threads); | ||||
| 		server.start_async(addr, jsonrpc_http_server::AccessControlAllowOrigin::Value(cors_domain.to_owned())) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -1,30 +0,0 @@ | ||||
| use self::jsonrpc_core::{IoHandler, IoDelegate}; | ||||
| 
 | ||||
| pub mod v1; | ||||
| 
 | ||||
| /// Http server.
 | ||||
| pub struct HttpServer { | ||||
| 	handler: IoHandler, | ||||
| 	threads: usize | ||||
| } | ||||
| 
 | ||||
| impl HttpServer { | ||||
| 	/// Construct new http server object with given number of threads.
 | ||||
| 	pub fn new(threads: usize) -> HttpServer { | ||||
| 		HttpServer { | ||||
| 			handler: IoHandler::new(), | ||||
| 			threads: threads | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/// Add io delegate.
 | ||||
| 	pub fn add_delegate<D>(&mut self, delegate: IoDelegate<D>) where D: Send + Sync + 'static { | ||||
| 		self.handler.add_delegate(delegate); | ||||
| 	} | ||||
| 
 | ||||
| 	/// Start server asynchronously in new thread
 | ||||
| 	pub fn start_async(self, addr: &str, cors_domain: &str) { | ||||
| 		let server = jsonrpc_http_server::Server::new(self.handler, self.threads); | ||||
| 		server.start_async(addr, jsonrpc_http_server::AccessControlAllowOrigin::Value(cors_domain.to_owned())) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										21
									
								
								rpc/src/v1/helpers/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								rpc/src/v1/helpers/mod.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | ||||
| // 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/>.
 | ||||
| 
 | ||||
| mod poll_manager; | ||||
| mod poll_filter; | ||||
| 
 | ||||
| pub use self::poll_manager::PollManager; | ||||
| pub use self::poll_filter::PollFilter; | ||||
							
								
								
									
										10
									
								
								rpc/src/v1/helpers/poll_filter.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								rpc/src/v1/helpers/poll_filter.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | ||||
| //! Helper type with all filter possibilities.
 | ||||
| 
 | ||||
| use ethcore::filter::Filter; | ||||
| 
 | ||||
| #[derive(Clone)] | ||||
| pub enum PollFilter { | ||||
| 	Block, | ||||
| 	PendingTransaction, | ||||
| 	Logs(Filter) | ||||
| } | ||||
							
								
								
									
										144
									
								
								rpc/src/v1/helpers/poll_manager.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										144
									
								
								rpc/src/v1/helpers/poll_manager.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,144 @@ | ||||
| // 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/>.
 | ||||
| 
 | ||||
| //! Indexes all rpc poll requests.
 | ||||
| 
 | ||||
| use transient_hashmap::{TransientHashMap, Timer, StandardTimer}; | ||||
| 
 | ||||
| /// Lifetime of poll (in seconds).
 | ||||
| const POLL_LIFETIME: u64 = 60; | ||||
| 
 | ||||
| pub type PollId = usize; | ||||
| pub type BlockNumber = u64; | ||||
| 
 | ||||
| pub struct PollInfo<F> { | ||||
| 	pub filter: F, | ||||
| 	pub block_number: BlockNumber | ||||
| } | ||||
| 
 | ||||
| impl<F> Clone for PollInfo<F> where F: Clone { | ||||
| 	fn clone(&self) -> Self { | ||||
| 		PollInfo { | ||||
| 			filter: self.filter.clone(), | ||||
| 			block_number: self.block_number.clone() | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /// Indexes all poll requests.
 | ||||
| ///
 | ||||
| /// Lazily garbage collects unused polls info.
 | ||||
| pub struct PollManager<F, T = StandardTimer> where T: Timer { | ||||
| 	polls: TransientHashMap<PollId, PollInfo<F>, T>, | ||||
| 	next_available_id: PollId | ||||
| } | ||||
| 
 | ||||
| impl<F> PollManager<F, StandardTimer> { | ||||
| 	/// Creates new instance of indexer.
 | ||||
| 	pub fn new() -> Self { | ||||
| 		PollManager::new_with_timer(Default::default()) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| impl<F, T> PollManager<F, T> where T: Timer { | ||||
| 	pub fn new_with_timer(timer: T) -> Self { | ||||
| 		PollManager { | ||||
| 			polls: TransientHashMap::new_with_timer(POLL_LIFETIME, timer), | ||||
| 			next_available_id: 0, | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/// Returns id which can be used for new poll.
 | ||||
| 	///
 | ||||
| 	/// Stores information when last poll happend.
 | ||||
| 	pub fn create_poll(&mut self, filter: F, block: BlockNumber) -> PollId { | ||||
| 		self.polls.prune(); | ||||
| 		let id = self.next_available_id; | ||||
| 		self.next_available_id += 1; | ||||
| 		self.polls.insert(id, PollInfo { | ||||
| 			filter: filter, | ||||
| 			block_number: block, | ||||
| 		}); | ||||
| 		id | ||||
| 	} | ||||
| 
 | ||||
| 	/// Updates information when last poll happend.
 | ||||
| 	pub fn update_poll(&mut self, id: &PollId, block: BlockNumber) { | ||||
| 		self.polls.prune(); | ||||
| 		if let Some(info) = self.polls.get_mut(id) { | ||||
| 			info.block_number = block; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/// Returns number of block when last poll happend.
 | ||||
| 	pub fn get_poll_info(&mut self, id: &PollId) -> Option<&PollInfo<F>> { | ||||
| 		self.polls.prune(); | ||||
| 		self.polls.get(id) | ||||
| 	} | ||||
| 
 | ||||
| 	/// Removes poll info.
 | ||||
| 	pub fn remove_poll(&mut self, id: &PollId) { | ||||
| 		self.polls.remove(id); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
| 	use std::cell::RefCell; | ||||
| 	use transient_hashmap::Timer; | ||||
| 	use v1::helpers::PollManager; | ||||
| 
 | ||||
| 	struct TestTimer<'a> { | ||||
| 		time: &'a RefCell<i64>, | ||||
| 	} | ||||
| 
 | ||||
| 	impl<'a> Timer for TestTimer<'a> { | ||||
| 		fn get_time(&self) -> i64 { | ||||
| 			*self.time.borrow() | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	#[test] | ||||
| 	fn test_poll_indexer() { | ||||
| 		let time = RefCell::new(0); | ||||
| 		let timer = TestTimer { | ||||
| 			time: &time, | ||||
| 		}; | ||||
| 
 | ||||
| 		let mut indexer = PollManager::new_with_timer(timer); | ||||
| 		assert_eq!(indexer.create_poll(false, 20), 0); | ||||
| 		assert_eq!(indexer.create_poll(true, 20), 1); | ||||
| 
 | ||||
| 		*time.borrow_mut() = 10; | ||||
| 		indexer.update_poll(&0, 21); | ||||
| 		assert_eq!(indexer.get_poll_info(&0).unwrap().filter, false); | ||||
| 		assert_eq!(indexer.get_poll_info(&0).unwrap().block_number, 21); | ||||
| 
 | ||||
| 		*time.borrow_mut() = 30; | ||||
| 		indexer.update_poll(&1, 23); | ||||
| 		assert_eq!(indexer.get_poll_info(&1).unwrap().filter, true); | ||||
| 		assert_eq!(indexer.get_poll_info(&1).unwrap().block_number, 23); | ||||
| 
 | ||||
| 		*time.borrow_mut() = 75; | ||||
| 		indexer.update_poll(&0, 30); | ||||
| 		assert!(indexer.get_poll_info(&0).is_none()); | ||||
| 		assert_eq!(indexer.get_poll_info(&1).unwrap().filter, true); | ||||
| 		assert_eq!(indexer.get_poll_info(&1).unwrap().block_number, 23); | ||||
| 
 | ||||
| 		indexer.remove_poll(&1); | ||||
| 		assert!(indexer.get_poll_info(&1).is_none()); | ||||
| 	} | ||||
| } | ||||
| @ -15,11 +15,12 @@ | ||||
| // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
| 
 | ||||
| //! Eth rpc implementation.
 | ||||
| use std::collections::HashMap; | ||||
| use std::sync::{Arc, Weak, Mutex, RwLock}; | ||||
| use ethsync::{EthSync, SyncState}; | ||||
| use jsonrpc_core::*; | ||||
| use util::numbers::*; | ||||
| use util::sha3::*; | ||||
| use util::standard::{RwLock, HashMap, Arc, Weak}; | ||||
| use util::rlp::encode; | ||||
| use ethcore::client::*; | ||||
| use ethcore::block::{IsBlock}; | ||||
| @ -29,6 +30,7 @@ use ethcore::ethereum::Ethash; | ||||
| use ethcore::ethereum::denominations::shannon; | ||||
| use v1::traits::{Eth, EthFilter}; | ||||
| use v1::types::{Block, BlockTransactions, BlockNumber, Bytes, SyncStatus, SyncInfo, Transaction, OptionalValue, Index, Filter, Log}; | ||||
| use v1::helpers::{PollFilter, PollManager}; | ||||
| 
 | ||||
| /// Eth rpc implementation.
 | ||||
| pub struct EthClient { | ||||
| @ -255,28 +257,98 @@ impl Eth for EthClient { | ||||
| 
 | ||||
| /// Eth filter rpc implementation.
 | ||||
| pub struct EthFilterClient { | ||||
| 	client: Weak<Client> | ||||
| 	client: Weak<Client>, | ||||
| 	polls: Mutex<PollManager<PollFilter>>, | ||||
| } | ||||
| 
 | ||||
| impl EthFilterClient { | ||||
| 	/// Creates new Eth filter client.
 | ||||
| 	pub fn new(client: &Arc<Client>) -> Self { | ||||
| 		EthFilterClient { | ||||
| 			client: Arc::downgrade(client) | ||||
| 			client: Arc::downgrade(client), | ||||
| 			polls: Mutex::new(PollManager::new()) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| impl EthFilter for EthFilterClient { | ||||
| 	fn new_block_filter(&self, _params: Params) -> Result<Value, Error> { | ||||
| 		Ok(Value::U64(0)) | ||||
| 	fn new_filter(&self, params: Params) -> Result<Value, Error> { | ||||
| 		from_params::<(Filter,)>(params) | ||||
| 			.and_then(|(filter,)| { | ||||
| 				let mut polls = self.polls.lock().unwrap(); | ||||
| 				let id = polls.create_poll(PollFilter::Logs(filter.into()), take_weak!(self.client).chain_info().best_block_number); | ||||
| 				to_value(&U256::from(id)) | ||||
| 			}) | ||||
| 	} | ||||
| 
 | ||||
| 	fn new_pending_transaction_filter(&self, _params: Params) -> Result<Value, Error> { | ||||
| 		Ok(Value::U64(1)) | ||||
| 	fn new_block_filter(&self, params: Params) -> Result<Value, Error> { | ||||
| 		match params { | ||||
| 			Params::None => { | ||||
| 				let mut polls = self.polls.lock().unwrap(); | ||||
| 				let id = polls.create_poll(PollFilter::Block, take_weak!(self.client).chain_info().best_block_number); | ||||
| 				to_value(&U256::from(id)) | ||||
| 			}, | ||||
| 			_ => Err(Error::invalid_params()) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	fn filter_changes(&self, _: Params) -> Result<Value, Error> { | ||||
| 		to_value(&take_weak!(self.client).chain_info().best_block_hash).map(|v| Value::Array(vec![v])) | ||||
| 	fn new_pending_transaction_filter(&self, params: Params) -> Result<Value, Error> { | ||||
| 		match params { | ||||
| 			Params::None => { | ||||
| 				let mut polls = self.polls.lock().unwrap(); | ||||
| 				let id = polls.create_poll(PollFilter::PendingTransaction, take_weak!(self.client).chain_info().best_block_number); | ||||
| 				to_value(&U256::from(id)) | ||||
| 			}, | ||||
| 			_ => Err(Error::invalid_params()) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	fn filter_changes(&self, params: Params) -> Result<Value, Error> { | ||||
| 		let client = take_weak!(self.client); | ||||
| 		from_params::<(Index,)>(params) | ||||
| 			.and_then(|(index,)| { | ||||
| 				let info = self.polls.lock().unwrap().get_poll_info(&index.value()).cloned(); | ||||
| 				match info { | ||||
| 					None => Ok(Value::Array(vec![] as Vec<Value>)), | ||||
| 					Some(info) => match info.filter { | ||||
| 						PollFilter::Block => { | ||||
| 							let current_number = client.chain_info().best_block_number; | ||||
| 							let hashes = (info.block_number..current_number).into_iter() | ||||
| 								.map(BlockId::Number) | ||||
| 								.filter_map(|id| client.block_hash(id)) | ||||
| 								.collect::<Vec<H256>>(); | ||||
| 
 | ||||
| 							self.polls.lock().unwrap().update_poll(&index.value(), current_number); | ||||
| 
 | ||||
| 							to_value(&hashes) | ||||
| 						}, | ||||
| 						PollFilter::PendingTransaction => { | ||||
| 							// TODO: fix implementation once TransactionQueue is merged
 | ||||
| 							to_value(&vec![] as &Vec<H256>) | ||||
| 						}, | ||||
| 						PollFilter::Logs(mut filter) => { | ||||
| 							filter.from_block = BlockId::Number(info.block_number); | ||||
| 							filter.to_block = BlockId::Latest; | ||||
| 							let logs = client.logs(filter) | ||||
| 								.into_iter() | ||||
| 								.map(From::from) | ||||
| 								.collect::<Vec<Log>>(); | ||||
| 
 | ||||
| 							let current_number = client.chain_info().best_block_number; | ||||
| 							self.polls.lock().unwrap().update_poll(&index.value(), current_number); | ||||
| 
 | ||||
| 							to_value(&logs) | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			}) | ||||
| 	} | ||||
| 
 | ||||
| 	fn uninstall_filter(&self, params: Params) -> Result<Value, Error> { | ||||
| 		from_params::<(Index,)>(params) | ||||
| 			.and_then(|(index,)| { | ||||
| 				self.polls.lock().unwrap().remove_poll(&index.value()); | ||||
| 				to_value(&true) | ||||
| 			}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -30,9 +30,7 @@ impl Web3Client { | ||||
| impl Web3 for Web3Client { | ||||
| 	fn client_version(&self, params: Params) -> Result<Value, Error> { | ||||
| 		match params { | ||||
| 			Params::None => { | ||||
| 				Ok(Value::String(version().to_owned().replace("Parity/", "Parity//"))), | ||||
| 			} | ||||
| 			Params::None => Ok(Value::String(version().to_owned().replace("Parity/", "Parity//"))), | ||||
| 			_ => Err(Error::invalid_params()) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @ -23,6 +23,7 @@ mod impls; | ||||
| mod types; | ||||
| #[cfg(test)] | ||||
| mod tests; | ||||
| mod helpers; | ||||
| 
 | ||||
| pub use self::traits::{Web3, Eth, EthFilter, Net}; | ||||
| pub use self::impls::*; | ||||
|  | ||||
| @ -14,22 +14,8 @@ | ||||
| // You should have received a copy of the GNU General Public License
 | ||||
| // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
| 
 | ||||
| mod block; | ||||
| mod block_number; | ||||
| mod bytes; | ||||
| mod filter; | ||||
| mod index; | ||||
| mod log; | ||||
| mod optionals; | ||||
| mod sync; | ||||
| mod transaction; | ||||
| #[cfg(feature = "serde_macros")] | ||||
| include!("mod.rs.in"); | ||||
| 
 | ||||
| pub use self::block::{Block, BlockTransactions}; | ||||
| pub use self::block_number::BlockNumber; | ||||
| pub use self::bytes::Bytes; | ||||
| pub use self::filter::Filter; | ||||
| pub use self::index::Index; | ||||
| pub use self::log::Log; | ||||
| pub use self::optionals::OptionalValue; | ||||
| pub use self::sync::{SyncStatus, SyncInfo}; | ||||
| pub use self::transaction::Transaction; | ||||
| #[cfg(not(feature = "serde_macros"))] | ||||
| include!(concat!(env!("OUT_DIR"), "/mod.rs")); | ||||
|  | ||||
							
								
								
									
										35
									
								
								rpc/src/v1/types/mod.rs.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								rpc/src/v1/types/mod.rs.in
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | ||||
| // 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/>.
 | ||||
| 
 | ||||
| mod block; | ||||
| mod block_number; | ||||
| mod bytes; | ||||
| mod filter; | ||||
| mod index; | ||||
| mod log; | ||||
| mod optionals; | ||||
| mod sync; | ||||
| mod transaction; | ||||
| 
 | ||||
| pub use self::block::{Block, BlockTransactions}; | ||||
| pub use self::block_number::BlockNumber; | ||||
| pub use self::bytes::Bytes; | ||||
| pub use self::filter::Filter; | ||||
| pub use self::index::Index; | ||||
| pub use self::log::Log; | ||||
| pub use self::optionals::OptionalValue; | ||||
| pub use self::sync::{SyncStatus, SyncInfo}; | ||||
| pub use self::transaction::Transaction; | ||||
| @ -635,16 +635,7 @@ impl ChainSync { | ||||
| 		match self.last_imported_block { None => 0, Some(x) => x } | ||||
| 	} | ||||
| 
 | ||||
| 	/// Find some headers or blocks to download for a peer.
 | ||||
| 	fn request_blocks(&mut self, io: &mut SyncIo, peer_id: PeerId, ignore_others: bool) { | ||||
| 		self.clear_peer_download(peer_id); | ||||
| 
 | ||||
| 		if io.chain().queue_info().is_full() { | ||||
| 			self.pause_sync(); | ||||
| 			return; | ||||
| 		} | ||||
| 
 | ||||
| 		// check to see if we need to download any block bodies first
 | ||||
| 	fn find_block_bodies_hashes_to_request(&self, ignore_others: bool) -> (Vec<H256>, Vec<BlockNumber>) { | ||||
| 		let mut needed_bodies: Vec<H256> = Vec::new(); | ||||
| 		let mut needed_numbers: Vec<BlockNumber> = Vec::new(); | ||||
| 
 | ||||
| @ -664,74 +655,88 @@ impl ChainSync { | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		(needed_bodies, needed_numbers) | ||||
| 	} | ||||
| 
 | ||||
| 	/// Find some headers or blocks to download for a peer.
 | ||||
| 	fn request_blocks(&mut self, io: &mut SyncIo, peer_id: PeerId, ignore_others: bool) { | ||||
| 		self.clear_peer_download(peer_id); | ||||
| 
 | ||||
| 		if io.chain().queue_info().is_full() { | ||||
| 			self.pause_sync(); | ||||
| 			return; | ||||
| 		} | ||||
| 
 | ||||
| 		// check to see if we need to download any block bodies first
 | ||||
| 		let (needed_bodies, needed_numbers) = self.find_block_bodies_hashes_to_request(ignore_others); | ||||
| 		if !needed_bodies.is_empty() { | ||||
| 			let (head, _) = self.headers.range_iter().next().unwrap(); | ||||
| 			if needed_numbers.first().unwrap() - head > self.max_download_ahead_blocks as BlockNumber { | ||||
| 				trace!(target: "sync", "{}: Stalled download ({} vs {}), helping with downloading block bodies", peer_id, needed_numbers.first().unwrap(), head); | ||||
| 				self.request_blocks(io, peer_id, true); | ||||
| 				return; | ||||
| 			} else { | ||||
| 				self.downloading_bodies.extend(needed_numbers.iter()); | ||||
| 				replace(&mut self.peers.get_mut(&peer_id).unwrap().asking_blocks, needed_numbers); | ||||
| 				self.request_bodies(io, peer_id, needed_bodies); | ||||
| 			} | ||||
| 			return; | ||||
| 		} | ||||
| 
 | ||||
| 		// check if need to download headers
 | ||||
| 		let mut start = 0; | ||||
| 		if !self.have_common_block { | ||||
| 			// download backwards until common block is found 1 header at a time
 | ||||
| 			let chain_info = io.chain().chain_info(); | ||||
| 			start = chain_info.best_block_number; | ||||
| 			if !self.headers.is_empty() { | ||||
| 				start = min(start, self.headers.range_iter().next().unwrap().0 - 1); | ||||
| 			} | ||||
| 			if start == 0 { | ||||
| 				self.have_common_block = true; //reached genesis
 | ||||
| 				self.last_imported_hash = Some(chain_info.genesis_hash); | ||||
| 				self.last_imported_block = Some(0); | ||||
| 			} | ||||
| 		} | ||||
| 		if self.have_common_block { | ||||
| 			let mut headers: Vec<BlockNumber> = Vec::new(); | ||||
| 			let mut prev = self.current_base_block() + 1; | ||||
| 			let head = self.headers.range_iter().next().map(|(h, _)| h); | ||||
| 			for (next, ref items) in self.headers.range_iter() { | ||||
| 				if !headers.is_empty() { | ||||
| 					break; | ||||
| 				} | ||||
| 				if next <= prev { | ||||
| 					prev = next + items.len() as BlockNumber; | ||||
| 					continue; | ||||
| 				} | ||||
| 				let mut block = prev; | ||||
| 				while block < next && headers.len() < MAX_HEADERS_TO_REQUEST { | ||||
| 					if ignore_others || !self.downloading_headers.contains(&(block as BlockNumber)) { | ||||
| 						headers.push(block as BlockNumber); | ||||
| 					} | ||||
| 					block += 1; | ||||
| 				} | ||||
| 				prev = next + items.len() as BlockNumber; | ||||
| 			} | ||||
| 
 | ||||
| 			if !headers.is_empty() { | ||||
| 				start = headers[0]; | ||||
| 				if head.is_some() && start > head.unwrap() && start - head.unwrap() > self.max_download_ahead_blocks as BlockNumber { | ||||
| 					trace!(target: "sync", "{}: Stalled download ({} vs {}), helping with downloading headers", peer_id, start, head.unwrap()); | ||||
| 					self.request_blocks(io, peer_id, true); | ||||
| 					return; | ||||
| 				} | ||||
| 				let count = headers.len(); | ||||
| 				self.downloading_headers.extend(headers.iter()); | ||||
| 				replace(&mut self.peers.get_mut(&peer_id).unwrap().asking_blocks, headers); | ||||
| 				assert!(!self.headers.have_item(&start)); | ||||
| 				self.request_headers_by_number(io, peer_id, start, count, 0, false); | ||||
| 			} | ||||
| 			self.downloading_bodies.extend(needed_numbers.iter()); | ||||
| 			replace(&mut self.peers.get_mut(&peer_id).unwrap().asking_blocks, needed_numbers); | ||||
| 			self.request_bodies(io, peer_id, needed_bodies); | ||||
| 		} | ||||
| 		else { | ||||
| 			// check if need to download headers
 | ||||
| 			let mut start = 0; | ||||
| 			if !self.have_common_block { | ||||
| 				// download backwards until common block is found 1 header at a time
 | ||||
| 				let chain_info = io.chain().chain_info(); | ||||
| 				start = chain_info.best_block_number; | ||||
| 				if !self.headers.is_empty() { | ||||
| 					start = min(start, self.headers.range_iter().next().unwrap().0 - 1); | ||||
| 				} | ||||
| 				if start == 0 { | ||||
| 					self.have_common_block = true; //reached genesis
 | ||||
| 					self.last_imported_hash = Some(chain_info.genesis_hash); | ||||
| 					self.last_imported_block = Some(0); | ||||
| 				} | ||||
| 			} | ||||
| 			if self.have_common_block { | ||||
| 				let mut headers: Vec<BlockNumber> = Vec::new(); | ||||
| 				let mut prev = self.current_base_block() + 1; | ||||
| 				let head = self.headers.range_iter().next().map(|(h, _)| h); | ||||
| 				for (next, ref items) in self.headers.range_iter() { | ||||
| 					if !headers.is_empty() { | ||||
| 						break; | ||||
| 					} | ||||
| 					if next <= prev { | ||||
| 						prev = next + items.len() as BlockNumber; | ||||
| 						continue; | ||||
| 					} | ||||
| 					let mut block = prev; | ||||
| 					while block < next && headers.len() < MAX_HEADERS_TO_REQUEST { | ||||
| 						if ignore_others || !self.downloading_headers.contains(&(block as BlockNumber)) { | ||||
| 							headers.push(block as BlockNumber); | ||||
| 						} | ||||
| 						block += 1; | ||||
| 					} | ||||
| 					prev = next + items.len() as BlockNumber; | ||||
| 				} | ||||
| 
 | ||||
| 				if !headers.is_empty() { | ||||
| 					start = headers[0]; | ||||
| 					if head.is_some() && start > head.unwrap() && start - head.unwrap() > self.max_download_ahead_blocks as BlockNumber { | ||||
| 						trace!(target: "sync", "{}: Stalled download ({} vs {}), helping with downloading headers", peer_id, start, head.unwrap()); | ||||
| 						self.request_blocks(io, peer_id, true); | ||||
| 						return; | ||||
| 					} | ||||
| 					let count = headers.len(); | ||||
| 					self.downloading_headers.extend(headers.iter()); | ||||
| 					replace(&mut self.peers.get_mut(&peer_id).unwrap().asking_blocks, headers); | ||||
| 					assert!(!self.headers.have_item(&start)); | ||||
| 					self.request_headers_by_number(io, peer_id, start, count, 0, false); | ||||
| 				} | ||||
| 			} | ||||
| 			else { | ||||
| 				// continue search for common block
 | ||||
| 				self.downloading_headers.insert(start); | ||||
| 				self.request_headers_by_number(io, peer_id, start, 1, 0, false); | ||||
| 			} | ||||
| 			// continue search for common block
 | ||||
| 			self.downloading_headers.insert(start); | ||||
| 			self.request_headers_by_number(io, peer_id, start, 1, 0, false); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
|  | ||||
| @ -105,6 +105,10 @@ impl BlockChainClient for TestBlockChainClient { | ||||
| 		Some(U256::zero()) | ||||
| 	} | ||||
| 
 | ||||
| 	fn block_hash(&self, id: BlockId) -> Option<H256> { | ||||
| 		unimplemented!(); | ||||
| 	} | ||||
| 
 | ||||
| 	fn code(&self, _address: &Address) -> Option<Bytes> { | ||||
| 		unimplemented!(); | ||||
| 	} | ||||
|  | ||||
| @ -2003,6 +2003,7 @@ mod tests { | ||||
| 
 | ||||
| 
 | ||||
| 	#[test] | ||||
| 	#[cfg_attr(feature = "dev", allow(cyclomatic_complexity))] | ||||
| 	fn u256_multi_full_mul() { | ||||
| 		let result = U256([0, 0, 0, 0]).full_mul(U256([0, 0, 0, 0])); | ||||
| 		assert_eq!(U512([0, 0, 0, 0, 0, 0, 0, 0]), result); | ||||
|  | ||||
| @ -111,7 +111,7 @@ impl<Row, Col, Val> Table<Row, Col, Val> | ||||
| 	///
 | ||||
| 	/// Returns previous value (if any)
 | ||||
| 	pub fn insert(&mut self, row: Row, col: Col, val: Val) -> Option<Val> { | ||||
| 		self.map.entry(row).or_insert_with(|| HashMap::new()).insert(col, val) | ||||
| 		self.map.entry(row).or_insert_with(HashMap::new).insert(col, val) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user