// Copyright 2017 Parity Technologies (UK) Ltd. // This file is part of Substrate. // Substrate 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. // Substrate 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 Substrate. If not, see . //! RocksDB-based light client blockchain storage. use std::sync::Arc; use parking_lot::RwLock; use kvdb::{KeyValueDB, DBTransaction}; use client::backend::NewBlockState; use client::blockchain::{BlockStatus, Cache as BlockchainCache, HeaderBackend as BlockchainHeaderBackend, Info as BlockchainInfo}; use client::cht; use client::error::{ErrorKind as ClientErrorKind, Result as ClientResult}; use client::light::blockchain::Storage as LightBlockchainStorage; use codec::{Decode, Encode}; use primitives::{AuthorityId, H256, Blake2Hasher}; use runtime_primitives::generic::BlockId; use runtime_primitives::traits::{Block as BlockT, Header as HeaderT, Zero, One, As, NumberFor}; use cache::DbCache; use utils::{meta_keys, Meta, db_err, number_to_lookup_key, open_database, read_db, read_id, read_meta}; use DatabaseSettings; pub(crate) mod columns { pub const META: Option = ::utils::COLUMN_META; pub const HASH_LOOKUP: Option = Some(1); pub const HEADER: Option = Some(2); pub const AUTHORITIES: Option = Some(3); pub const CHT: Option = Some(4); } /// Keep authorities for last 'AUTHORITIES_ENTRIES_TO_KEEP' blocks. #[allow(unused)] pub(crate) const AUTHORITIES_ENTRIES_TO_KEEP: u64 = cht::SIZE; /// Light blockchain storage. Stores most recent headers + CHTs for older headers. pub struct LightStorage { db: Arc, meta: RwLock::Header as HeaderT>::Number, Block::Hash>>, _cache: DbCache, } #[derive(Clone, PartialEq, Debug)] struct BestAuthorities { /// first block, when this set became actual valid_from: N, /// None means that we do not know the set starting from `valid_from` block authorities: Option>, } impl LightStorage where Block: BlockT, { /// Create new storage with given settings. pub fn new(config: DatabaseSettings) -> ClientResult { let db = open_database(&config, "light")?; Self::from_kvdb(db as Arc<_>) } #[cfg(test)] pub(crate) fn new_test() -> Self { use utils::NUM_COLUMNS; let db = Arc::new(::kvdb_memorydb::create(NUM_COLUMNS)); Self::from_kvdb(db as Arc<_>).expect("failed to create test-db") } fn from_kvdb(db: Arc) -> ClientResult { let cache = DbCache::new( db.clone(), columns::HASH_LOOKUP, columns::HEADER, columns::AUTHORITIES )?; let meta = RwLock::new(read_meta::(&*db, columns::HEADER)?); Ok(LightStorage { db, meta, _cache: cache, }) } #[cfg(test)] pub(crate) fn db(&self) -> &Arc { &self.db } #[cfg(test)] pub(crate) fn cache(&self) -> &DbCache { &self._cache } fn update_meta( &self, hash: Block::Hash, number: <::Header as HeaderT>::Number, is_best: bool, is_finalized: bool, ) { let mut meta = self.meta.write(); if is_best { if number == <::Header as HeaderT>::Number::zero() { meta.genesis_hash = hash; } meta.best_number = number; meta.best_hash = hash; } if is_finalized { if number == <::Header as HeaderT>::Number::zero() { meta.genesis_hash = hash; } meta.finalized_number = number; meta.finalized_hash = hash; } } } impl BlockchainHeaderBackend for LightStorage where Block: BlockT, { fn header(&self, id: BlockId) -> ClientResult> { ::utils::read_header(&*self.db, columns::HASH_LOOKUP, columns::HEADER, id) } fn info(&self) -> ClientResult> { let meta = self.meta.read(); Ok(BlockchainInfo { best_hash: meta.best_hash, best_number: meta.best_number, genesis_hash: meta.genesis_hash, }) } fn status(&self, id: BlockId) -> ClientResult { let exists = match id { BlockId::Hash(_) => read_db( &*self.db, columns::HASH_LOOKUP, columns::HEADER, id )?.is_some(), BlockId::Number(n) => n <= self.meta.read().best_number, }; match exists { true => Ok(BlockStatus::InChain), false => Ok(BlockStatus::Unknown), } } fn number(&self, hash: Block::Hash) -> ClientResult::Header as HeaderT>::Number>> { self.header(BlockId::Hash(hash)).and_then(|key| match key { Some(hdr) => Ok(Some(hdr.number().clone())), None => Ok(None), }) } fn hash(&self, number: <::Header as HeaderT>::Number) -> ClientResult> { read_id::(&*self.db, columns::HASH_LOOKUP, BlockId::Number(number)) } } impl LightStorage where Block::Hash: From { // note that a block is finalized. ensure that best chain contains the finalized // block number first. fn note_finalized(&self, transaction: &mut DBTransaction, header: &Block::Header, hash: Block::Hash) -> ClientResult<()> { const NOTEWORTHY_FINALIZATION_GAP: u64 = 32; // TODO: ensure this doesn't conflict with old finalized block. let meta = self.meta.read(); let f_num = header.number().clone(); let number_u64: u64 = f_num.as_(); transaction.put(columns::META, meta_keys::FINALIZED_BLOCK, hash.as_ref()); let (last_finalized_hash, last_finalized_number) = (meta.finalized_hash.clone(), meta.finalized_number); let finalized_gap = f_num - last_finalized_number; if finalized_gap.as_() >= NOTEWORTHY_FINALIZATION_GAP { info!(target: "db", "Finalizing large run of blocks from {:?} to {:?}", (&last_finalized_hash, last_finalized_number), (&hash, f_num)); } else { debug!(target: "db", "Finalizing blocks from {:?} to {:?}", (&last_finalized_hash, last_finalized_number), (&hash, f_num)); } // build new CHT if required let mut build_cht = |header: &Block::Header| -> ClientResult<()> { if let Some(new_cht_number) = cht::is_build_required(cht::SIZE, *header.number()) { let new_cht_start: NumberFor = cht::start_number(cht::SIZE, new_cht_number); let new_cht_root: Option = cht::compute_root::( cht::SIZE, new_cht_number, (new_cht_start.as_()..) .map(|num| self.hash(As::sa(num)).unwrap_or_default()) ); if let Some(new_cht_root) = new_cht_root { transaction.put(columns::CHT, &number_to_lookup_key(new_cht_start), new_cht_root.as_ref()); let mut prune_block = new_cht_start; let new_cht_end = cht::end_number(cht::SIZE, new_cht_number); trace!(target: "db", "Replacing blocks [{}..{}] with CHT#{}", new_cht_start, new_cht_end, new_cht_number); while prune_block <= new_cht_end { let id = read_id::(&*self.db, columns::HASH_LOOKUP, BlockId::Number(prune_block))?; if let Some(hash) = id { let lookup_key = number_to_lookup_key(prune_block); transaction.delete(columns::HASH_LOOKUP, &lookup_key); transaction.delete(columns::HEADER, hash.as_ref()); } prune_block += <::Header as HeaderT>::Number::one(); } } } Ok(()) }; // attempt to build CHT for all newly finalized blocks. let last_finalized_u64 = last_finalized_number.as_(); for num in (last_finalized_u64..number_u64).map(|x| x + 1) { let num = As::sa(num); if num == f_num { build_cht(header)?; } else { let old_header = match self.header(BlockId::Number(num))? { Some(x) => x, None => panic!("finalizing block {} implies existence of block {}; qed", f_num, num), }; build_cht(&old_header)?; } } Ok(()) } } impl LightBlockchainStorage for LightStorage where Block: BlockT, Block::Hash: From, { fn import_header( &self, header: Block::Header, _authorities: Option>, leaf_state: NewBlockState, ) -> ClientResult<()> { let mut transaction = DBTransaction::new(); let hash = header.hash(); let number = *header.number(); transaction.put(columns::HEADER, hash.as_ref(), &header.encode()); transaction.put(columns::HASH_LOOKUP, &number_to_lookup_key(number), hash.as_ref()); if leaf_state.is_best() { transaction.put(columns::META, meta_keys::BEST_BLOCK, hash.as_ref()); // handle reorg. { let meta = self.meta.read(); if meta.best_hash != Default::default() { let parent_hash = *header.parent_hash(); let tree_route = ::utils::tree_route::( &*self.db, columns::HEADER, meta.best_hash, parent_hash, )?; // update block number to hash lookup entries. for retracted in tree_route.retracted() { if retracted.hash == meta.finalized_hash { // TODO: can we recover here? warn!("Safety failure: reverting finalized block {:?}", (&retracted.number, &retracted.hash)); } transaction.delete( columns::HASH_LOOKUP, &::utils::number_to_lookup_key(retracted.number) ); } for enacted in tree_route.enacted() { let hash: &Block::Hash = &enacted.hash; transaction.put( columns::HASH_LOOKUP, &::utils::number_to_lookup_key(enacted.number), hash.as_ref(), ) } } } // TODO: cache authorities for previous block, accounting for reorgs. } let finalized = match leaf_state { NewBlockState::Final => true, _ => false, }; if finalized { self.note_finalized(&mut transaction, &header, hash)?; } debug!("Light DB Commit {:?} ({})", hash, number); self.db.write(transaction).map_err(db_err)?; self.update_meta(hash, number, leaf_state.is_best(), finalized); Ok(()) } fn cht_root(&self, cht_size: u64, block: <::Header as HeaderT>::Number) -> ClientResult { let no_cht_for_block = || ClientErrorKind::Backend(format!("CHT for block {} not exists", block)).into(); let cht_number = cht::block_to_cht_number(cht_size, block).ok_or_else(no_cht_for_block)?; let cht_start = cht::start_number(cht_size, cht_number); self.db.get(columns::CHT, &number_to_lookup_key(cht_start)).map_err(db_err)? .ok_or_else(no_cht_for_block) .and_then(|hash| Block::Hash::decode(&mut &*hash).ok_or_else(no_cht_for_block)) } fn finalize_header(&self, id: BlockId) -> ClientResult<()> { if let Some(header) = self.header(id)? { let mut transaction = DBTransaction::new(); // TODO: ensure best chain contains this block. let hash = header.hash(); self.note_finalized(&mut transaction, &header, hash.clone())?; self.db.write(transaction).map_err(db_err)?; self.update_meta(hash, header.number().clone(), false, true); Ok(()) } else { Err(ClientErrorKind::UnknownBlock(format!("Cannot finalize block {:?}", id)).into()) } } fn last_finalized(&self) -> ClientResult { Ok(self.meta.read().finalized_hash.clone()) } fn cache(&self) -> Option<&BlockchainCache> { None } } #[cfg(test)] pub(crate) mod tests { use client::cht; use runtime_primitives::testing::{H256 as Hash, Header, Block as RawBlock}; use super::*; type Block = RawBlock; pub fn insert_block( db: &LightStorage, parent: &Hash, number: u64, authorities: Option> ) -> Hash { let header = Header { number: number.into(), parent_hash: *parent, state_root: Default::default(), digest: Default::default(), extrinsics_root: Default::default(), }; let hash = header.hash(); db.import_header(header, authorities, NewBlockState::Best).unwrap(); hash } #[test] fn returns_known_header() { let db = LightStorage::new_test(); let known_hash = insert_block(&db, &Default::default(), 0, None); let header_by_hash = db.header(BlockId::Hash(known_hash)).unwrap().unwrap(); let header_by_number = db.header(BlockId::Number(0)).unwrap().unwrap(); assert_eq!(header_by_hash, header_by_number); } #[test] fn does_not_return_unknown_header() { let db = LightStorage::::new_test(); assert!(db.header(BlockId::Hash(1.into())).unwrap().is_none()); assert!(db.header(BlockId::Number(0)).unwrap().is_none()); } #[test] fn returns_info() { let db = LightStorage::new_test(); let genesis_hash = insert_block(&db, &Default::default(), 0, None); let info = db.info().unwrap(); assert_eq!(info.best_hash, genesis_hash); assert_eq!(info.best_number, 0); assert_eq!(info.genesis_hash, genesis_hash); let best_hash = insert_block(&db, &genesis_hash, 1, None); let info = db.info().unwrap(); assert_eq!(info.best_hash, best_hash); assert_eq!(info.best_number, 1); assert_eq!(info.genesis_hash, genesis_hash); } #[test] fn returns_block_status() { let db = LightStorage::new_test(); let genesis_hash = insert_block(&db, &Default::default(), 0, None); assert_eq!(db.status(BlockId::Hash(genesis_hash)).unwrap(), BlockStatus::InChain); assert_eq!(db.status(BlockId::Number(0)).unwrap(), BlockStatus::InChain); assert_eq!(db.status(BlockId::Hash(1.into())).unwrap(), BlockStatus::Unknown); assert_eq!(db.status(BlockId::Number(1)).unwrap(), BlockStatus::Unknown); } #[test] fn returns_block_hash() { let db = LightStorage::new_test(); let genesis_hash = insert_block(&db, &Default::default(), 0, None); assert_eq!(db.hash(0).unwrap(), Some(genesis_hash)); assert_eq!(db.hash(1).unwrap(), None); } #[test] fn import_header_works() { let db = LightStorage::new_test(); let genesis_hash = insert_block(&db, &Default::default(), 0, None); assert_eq!(db.db.iter(columns::HEADER).count(), 1); assert_eq!(db.db.iter(columns::HASH_LOOKUP).count(), 1); let _ = insert_block(&db, &genesis_hash, 1, None); assert_eq!(db.db.iter(columns::HEADER).count(), 2); assert_eq!(db.db.iter(columns::HASH_LOOKUP).count(), 2); } #[test] fn finalized_ancient_headers_are_replaced_with_cht() { let db = LightStorage::new_test(); // insert genesis block header (never pruned) let mut prev_hash = insert_block(&db, &Default::default(), 0, None); // insert SIZE blocks && ensure that nothing is pruned for number in 0..cht::SIZE { prev_hash = insert_block(&db, &prev_hash, 1 + number, None); } assert_eq!(db.db.iter(columns::HEADER).count(), (1 + cht::SIZE) as usize); assert_eq!(db.db.iter(columns::CHT).count(), 0); // insert next SIZE blocks && ensure that nothing is pruned for number in 0..cht::SIZE { prev_hash = insert_block(&db, &prev_hash, 1 + cht::SIZE + number, None); } assert_eq!(db.db.iter(columns::HEADER).count(), (1 + cht::SIZE + cht::SIZE) as usize); assert_eq!(db.db.iter(columns::CHT).count(), 0); // insert block #{2 * cht::SIZE + 1} && check that new CHT is created + headers of this CHT are pruned // nothing is yet finalized, so nothing is pruned. prev_hash = insert_block(&db, &prev_hash, 1 + cht::SIZE + cht::SIZE, None); assert_eq!(db.db.iter(columns::HEADER).count(), (2 + cht::SIZE + cht::SIZE) as usize); assert_eq!(db.db.iter(columns::CHT).count(), 0); // now finalize the block. db.finalize_header(BlockId::Hash(prev_hash)).unwrap(); assert_eq!(db.db.iter(columns::HEADER).count(), (1 + cht::SIZE + 1) as usize); assert_eq!(db.db.iter(columns::CHT).count(), 1); assert!((0..cht::SIZE).all(|i| db.db.get(columns::HEADER, &number_to_lookup_key(1 + i)).unwrap().is_none())); } #[test] fn get_cht_fails_for_genesis_block() { assert!(LightStorage::::new_test().cht_root(cht::SIZE, 0).is_err()); } #[test] fn get_cht_fails_for_non_existant_cht() { assert!(LightStorage::::new_test().cht_root(cht::SIZE, (cht::SIZE / 2) as u64).is_err()); } #[test] fn get_cht_works() { let db = LightStorage::new_test(); // insert 1 + SIZE + SIZE + 1 blocks so that CHT#0 is created let mut prev_hash = Default::default(); for i in 0..1 + cht::SIZE + cht::SIZE + 1 { prev_hash = insert_block(&db, &prev_hash, i as u64, None); } db.finalize_header(BlockId::Hash(prev_hash)).unwrap(); let cht_root_1 = db.cht_root(cht::SIZE, cht::start_number(cht::SIZE, 0)).unwrap(); let cht_root_2 = db.cht_root(cht::SIZE, (cht::start_number(cht::SIZE, 0) + cht::SIZE / 2) as u64).unwrap(); let cht_root_3 = db.cht_root(cht::SIZE, cht::end_number(cht::SIZE, 0)).unwrap(); assert_eq!(cht_root_1, cht_root_2); assert_eq!(cht_root_2, cht_root_3); } }