diff --git a/client/rpc/src/eth/cache/mod.rs b/client/rpc/src/eth/cache/mod.rs index 40756a8ffd..6aa019e036 100644 --- a/client/rpc/src/eth/cache/mod.rs +++ b/client/rpc/src/eth/cache/mod.rs @@ -17,7 +17,6 @@ // along with this program. If not, see . mod lru_cache; -mod tests; use std::{ collections::{BTreeMap, HashMap}, @@ -40,7 +39,7 @@ use sp_api::ProvideRuntimeApi; use sp_blockchain::HeaderBackend; use sp_runtime::{ generic::BlockId, - traits::{BlakeTwo256, Block as BlockT, Header as HeaderT, UniqueSaturatedInto, Zero}, + traits::{BlakeTwo256, Block as BlockT, Header as HeaderT, UniqueSaturatedInto}, }; use fc_rpc_core::types::*; @@ -288,120 +287,6 @@ where BE: Backend + 'static, BE::State: StateBackend, { - /// Task that caches at which substrate hash a new EthereumStorageSchema was inserted in the Runtime Storage. - pub async fn ethereum_schema_cache_task(client: Arc, backend: Arc>) { - if let Ok(None) = frontier_backend_client::load_cached_schema::(backend.as_ref()) { - // Initialize the schema cache at genesis. - let mut cache: Vec<(EthereumStorageSchema, H256)> = Vec::new(); - let id = BlockId::Number(Zero::zero()); - if let Ok(Some(header)) = client.header(id) { - let genesis_schema_version = frontier_backend_client::onchain_storage_schema::< - B, - C, - BE, - >(client.as_ref(), id); - cache.push((genesis_schema_version, header.hash())); - let _ = frontier_backend_client::write_cached_schema::(backend.as_ref(), cache) - .map_err(|err| { - log::warn!("Error schema cache insert for genesis: {:?}", err); - }); - } else { - log::warn!("Error genesis header unreachable"); - } - } - - // Returns the schema for the given block hash and its parent. - let current_and_parent_schema = - |hash: B::Hash| -> Option<(EthereumStorageSchema, EthereumStorageSchema)> { - let id = BlockId::Hash(hash); - if let Ok(Some(header)) = client.header(id) { - let new_schema = frontier_backend_client::onchain_storage_schema::( - client.as_ref(), - id, - ); - - let parent_hash = header.parent_hash(); - let parent_id: BlockId = BlockId::Hash(*parent_hash); - let parent_schema = frontier_backend_client::onchain_storage_schema::( - client.as_ref(), - parent_id, - ); - return Some((new_schema, parent_schema)); - } - None - }; - - let mut notification_st = client.import_notification_stream(); - while let Some(notification) = notification_st.next().await { - let imported_hash = notification.hash; - if let (Some((new_schema, parent_schema)), Ok(Some(old_cache))) = ( - current_and_parent_schema(imported_hash), - frontier_backend_client::load_cached_schema::(backend.as_ref()), - ) { - let mut new_cache: Vec<(EthereumStorageSchema, H256)> = old_cache.clone(); - - if new_schema != parent_schema && notification.is_new_best { - // Always update cache on best block if there is a schema change. - new_cache.push((new_schema, imported_hash)); - } - - // Re-org handling. - if let Some(tree_route) = notification.tree_route { - // Imported block belongs to a re-org. - // First remove the retracted hashes from cache, if any. - let retracted = tree_route - .retracted() - .iter() - .map(|hash_and_number| hash_and_number.hash) - .collect::>(); - let to_remove = old_cache - .iter() - .enumerate() - .filter_map(|(index, (_, hash))| { - if retracted.contains(hash) { - Some(index) - } else { - None - } - }) - .collect::>(); - for index in to_remove { - new_cache.remove(index); - } - // Next add if there is a schema change in the branch. - let to_add = tree_route - .enacted() - .iter() - .filter_map(|hash_and_number| { - if let Some((new_schema, parent_schema)) = - current_and_parent_schema(hash_and_number.hash) - { - if new_schema != parent_schema { - return Some((new_schema, hash_and_number.hash)); - } - return None; - } - None - }) - .collect::>(); - for item in to_add { - new_cache.push(item); - } - } - // Write cache. - if new_cache != old_cache { - let _ = frontier_backend_client::write_cached_schema::( - backend.as_ref(), - new_cache, - ) - .map_err(|err| { - log::warn!("Error schema cache insert: {:?}", err); - }); - } - } - } - } - pub async fn filter_pool_task( client: Arc, filter_pool: Arc>>, diff --git a/client/rpc/src/eth/cache/tests.rs b/client/rpc/src/eth/cache/tests.rs deleted file mode 100644 index 892ec42925..0000000000 --- a/client/rpc/src/eth/cache/tests.rs +++ /dev/null @@ -1,273 +0,0 @@ -// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0 -// This file is part of Frontier. -// -// Copyright (c) 2020 Parity Technologies (UK) Ltd. -// -// This program 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. -// -// This program 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 this program. If not, see . - -#[cfg(test)] -mod tests { - use crate::{frontier_backend_client, EthTask}; - - use codec::Encode; - use std::{path::PathBuf, sync::Arc, thread, time}; - - use fp_storage::{EthereumStorageSchema, PALLET_ETHEREUM_SCHEMA}; - use frontier_template_runtime::RuntimeApi; - use futures::executor; - use sc_block_builder::BlockBuilderProvider; - use sp_consensus::BlockOrigin; - use sp_core::traits::SpawnEssentialNamed; - use sp_runtime::{ - generic::{Block, BlockId, Header}, - traits::BlakeTwo256, - }; - use substrate_test_runtime_client::{ - prelude::*, DefaultTestClientBuilderExt, TestClientBuilder, TestClientBuilderExt, - }; - use tempfile::tempdir; - - type OpaqueBlock = - Block, substrate_test_runtime_client::runtime::Extrinsic>; - - pub fn open_frontier_backend( - path: PathBuf, - ) -> Result>, String> { - Ok(Arc::new(fc_db::Backend::::new( - &fc_db::DatabaseSettings { - source: sc_client_db::DatabaseSource::RocksDb { - path, - cache_size: 0, - }, - }, - )?)) - } - - #[test] - fn should_cache_pallet_ethereum_schema() { - let tmp = tempdir().expect("create a temporary directory"); - // Initialize storage with schema V1. - let builder = TestClientBuilder::new().add_extra_storage( - PALLET_ETHEREUM_SCHEMA.to_vec(), - Encode::encode(&EthereumStorageSchema::V1), - ); - let (client, _) = builder.build_with_native_executor::(None); - let mut client = Arc::new(client); - - // Create a temporary frontier secondary DB. - let frontier_backend = open_frontier_backend(tmp.into_path()).unwrap(); - - // Spawn `frontier-schema-cache-task` background task. - let spawner = sp_core::testing::TaskExecutor::new(); - spawner.spawn_essential_blocking( - "frontier-schema-cache-task", - None, - Box::pin(EthTask::ethereum_schema_cache_task( - Arc::clone(&client), - Arc::clone(&frontier_backend), - )), - ); - - // Create some blocks. - for nonce in [1, 2, 3, 4, 5].into_iter() { - let mut builder = client.new_block(Default::default()).unwrap(); - builder.push_storage_change(vec![nonce], None).unwrap(); - let block = builder.build().unwrap().block; - executor::block_on(client.import(BlockOrigin::Own, block)).unwrap(); - } - - // Expect: only genesis block is cached to schema V1. - assert_eq!( - frontier_backend_client::load_cached_schema::<_>(frontier_backend.as_ref()).unwrap(), - Some(vec![(EthereumStorageSchema::V1, client.genesis_hash())]) - ); - - // Create another block and push a schema change (V2). - let mut builder = client.new_block(Default::default()).unwrap(); - builder - .push_storage_change( - PALLET_ETHEREUM_SCHEMA.to_vec(), - Some(Encode::encode(&EthereumStorageSchema::V2)), - ) - .unwrap(); - let block = builder.build().unwrap().block; - let block_hash = block.header.hash(); - executor::block_on(client.import(BlockOrigin::Own, block)).unwrap(); - - // Give some time to consume and process the import notification stream. - thread::sleep(time::Duration::from_millis(10)); - - // Expect: genesis still cached (V1), latest block cached (V2) - assert_eq!( - frontier_backend_client::load_cached_schema::<_>(frontier_backend.as_ref()).unwrap(), - Some(vec![ - (EthereumStorageSchema::V1, client.genesis_hash()), - (EthereumStorageSchema::V2, block_hash) - ]) - ); - } - - #[test] - fn should_handle_cache_on_multiple_forks() { - let tmp = tempdir().expect("create a temporary directory"); - // Initialize storage with schema V1. - let builder = TestClientBuilder::new().add_extra_storage( - PALLET_ETHEREUM_SCHEMA.to_vec(), - Encode::encode(&EthereumStorageSchema::V1), - ); - let (client, _) = builder.build_with_native_executor::(None); - let mut client = Arc::new(client); - - // Create a temporary frontier secondary DB. - let frontier_backend = open_frontier_backend(tmp.into_path()).unwrap(); - - // Spawn `frontier-schema-cache-task` background task. - let spawner = sp_core::testing::TaskExecutor::new(); - spawner.spawn_essential_blocking( - "frontier-schema-cache-task", - None, - Box::pin(EthTask::ethereum_schema_cache_task( - Arc::clone(&client), - Arc::clone(&frontier_backend), - )), - ); - - // G -> A1. - let mut builder = client.new_block(Default::default()).unwrap(); - builder.push_storage_change(vec![1], None).unwrap(); - let a1 = builder.build().unwrap().block; - let a1_hash = a1.header.hash(); - executor::block_on(client.import(BlockOrigin::Own, a1)).unwrap(); - - // A1 -> A2, we store V2 schema. - let mut builder = client - .new_block_at(&BlockId::Hash(a1_hash), Default::default(), false) - .unwrap(); - builder - .push_storage_change( - PALLET_ETHEREUM_SCHEMA.to_vec(), - Some(Encode::encode(&EthereumStorageSchema::V2)), - ) - .unwrap(); - let a2 = builder.build().unwrap().block; - let a2_hash = a2.header.hash(); - executor::block_on(client.import(BlockOrigin::Own, a2)).unwrap(); - - // Give some time to consume and process the import notification stream. - thread::sleep(time::Duration::from_millis(100)); - - // Expect: genesis with schema V1, A2 with schema V2. - assert_eq!( - frontier_backend_client::load_cached_schema::<_>(frontier_backend.as_ref()).unwrap(), - Some(vec![ - (EthereumStorageSchema::V1, client.genesis_hash()), - (EthereumStorageSchema::V2, a2_hash) - ]) - ); - - // A1 -> B2. A new block on top of A1. - let mut builder = client - .new_block_at(&BlockId::Hash(a1_hash), Default::default(), false) - .unwrap(); - builder.push_storage_change(vec![2], None).unwrap(); - let b2 = builder.build().unwrap().block; - let b2_hash = b2.header.hash(); - executor::block_on(client.import(BlockOrigin::Own, b2)).unwrap(); - - // B2 -> B3, we store V2 schema again. This is the longest chain. - let mut builder = client - .new_block_at(&BlockId::Hash(b2_hash), Default::default(), false) - .unwrap(); - builder - .push_storage_change( - PALLET_ETHEREUM_SCHEMA.to_vec(), - Some(Encode::encode(&EthereumStorageSchema::V2)), - ) - .unwrap(); - let b3 = builder.build().unwrap().block; - let b3_hash = b3.header.hash(); - executor::block_on(client.import(BlockOrigin::Own, b3)).unwrap(); - - // Give some time to consume and process the import notification stream. - thread::sleep(time::Duration::from_millis(100)); - - // Expect: A2 to be retracted, genesis with schema V1, B3 with schema V2. - assert_eq!( - frontier_backend_client::load_cached_schema::<_>(frontier_backend.as_ref()).unwrap(), - Some(vec![ - (EthereumStorageSchema::V1, client.genesis_hash()), - (EthereumStorageSchema::V2, b3_hash) - ]) - ); - - // A1 -> C2, a wild new block on top of A1. - let mut builder = client - .new_block_at(&BlockId::Hash(a1_hash), Default::default(), false) - .unwrap(); - builder - .push_storage_change( - PALLET_ETHEREUM_SCHEMA.to_vec(), - Some(Encode::encode(&EthereumStorageSchema::V2)), - ) - .unwrap(); - builder.push_storage_change(vec![3], None).unwrap(); - let c2 = builder.build().unwrap().block; - let c2_hash = c2.header.hash(); - executor::block_on(client.import(BlockOrigin::Own, c2)).unwrap(); - - // Give some time to consume and process the import notification stream. - thread::sleep(time::Duration::from_millis(100)); - - // Expect: genesis with schema V1, B3 still with schema V2. - // C2 still not best block and not cached. - assert_eq!( - frontier_backend_client::load_cached_schema::<_>(frontier_backend.as_ref()).unwrap(), - Some(vec![ - (EthereumStorageSchema::V1, client.genesis_hash()), - (EthereumStorageSchema::V2, b3_hash) - ]) - ); - - // Make C2 branch the longest chain. - // C2 -> C3 - let mut builder = client - .new_block_at(&BlockId::Hash(c2_hash), Default::default(), false) - .unwrap(); - builder.push_storage_change(vec![2], None).unwrap(); - let c3 = builder.build().unwrap().block; - let c3_hash = c3.header.hash(); - executor::block_on(client.import(BlockOrigin::Own, c3)).unwrap(); - - // C3 -> C4 - let mut builder = client - .new_block_at(&BlockId::Hash(c3_hash), Default::default(), false) - .unwrap(); - builder.push_storage_change(vec![3], None).unwrap(); - let c4 = builder.build().unwrap().block; - executor::block_on(client.import(BlockOrigin::Own, c4)).unwrap(); - - // Give some time to consume and process the import notification stream. - thread::sleep(time::Duration::from_millis(100)); - - // Expect: B2 branch to be retracted, genesis with schema V1, C2 with schema V2. - // C4 became new best, chain reorged, we expect the C2 ancestor to be cached. - assert_eq!( - frontier_backend_client::load_cached_schema::<_>(frontier_backend.as_ref()).unwrap(), - Some(vec![ - (EthereumStorageSchema::V1, client.genesis_hash()), - (EthereumStorageSchema::V2, c2_hash) - ]) - ); - } -} diff --git a/client/rpc/src/eth/filter.rs b/client/rpc/src/eth/filter.rs index fca7a3f89d..6e0dd142f7 100644 --- a/client/rpc/src/eth/filter.rs +++ b/client/rpc/src/eth/filter.rs @@ -16,7 +16,7 @@ // You should have received a copy of the GNU General Public License // along with this program. If not, see . -use std::{collections::BTreeMap, marker::PhantomData, sync::Arc, time}; +use std::{marker::PhantomData, sync::Arc, time}; use ethereum::BlockV2 as EthereumBlock; use ethereum_types::{H256, U256}; @@ -28,15 +28,11 @@ use sp_blockchain::HeaderBackend; use sp_core::hashing::keccak_256; use sp_runtime::{ generic::BlockId, - traits::{ - BlakeTwo256, Block as BlockT, Header as HeaderT, NumberFor, One, Saturating, - UniqueSaturatedInto, - }, + traits::{BlakeTwo256, Block as BlockT, NumberFor, One, Saturating, UniqueSaturatedInto}, }; use fc_rpc_core::{types::*, EthFilterApiServer}; use fp_rpc::{EthereumRuntimeRPCApi, TransactionStatus}; -use fp_storage::EthereumStorageSchema; use crate::{eth::cache::EthBlockDataCacheTask, frontier_backend_client, internal_err}; @@ -236,7 +232,6 @@ where let client = Arc::clone(&self.client); let block_data_cache = Arc::clone(&self.block_data_cache); - let backend = Arc::clone(&self.backend); let max_past_logs = self.max_past_logs; match path { @@ -269,7 +264,6 @@ where let mut ret: Vec = Vec::new(); let _ = filter_range_logs( client.as_ref(), - backend.as_ref(), &block_data_cache, &mut ret, max_past_logs, @@ -310,7 +304,6 @@ where let client = Arc::clone(&self.client); let block_data_cache = Arc::clone(&self.block_data_cache); - let backend = Arc::clone(&self.backend); let max_past_logs = self.max_past_logs; let filter = filter_result?; @@ -335,7 +328,6 @@ where let mut ret: Vec = Vec::new(); let _ = filter_range_logs( client.as_ref(), - backend.as_ref(), &block_data_cache, &mut ret, max_past_logs, @@ -411,7 +403,6 @@ where let _ = filter_range_logs( client.as_ref(), - backend.as_ref(), &block_data_cache, &mut ret, max_past_logs, @@ -427,7 +418,6 @@ where async fn filter_range_logs( client: &C, - backend: &fc_db::Backend, block_data_cache: &EthBlockDataCacheTask, ret: &mut Vec, max_past_logs: u32, @@ -459,52 +449,13 @@ where let address_bloom_filter = FilteredParams::adresses_bloom_filter(&filter.address); let topics_bloom_filter = FilteredParams::topics_bloom_filter(&topics_input); - // Get schema cache. A single read before the block range iteration. - // This prevents having to do an extra DB read per block range iteration to getthe actual schema. - let mut local_cache: BTreeMap, EthereumStorageSchema> = BTreeMap::new(); - if let Ok(Some(schema_cache)) = frontier_backend_client::load_cached_schema::(backend) { - for (schema, hash) in schema_cache { - if let Ok(Some(header)) = client.header(BlockId::Hash(hash)) { - let number = *header.number(); - local_cache.insert(number, schema); - } - } - } - let cache_keys: Vec> = local_cache.keys().cloned().collect(); - let mut default_schema: Option<&EthereumStorageSchema> = None; - if cache_keys.len() == 1 { - // There is only one schema and that's the one we use. - default_schema = local_cache.get(&cache_keys[0]); - } - while current_number <= to { let id = BlockId::Number(current_number); let substrate_hash = client .expect_block_hash_from_id(&id) .map_err(|_| internal_err(format!("Expect block number from id: {}", id)))?; - let schema = match default_schema { - // If there is a single schema, we just assign. - Some(default_schema) => *default_schema, - _ => { - // If there are multiple schemas, we iterate over the - hopefully short - list - // of keys and assign the one belonging to the current_number. - // Because there are more than 1 schema, and current_number cannot be < 0, - // (i - 1) will always be >= 0. - let mut default_schema: Option<&EthereumStorageSchema> = None; - for (i, k) in cache_keys.iter().enumerate() { - if ¤t_number < k { - default_schema = local_cache.get(&cache_keys[i - 1]); - } - } - match default_schema { - Some(schema) => *schema, - // Fallback to DB read. This will happen i.e. when there is no cache - // task configured at service level. - _ => frontier_backend_client::onchain_storage_schema::(client, id), - } - } - }; + let schema = frontier_backend_client::onchain_storage_schema::(client, id); let block = block_data_cache.current_block(schema, substrate_hash).await; diff --git a/template/node/src/service.rs b/template/node/src/service.rs index 35dcbdee30..0ac8cf1651 100644 --- a/template/node/src/service.rs +++ b/template/node/src/service.rs @@ -743,7 +743,7 @@ fn spawn_frontier_tasks( Duration::new(6, 0), client.clone(), backend, - frontier_backend.clone(), + frontier_backend, 3, 0, SyncStrategy::Normal, @@ -767,16 +767,10 @@ fn spawn_frontier_tasks( "frontier-fee-history", None, EthTask::fee_history_task( - client.clone(), + client, overrides, fee_history_cache, fee_history_cache_limit, ), ); - - task_manager.spawn_essential_handle().spawn( - "frontier-schema-cache-task", - None, - EthTask::ethereum_schema_cache_task(client, frontier_backend), - ); }