// Copyright 2015-2020 Parity Technologies (UK) Ltd. // This file is part of OpenEthereum. // OpenEthereum 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. // OpenEthereum 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 OpenEthereum. If not, see . //! `JournalDB` interface and implementation. #[macro_use] extern crate log; extern crate ethcore_db; extern crate ethereum_types; extern crate fastmap; extern crate hash_db; extern crate keccak_hasher; extern crate kvdb; extern crate memory_db; extern crate parity_bytes as bytes; extern crate parity_util_mem; extern crate parking_lot; extern crate rlp; #[cfg(test)] extern crate env_logger; #[cfg(test)] extern crate keccak_hash as keccak; #[cfg(test)] extern crate kvdb_memorydb; use std::{fmt, io, str, sync::Arc}; mod archivedb; mod as_hash_db_impls; mod earlymergedb; mod overlayrecentdb; mod refcounteddb; /// Export the journaldb module. mod traits; mod util; pub mod overlaydb; /// Export the `JournalDB` trait. pub use self::traits::JournalDB; /// Export as keyed hash trait pub use self::traits::AsKeyedHashDB; /// Export keyed hash trait pub use self::traits::KeyedHashDB; /// Journal database operating strategy. #[derive(Debug, PartialEq, Clone, Copy)] pub enum Algorithm { /// Keep all keys forever. Archive, /// Ancient and recent history maintained separately; recent history lasts for particular /// number of blocks. /// /// Inserts go into backing database, journal retains knowledge of whether backing DB key is /// ancient or recent. Non-canon inserts get explicitly reverted and removed from backing DB. EarlyMerge, /// Ancient and recent history maintained separately; recent history lasts for particular /// number of blocks. /// /// Inserts go into memory overlay, which is tried for key fetches. Memory overlay gets /// flushed in backing only at end of recent history. OverlayRecent, /// Ancient and recent history maintained separately; recent history lasts for particular /// number of blocks. /// /// References are counted in disk-backed DB. RefCounted, } impl str::FromStr for Algorithm { type Err = String; fn from_str(s: &str) -> Result { match s { "archive" => Ok(Algorithm::Archive), "light" => Ok(Algorithm::EarlyMerge), "fast" => Ok(Algorithm::OverlayRecent), "basic" => Ok(Algorithm::RefCounted), e => Err(format!("Invalid algorithm: {}", e)), } } } impl Algorithm { /// Returns static str describing journal database algorithm. pub fn as_str(&self) -> &'static str { match *self { Algorithm::Archive => "archive", Algorithm::EarlyMerge => "light", Algorithm::OverlayRecent => "fast", Algorithm::RefCounted => "basic", } } /// Returns static str describing journal database algorithm. pub fn as_internal_name_str(&self) -> &'static str { match *self { Algorithm::Archive => "archive", Algorithm::EarlyMerge => "earlymerge", Algorithm::OverlayRecent => "overlayrecent", Algorithm::RefCounted => "refcounted", } } /// Returns true if pruning strategy is stable pub fn is_stable(&self) -> bool { match *self { Algorithm::Archive | Algorithm::OverlayRecent => true, _ => false, } } /// Returns all algorithm types. pub fn all_types() -> Vec { vec![ Algorithm::Archive, Algorithm::EarlyMerge, Algorithm::OverlayRecent, Algorithm::RefCounted, ] } } impl fmt::Display for Algorithm { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "{}", self.as_str()) } } /// Create a new `JournalDB` trait object over a generic key-value database. pub fn new( backing: Arc, algorithm: Algorithm, col: Option, ) -> Box { match algorithm { Algorithm::Archive => Box::new(archivedb::ArchiveDB::new(backing, col)), Algorithm::EarlyMerge => Box::new(earlymergedb::EarlyMergeDB::new(backing, col)), Algorithm::OverlayRecent => Box::new(overlayrecentdb::OverlayRecentDB::new(backing, col)), Algorithm::RefCounted => Box::new(refcounteddb::RefCountedDB::new(backing, col)), } } // all keys must be at least 12 bytes const DB_PREFIX_LEN: usize = ::kvdb::PREFIX_LEN; const LATEST_ERA_KEY: [u8; ::kvdb::PREFIX_LEN] = [b'l', b'a', b's', b't', 0, 0, 0, 0, 0, 0, 0, 0]; fn error_key_already_exists(hash: ðereum_types::H256) -> io::Error { io::Error::new(io::ErrorKind::AlreadyExists, hash.to_string()) } fn error_negatively_reference_hash(hash: ðereum_types::H256) -> io::Error { io::Error::new( io::ErrorKind::Other, format!( "Entry {} removed from database more times than it was added.", hash ), ) } pub fn new_memory_db() -> memory_db::MemoryDB { memory_db::MemoryDB::from_null_node(&rlp::NULL_RLP, rlp::NULL_RLP.as_ref().into()) } #[cfg(test)] mod tests { use super::Algorithm; #[test] fn test_journal_algorithm_parsing() { assert_eq!(Algorithm::Archive, "archive".parse().unwrap()); assert_eq!(Algorithm::EarlyMerge, "light".parse().unwrap()); assert_eq!(Algorithm::OverlayRecent, "fast".parse().unwrap()); assert_eq!(Algorithm::RefCounted, "basic".parse().unwrap()); } #[test] fn test_journal_algorithm_printing() { assert_eq!(Algorithm::Archive.to_string(), "archive".to_owned()); assert_eq!(Algorithm::EarlyMerge.to_string(), "light".to_owned()); assert_eq!(Algorithm::OverlayRecent.to_string(), "fast".to_owned()); assert_eq!(Algorithm::RefCounted.to_string(), "basic".to_owned()); } #[test] fn test_journal_algorithm_is_stable() { assert!(Algorithm::Archive.is_stable()); assert!(Algorithm::OverlayRecent.is_stable()); assert!(!Algorithm::EarlyMerge.is_stable()); assert!(!Algorithm::RefCounted.is_stable()); } #[test] fn test_journal_algorithm_all_types() { // compiling should fail if some cases are not covered let mut archive = 0; let mut earlymerge = 0; let mut overlayrecent = 0; let mut refcounted = 0; for a in &Algorithm::all_types() { match *a { Algorithm::Archive => archive += 1, Algorithm::EarlyMerge => earlymerge += 1, Algorithm::OverlayRecent => overlayrecent += 1, Algorithm::RefCounted => refcounted += 1, } } assert_eq!(archive, 1); assert_eq!(earlymerge, 1); assert_eq!(overlayrecent, 1); assert_eq!(refcounted, 1); } }