Blockchain repair on missing state root (#1646)
* BC locking; Recovery batch * Missing state root recovery * Test
This commit is contained in:
		
							parent
							
								
									0d2f516ad7
								
							
						
					
					
						commit
						dd17c766b8
					
				| @ -296,32 +296,20 @@ impl BlockChain { | |||||||
| 		// load best block
 | 		// load best block
 | ||||||
| 		let best_block_hash = match bc.extras_db.get(b"best").unwrap() { | 		let best_block_hash = match bc.extras_db.get(b"best").unwrap() { | ||||||
| 			Some(best) => { | 			Some(best) => { | ||||||
| 				let best = H256::from_slice(&best); | 				let mut new_best = H256::from_slice(&best); | ||||||
| 				let mut b = best.clone(); | 				while !bc.blocks_db.get(&new_best).unwrap().is_some() { | ||||||
| 				let mut removed = 0; | 					match bc.rewind() { | ||||||
| 				let mut best_num = 0; | 						Some(h) => { | ||||||
| 				while !bc.blocks_db.get(&b).unwrap().is_some() { | 							new_best = h; | ||||||
| 					// track back to the best block we have in the blocks database
 |  | ||||||
| 					let extras: BlockDetails = bc.extras_db.read(&b).unwrap(); |  | ||||||
| 					type DetailsKey = Key<BlockDetails, Target=H264>; |  | ||||||
| 					bc.extras_db.delete(&(DetailsKey::key(&b))).unwrap(); |  | ||||||
| 					b = extras.parent; |  | ||||||
| 					best_num = extras.number; |  | ||||||
| 					removed += 1; |  | ||||||
| 						} | 						} | ||||||
| 				if b != best { | 						None => { | ||||||
| 					let batch = DBTransaction::new(); | 							warn!("Can't rewind blockchain"); | ||||||
| 					let range = (best_num + 1) as bc::Number .. (best_num + removed) as bc::Number; | 							break; | ||||||
| 					let chain = bc::group::BloomGroupChain::new(bc.blooms_config, &bc); |  | ||||||
| 					let changes = chain.replace(&range, vec![]); |  | ||||||
| 					for (k, v) in changes.into_iter() { |  | ||||||
| 						batch.write(&LogGroupPosition::from(k), &BloomGroup::from(v)); |  | ||||||
| 						} | 						} | ||||||
| 					batch.put(b"best", &b).unwrap(); |  | ||||||
| 					bc.extras_db.write(batch).unwrap(); |  | ||||||
| 					info!("Restored mismatched best block. Was: {}, new: {}", best.hex(), b.hex()); |  | ||||||
| 					} | 					} | ||||||
| 				b | 					info!("Restored mismatched best block. Was: {}, new: {}", H256::from_slice(&best).hex(), new_best.hex()); | ||||||
|  | 				} | ||||||
|  | 				new_best | ||||||
| 			} | 			} | ||||||
| 			None => { | 			None => { | ||||||
| 				// best block does not exist
 | 				// best block does not exist
 | ||||||
| @ -365,6 +353,46 @@ impl BlockChain { | |||||||
| 		self.extras_db.read_with_cache(&self.block_details, parent).map_or(false, |d| d.children.contains(hash)) | 		self.extras_db.read_with_cache(&self.block_details, parent).map_or(false, |d| d.children.contains(hash)) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/// Rewind to a previous block
 | ||||||
|  | 	pub fn rewind(&self) -> Option<H256> { | ||||||
|  | 		let batch = DBTransaction::new(); | ||||||
|  | 		// track back to the best block we have in the blocks database
 | ||||||
|  | 		if let Some(best_block_hash) = self.extras_db.get(b"best").unwrap() { | ||||||
|  | 			let best_block_hash = H256::from_slice(&best_block_hash); | ||||||
|  | 			if best_block_hash == self.genesis_hash() { | ||||||
|  | 				return None; | ||||||
|  | 			} | ||||||
|  | 			if let Some(extras) = self.extras_db.read(&best_block_hash) as Option<BlockDetails> { | ||||||
|  | 				type DetailsKey = Key<BlockDetails, Target=H264>; | ||||||
|  | 				batch.delete(&(DetailsKey::key(&best_block_hash))).unwrap(); | ||||||
|  | 				let hash = extras.parent; | ||||||
|  | 				let range = extras.number as bc::Number .. extras.number as bc::Number; | ||||||
|  | 				let chain = bc::group::BloomGroupChain::new(self.blooms_config, self); | ||||||
|  | 				let changes = chain.replace(&range, vec![]); | ||||||
|  | 				for (k, v) in changes.into_iter() { | ||||||
|  | 					batch.write(&LogGroupPosition::from(k), &BloomGroup::from(v)); | ||||||
|  | 				} | ||||||
|  | 				batch.put(b"best", &hash).unwrap(); | ||||||
|  | 				let mut best_block = self.best_block.write(); | ||||||
|  | 				best_block.number = extras.number - 1; | ||||||
|  | 				best_block.total_difficulty = self.block_details(&hash).unwrap().total_difficulty; | ||||||
|  | 				best_block.hash = hash; | ||||||
|  | 				// update parent extras
 | ||||||
|  | 				if let Some(mut details) = self.extras_db.read(&hash) as Option<BlockDetails> { | ||||||
|  | 					details.children.clear(); | ||||||
|  | 					batch.write(&hash, &details); | ||||||
|  | 				} | ||||||
|  | 				self.extras_db.write(batch).unwrap(); | ||||||
|  | 				self.block_details.write().clear(); | ||||||
|  | 				self.block_hashes.write().clear(); | ||||||
|  | 				self.blocks.write().clear(); | ||||||
|  | 				self.block_receipts.write().clear(); | ||||||
|  | 				return Some(hash); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return None; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/// Set the cache configuration.
 | 	/// Set the cache configuration.
 | ||||||
| 	pub fn configure_cache(&self, pref_cache_size: usize, max_cache_size: usize) { | 	pub fn configure_cache(&self, pref_cache_size: usize, max_cache_size: usize) { | ||||||
| 		self.pref_cache_size.store(pref_cache_size, AtomicOrder::Relaxed); | 		self.pref_cache_size.store(pref_cache_size, AtomicOrder::Relaxed); | ||||||
| @ -514,14 +542,15 @@ impl BlockChain { | |||||||
| 			batch.extend_with_cache(&mut *write_blocks_blooms, update.blocks_blooms, CacheUpdatePolicy::Remove); | 			batch.extend_with_cache(&mut *write_blocks_blooms, update.blocks_blooms, CacheUpdatePolicy::Remove); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		// These cached values must be updated last and togeterh
 | 		// These cached values must be updated last with all three locks taken to avoid
 | ||||||
|  | 		// cache decoherence
 | ||||||
| 		{ | 		{ | ||||||
|  | 			let mut best_block = self.best_block.write(); | ||||||
| 			// update best block
 | 			// update best block
 | ||||||
| 			match update.info.location { | 			match update.info.location { | ||||||
| 				BlockLocation::Branch => (), | 				BlockLocation::Branch => (), | ||||||
| 				_ => { | 				_ => { | ||||||
| 					batch.put(b"best", &update.info.hash).unwrap(); | 					batch.put(b"best", &update.info.hash).unwrap(); | ||||||
| 					let mut best_block = self.best_block.write(); |  | ||||||
| 					*best_block = BestBlock { | 					*best_block = BestBlock { | ||||||
| 						hash: update.info.hash, | 						hash: update.info.hash, | ||||||
| 						number: update.info.number, | 						number: update.info.number, | ||||||
| @ -1222,4 +1251,30 @@ mod tests { | |||||||
| 		let bc = BlockChain::new(Config::default(), &genesis, temp.as_path()); | 		let bc = BlockChain::new(Config::default(), &genesis, temp.as_path()); | ||||||
| 		assert_eq!(bc.best_block_number(), 5); | 		assert_eq!(bc.best_block_number(), 5); | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn test_rewind() { | ||||||
|  | 		let mut canon_chain = ChainGenerator::default(); | ||||||
|  | 		let mut finalizer = BlockFinalizer::default(); | ||||||
|  | 		let genesis = canon_chain.generate(&mut finalizer).unwrap(); | ||||||
|  | 		let first = canon_chain.generate(&mut finalizer).unwrap(); | ||||||
|  | 		let second = canon_chain.generate(&mut finalizer).unwrap(); | ||||||
|  | 		let genesis_hash = BlockView::new(&genesis).header_view().sha3(); | ||||||
|  | 		let first_hash = BlockView::new(&first).header_view().sha3(); | ||||||
|  | 		let second_hash = BlockView::new(&second).header_view().sha3(); | ||||||
|  | 
 | ||||||
|  | 		let temp = RandomTempPath::new(); | ||||||
|  | 		let bc = BlockChain::new(Config::default(), &genesis, temp.as_path()); | ||||||
|  | 
 | ||||||
|  | 		bc.insert_block(&first, vec![]); | ||||||
|  | 		bc.insert_block(&second, vec![]); | ||||||
|  | 
 | ||||||
|  | 		assert_eq!(bc.rewind(), Some(first_hash.clone())); | ||||||
|  | 		assert!(!bc.is_known(&second_hash)); | ||||||
|  | 		assert_eq!(bc.best_block_number(), 1); | ||||||
|  | 		assert_eq!(bc.best_block_hash(), first_hash.clone()); | ||||||
|  | 
 | ||||||
|  | 		assert_eq!(bc.rewind(), Some(genesis_hash.clone())); | ||||||
|  | 		assert_eq!(bc.rewind(), None); | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -205,6 +205,11 @@ impl Client { | |||||||
| 			state_db.commit(0, &spec.genesis_header().hash(), None).expect("Error commiting genesis state to state DB"); | 			state_db.commit(0, &spec.genesis_header().hash(), None).expect("Error commiting genesis state to state DB"); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		while !chain.block_header(&chain.best_block_hash()).map_or(true, |h| state_db.contains(h.state_root())) { | ||||||
|  | 			warn!("State root not found for block #{} ({}), recovering...", chain.best_block_number(), chain.best_block_hash().hex()); | ||||||
|  | 			chain.rewind(); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		let engine = Arc::new(spec.engine); | 		let engine = Arc::new(spec.engine); | ||||||
| 
 | 
 | ||||||
| 		let block_queue = BlockQueue::new(config.queue, engine.clone(), message_channel.clone()); | 		let block_queue = BlockQueue::new(config.queue, engine.clone(), message_channel.clone()); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user