client.rs 10.8 KiB
Newer Older
Gav Wood's avatar
Gav Wood committed
use rocksdb::{Options, DB};
Gav Wood's avatar
Gav Wood committed
use blockchain::{BlockChain, BlockProvider, CacheSize};
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
use views::BlockView;
use error::*;
use spec::Spec;
use engine::Engine;
use queue::BlockQueue;
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
use sync::NetSyncMessage;
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
use env_info::LastHashes;
use verification::*;
use block::*;
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed

/// General block status
#[derive(Debug)]
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
pub enum BlockStatus {
	/// Part of the blockchain.
	InChain,
	/// Queued for import.
	Queued,
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	/// Known as bad.
	Bad,
	/// Unknown.
	Unknown,
}

/// Information about the blockchain gthered together.
#[derive(Debug)]
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
pub struct BlockChainInfo {
	/// Blockchain difficulty.
	pub total_difficulty: U256,
	/// Block queue difficulty.
	pub pending_total_difficulty: U256,
	/// Genesis block hash.
	pub genesis_hash: H256,
	/// Best blockchain block hash.
	pub best_block_hash: H256,
	/// Best blockchain block number.
	pub best_block_number: BlockNumber
}

Gav Wood's avatar
Gav Wood committed
impl fmt::Display for BlockChainInfo {
	fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
		write!(f, "#{}.{}", self.best_block_number, self.best_block_hash)
	}
}

Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
/// Block queue status
#[derive(Debug)]
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
pub struct BlockQueueStatus {
	/// TODO [arkpar] Please document me
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	pub full: bool,
}

/// TODO [arkpar] Please document me
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
pub type TreeRoute = ::blockchain::TreeRoute;

/// Blockchain database client. Owns and manages a blockchain and a block queue.
pub trait BlockChainClient : Sync + Send {
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	/// Get raw block header data by block header hash.
	fn block_header(&self, hash: &H256) -> Option<Bytes>;

	/// Get raw block body data by block header hash.
	/// Block body is an RLP list of two items: uncles and transactions.
	fn block_body(&self, hash: &H256) -> Option<Bytes>;

	/// Get raw block data by block header hash.
	fn block(&self, hash: &H256) -> Option<Bytes>;

	/// Get block status by block header hash.
	fn block_status(&self, hash: &H256) -> BlockStatus;

	/// Get raw block header data by block number.
	fn block_header_at(&self, n: BlockNumber) -> Option<Bytes>;

	/// Get raw block body data by block number.
	/// Block body is an RLP list of two items: uncles and transactions.
	fn block_body_at(&self, n: BlockNumber) -> Option<Bytes>;

	/// Get raw block data by block number.
	fn block_at(&self, n: BlockNumber) -> Option<Bytes>;

	/// Get block status by block number.
	fn block_status_at(&self, n: BlockNumber) -> BlockStatus;

	/// Get a tree route between `from` and `to`.
	/// See `BlockChain::tree_route`.
	fn tree_route(&self, from: &H256, to: &H256) -> Option<TreeRoute>;
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed

	/// Get latest state node
	fn state_data(&self, hash: &H256) -> Option<Bytes>;

	/// Get raw block receipts data by block header hash.
	fn block_receipts(&self, hash: &H256) -> Option<Bytes>;

	/// Import a block into the blockchain.
	fn import_block(&mut self, bytes: Bytes) -> ImportResult;
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed

	/// Get block queue information.
	fn queue_status(&self) -> BlockQueueStatus;

	/// Clear block queue and abort all import activity.
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	fn clear_queue(&mut self);

	/// Get blockchain information.
	fn chain_info(&self) -> BlockChainInfo;
}

Gav Wood's avatar
Gav Wood committed
#[derive(Default, Clone, Debug, Eq, PartialEq)]
/// TODO [Gav Wood] Please document me
Gav Wood's avatar
Gav Wood committed
pub struct ClientReport {
	/// TODO [Gav Wood] Please document me
Gav Wood's avatar
Gav Wood committed
	pub blocks_imported: usize,
	/// TODO [Gav Wood] Please document me
Gav Wood's avatar
Gav Wood committed
	pub transactions_applied: usize,
	/// TODO [Gav Wood] Please document me
Gav Wood's avatar
Gav Wood committed
	pub gas_processed: U256,
}

impl ClientReport {
	/// TODO [Gav Wood] Please document me
Gav Wood's avatar
Gav Wood committed
	pub fn accrue_block(&mut self, block: &PreVerifiedBlock) {
		self.blocks_imported += 1;
		self.transactions_applied += block.transactions.len();
		self.gas_processed += block.header.gas_used;
	}
}

Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
/// Blockchain database client backed by a persistent database. Owns and manages a blockchain and a block queue.
pub struct Client {
	chain: Arc<RwLock<BlockChain>>,
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	engine: Arc<Box<Engine>>,
	state_db: JournalDB,
	queue: BlockQueue,
Gav Wood's avatar
Gav Wood committed
	report: ClientReport,
const HISTORY: u64 = 1000;

Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
impl Client {
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	/// Create a new client with given spec and DB path.
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	pub fn new(spec: Spec, path: &Path, message_channel: IoChannel<NetSyncMessage> ) -> Result<Client, Error> {
		let chain = Arc::new(RwLock::new(BlockChain::new(&spec.genesis_block(), path)));
Gav Wood's avatar
Gav Wood committed
		let mut opts = Options::new();
		opts.set_max_open_files(256);
Gav Wood's avatar
Gav Wood committed
		opts.create_if_missing(true);
Gav Wood's avatar
Gav Wood committed
		/*opts.set_use_fsync(false);
Gav Wood's avatar
Gav Wood committed
		opts.set_bytes_per_sync(8388608);
		opts.set_disable_data_sync(false);
		opts.set_block_cache_size_mb(1024);
		opts.set_table_cache_num_shard_bits(6);
		opts.set_max_write_buffer_number(32);
		opts.set_write_buffer_size(536870912);
		opts.set_target_file_size_base(1073741824);
		opts.set_min_write_buffer_number_to_merge(4);
		opts.set_level_zero_stop_writes_trigger(2000);
		opts.set_level_zero_slowdown_writes_trigger(0);
		opts.set_compaction_style(DBUniversalCompaction);
		opts.set_max_background_compactions(4);
		opts.set_max_background_flushes(4);
		opts.set_filter_deletes(false);
Gav Wood's avatar
Gav Wood committed
		opts.set_disable_auto_compactions(false);*/
Gav Wood's avatar
Gav Wood committed

Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
		let mut state_path = path.to_path_buf();
		state_path.push("state");
Gav Wood's avatar
Gav Wood committed
		let db = DB::open(&opts, state_path.to_str().unwrap()).unwrap();
		let mut state_db = JournalDB::new(db);
Gav Wood's avatar
Gav Wood committed
		
		let engine = Arc::new(try!(spec.to_engine()));
		if engine.spec().ensure_db_good(&mut state_db) {
			state_db.commit(0, &engine.spec().genesis_header().hash(), None).expect("Error commiting genesis state to state DB");
		}
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed

Gav Wood's avatar
Gav Wood committed
//		chain.write().unwrap().ensure_good(&state_db);
Gav Wood's avatar
Gav Wood committed

Gav Wood's avatar
Gav Wood committed
			chain: chain,
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
			engine: engine.clone(),
			state_db: state_db,
Gav Wood's avatar
Gav Wood committed
			queue: BlockQueue::new(engine, message_channel),
Gav Wood's avatar
Gav Wood committed
			report: Default::default(),
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed

Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	/// This is triggered by a message coming from a block queue when the block is ready for insertion
	pub fn import_verified_blocks(&mut self) {
		
		let mut bad = HashSet::new();
		let blocks = self.queue.drain(128); 
		if blocks.is_empty() {
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
			return;
		}

		for block in blocks {
			if bad.contains(&block.header.parent_hash) {
				self.queue.mark_as_bad(&block.header.hash());
				bad.insert(block.header.hash());
				continue;
			}

			let header = &block.header;
			if let Err(e) = verify_block_family(&header, &block.bytes, self.engine.deref().deref(), self.chain.read().unwrap().deref()) {
				warn!(target: "client", "Stage 3 block verification failed for #{} ({})\nError: {:?}", header.number(), header.hash(), e);
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
				self.queue.mark_as_bad(&header.hash());
				bad.insert(block.header.hash());
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
				return;
			};
			let parent = match self.chain.read().unwrap().block_header(&header.parent_hash) {
				Some(p) => p,
				None => {
					warn!(target: "client", "Block import failed for #{} ({}): Parent not found ({}) ", header.number(), header.hash(), header.parent_hash);
					self.queue.mark_as_bad(&header.hash());
					bad.insert(block.header.hash());
					return;
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
				},
			};
			// build last hashes
			let mut last_hashes = LastHashes::new();
			last_hashes.resize(256, H256::new());
			last_hashes[0] = header.parent_hash.clone();
			for i in 0..255 {
				match self.chain.read().unwrap().block_details(&last_hashes[i]) {
					Some(details) => {
						last_hashes[i + 1] = details.parent.clone();
					},
					None => break,
				}
			let result = match enact_verified(&block, self.engine.deref().deref(), self.state_db.clone(), &parent, &last_hashes) {
				Ok(b) => b,
				Err(e) => {
					warn!(target: "client", "Block import failed for #{} ({})\nError: {:?}", header.number(), header.hash(), e);
				bad.insert(block.header.hash());
					self.queue.mark_as_bad(&header.hash());
					return;
				}
			};
			if let Err(e) = verify_block_final(&header, result.block().header()) {
				warn!(target: "client", "Stage 4 block verification failed for #{} ({})\nError: {:?}", header.number(), header.hash(), e);
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
				self.queue.mark_as_bad(&header.hash());
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed

			self.chain.write().unwrap().insert_block(&block.bytes); //TODO: err here?
Gav Wood's avatar
Gav Wood committed
			let ancient = if header.number() >= HISTORY { Some(header.number() - HISTORY) } else { None };
			match result.drain().commit(header.number(), &header.hash(), ancient.map(|n|(n, self.chain.read().unwrap().block_hash(n).unwrap()))) {
				Ok(_) => (),
				Err(e) => {
					warn!(target: "client", "State DB commit failed: {:?}", e);
					return;
				}
Gav Wood's avatar
Gav Wood committed
			self.report.accrue_block(&block);
			trace!(target: "client", "Imported #{} ({})", header.number(), header.hash());
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
		}
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}
Gav Wood's avatar
Gav Wood committed

	/// Get info on the cache.
	pub fn cache_info(&self) -> CacheSize {
		self.chain.read().unwrap().cache_size()
	}

Gav Wood's avatar
Gav Wood committed
	/// Get the report.
	pub fn report(&self) -> ClientReport {
		self.report.clone()
	}

Gav Wood's avatar
Gav Wood committed
	/// Tick the client.
	pub fn tick(&self) {
		self.chain.read().unwrap().collect_garbage(false);
	}
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
}

impl BlockChainClient for Client {
	fn block_header(&self, hash: &H256) -> Option<Bytes> {
		self.chain.read().unwrap().block(hash).map(|bytes| BlockView::new(&bytes).rlp().at(0).as_raw().to_vec())
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn block_body(&self, hash: &H256) -> Option<Bytes> {
		self.chain.read().unwrap().block(hash).map(|bytes| {
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
			let rlp = Rlp::new(&bytes);
			let mut body = RlpStream::new();
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
			body.append_raw(rlp.at(1).as_raw(), 1);
			body.append_raw(rlp.at(2).as_raw(), 1);
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
			body.out()
		})
	}

	fn block(&self, hash: &H256) -> Option<Bytes> {
		self.chain.read().unwrap().block(hash)
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn block_status(&self, hash: &H256) -> BlockStatus {
		if self.chain.read().unwrap().is_known(&hash) { BlockStatus::InChain } else { BlockStatus::Unknown }
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn block_header_at(&self, n: BlockNumber) -> Option<Bytes> {
		self.chain.read().unwrap().block_hash(n).and_then(|h| self.block_header(&h))
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn block_body_at(&self, n: BlockNumber) -> Option<Bytes> {
		self.chain.read().unwrap().block_hash(n).and_then(|h| self.block_body(&h))
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn block_at(&self, n: BlockNumber) -> Option<Bytes> {
		self.chain.read().unwrap().block_hash(n).and_then(|h| self.block(&h))
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn block_status_at(&self, n: BlockNumber) -> BlockStatus {
		match self.chain.read().unwrap().block_hash(n) {
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
			Some(h) => self.block_status(&h),
			None => BlockStatus::Unknown
		}
	}

	fn tree_route(&self, from: &H256, to: &H256) -> Option<TreeRoute> {
		self.chain.read().unwrap().tree_route(from.clone(), to.clone())
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn state_data(&self, _hash: &H256) -> Option<Bytes> {
		unimplemented!();
	}

	fn block_receipts(&self, _hash: &H256) -> Option<Bytes> {
		unimplemented!();
	}

	fn import_block(&mut self, bytes: Bytes) -> ImportResult {
		let header = BlockView::new(&bytes).header();
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
		if self.chain.read().unwrap().is_known(&header.hash()) {
			return Err(ImportError::AlreadyInChain);
		}
		self.queue.import_block(bytes)
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
	}

	fn queue_status(&self) -> BlockQueueStatus {
		BlockQueueStatus {
			full: false
		}
	}

	fn clear_queue(&mut self) {
	}

	fn chain_info(&self) -> BlockChainInfo {
		let chain = self.chain.read().unwrap();
Arkadiy Paronyan's avatar
Arkadiy Paronyan committed
		BlockChainInfo {
			total_difficulty: chain.best_block_total_difficulty(),
			pending_total_difficulty: chain.best_block_total_difficulty(),
			genesis_hash: chain.genesis_hash(),
			best_block_hash: chain.best_block_hash(),
			best_block_number: From::from(chain.best_block_number())