use std::{collections::HashSet, marker::PhantomData, sync::Arc, time};
use ethereum::BlockV2 as EthereumBlock;
use ethereum_types::{H256, U256};
use jsonrpsee::core::{async_trait, RpcResult};
use sc_client_api::backend::{Backend, StorageProvider};
use sc_transaction_pool::{ChainApi, Pool};
use sc_transaction_pool_api::InPoolTransaction;
use sp_api::ProvideRuntimeApi;
use sp_blockchain::HeaderBackend;
use sp_core::hashing::keccak_256;
use sp_runtime::{
generic::BlockId,
traits::{Block as BlockT, NumberFor, One, Saturating, UniqueSaturatedInto},
};
use fc_rpc_core::{types::*, EthFilterApiServer};
use fp_rpc::{EthereumRuntimeRPCApi, TransactionStatus};
use crate::{eth::cache::EthBlockDataCacheTask, frontier_backend_client, internal_err};
pub struct EthFilter<B: BlockT, C, BE, A: ChainApi> {
client: Arc<C>,
backend: Arc<dyn fc_db::BackendReader<B> + Send + Sync>,
graph: Arc<Pool<A>>,
filter_pool: FilterPool,
max_stored_filters: usize,
max_past_logs: u32,
block_data_cache: Arc<EthBlockDataCacheTask<B>>,
_marker: PhantomData<BE>,
}
impl<B: BlockT, C, BE, A: ChainApi> EthFilter<B, C, BE, A> {
pub fn new(
client: Arc<C>,
backend: Arc<dyn fc_db::BackendReader<B> + Send + Sync>,
graph: Arc<Pool<A>>,
filter_pool: FilterPool,
max_stored_filters: usize,
max_past_logs: u32,
block_data_cache: Arc<EthBlockDataCacheTask<B>>,
) -> Self {
Self {
client,
backend,
graph,
filter_pool,
max_stored_filters,
max_past_logs,
block_data_cache,
_marker: PhantomData,
}
}
}
impl<B, C, BE, A> EthFilter<B, C, BE, A>
where
B: BlockT,
C: ProvideRuntimeApi<B>,
C::Api: EthereumRuntimeRPCApi<B>,
C: HeaderBackend<B> + 'static,
A: ChainApi<Block = B> + 'static,
{
fn create_filter(&self, filter_type: FilterType) -> RpcResult<U256> {
let info = self.client.info();
let best_hash = info.best_hash;
let best_number = UniqueSaturatedInto::<u64>::unique_saturated_into(info.best_number);
let pool = self.filter_pool.clone();
let response = if let Ok(locked) = &mut pool.lock() {
if locked.len() >= self.max_stored_filters {
return Err(internal_err(format!(
"Filter pool is full (limit {:?}).",
self.max_stored_filters
)))
}
let last_key = match {
let mut iter = locked.iter();
iter.next_back()
} {
Some((k, _)) => *k,
None => U256::zero(),
};
let pending_transaction_hashes = if let FilterType::PendingTransaction = filter_type {
let txs_ready = self
.graph
.validated_pool()
.ready()
.map(|in_pool_tx| in_pool_tx.data().clone())
.collect();
let api = self.client.runtime_api();
api.extrinsic_filter(best_hash, txs_ready)
.map_err(|err| {
internal_err(format!("fetch ready transactions failed: {err:?}"))
})?
.into_iter()
.map(|tx| tx.hash())
.collect::<HashSet<_>>()
} else {
HashSet::new()
};
let key = last_key.checked_add(U256::one()).unwrap();
locked.insert(
key,
FilterPoolItem {
last_poll: BlockNumber::Num(best_number),
filter_type,
at_block: best_number,
pending_transaction_hashes,
},
);
Ok(key)
} else {
Err(internal_err("Filter pool is not available."))
};
response
}
}
#[async_trait]
impl<B, C, BE, A> EthFilterApiServer for EthFilter<B, C, BE, A>
where
B: BlockT,
C: ProvideRuntimeApi<B>,
C::Api: EthereumRuntimeRPCApi<B>,
C: HeaderBackend<B> + StorageProvider<B, BE> + 'static,
BE: Backend<B> + 'static,
A: ChainApi<Block = B> + 'static,
{
fn new_filter(&self, filter: Filter) -> RpcResult<U256> {
self.create_filter(FilterType::Log(filter))
}
fn new_block_filter(&self) -> RpcResult<U256> {
self.create_filter(FilterType::Block)
}
fn new_pending_transaction_filter(&self) -> RpcResult<U256> {
self.create_filter(FilterType::PendingTransaction)
}
async fn filter_changes(&self, index: Index) -> RpcResult<FilterChanges> {
enum FuturePath<B: BlockT> {
Block {
last: u64,
next: u64,
},
PendingTransaction {
new_hashes: Vec<H256>,
},
Log {
filter: Filter,
from_number: NumberFor<B>,
current_number: NumberFor<B>,
},
Error(jsonrpsee::core::Error),
}
let key = U256::from(index.value());
let info = self.client.info();
let best_hash = info.best_hash;
let best_number = UniqueSaturatedInto::<u64>::unique_saturated_into(info.best_number);
let pool = self.filter_pool.clone();
let path = if let Ok(locked) = &mut pool.lock() {
if let Some(pool_item) = locked.get(&key).cloned() {
match &pool_item.filter_type {
FilterType::Block => {
let last = pool_item.last_poll.to_min_block_num().unwrap();
let next = best_number + 1;
locked.insert(
key,
FilterPoolItem {
last_poll: BlockNumber::Num(next),
filter_type: pool_item.filter_type.clone(),
at_block: pool_item.at_block,
pending_transaction_hashes: HashSet::new(),
},
);
FuturePath::<B>::Block { last, next }
},
FilterType::PendingTransaction => {
let previous_hashes = pool_item.pending_transaction_hashes;
let txs_ready = self
.graph
.validated_pool()
.ready()
.map(|in_pool_tx| in_pool_tx.data().clone())
.collect();
let api = self.client.runtime_api();
let current_hashes = api
.extrinsic_filter(best_hash, txs_ready)
.map_err(|err| {
internal_err(format!("fetch ready transactions failed: {err:?}"))
})?
.into_iter()
.map(|tx| tx.hash())
.collect::<HashSet<_>>();
locked.insert(
key,
FilterPoolItem {
last_poll: BlockNumber::Num(best_number + 1),
filter_type: pool_item.filter_type.clone(),
at_block: pool_item.at_block,
pending_transaction_hashes: current_hashes.clone(),
},
);
let mew_hashes = current_hashes
.difference(&previous_hashes)
.collect::<HashSet<&H256>>();
FuturePath::PendingTransaction {
new_hashes: mew_hashes.into_iter().copied().collect(),
}
},
FilterType::Log(filter) => {
locked.insert(
key,
FilterPoolItem {
last_poll: BlockNumber::Num(best_number + 1),
filter_type: pool_item.filter_type.clone(),
at_block: pool_item.at_block,
pending_transaction_hashes: HashSet::new(),
},
);
let best_number = self.client.info().best_number;
let mut current_number = filter
.to_block
.and_then(|v| v.to_min_block_num())
.map(|s| s.unique_saturated_into())
.unwrap_or(best_number);
if current_number > best_number {
current_number = best_number;
}
let last_poll = pool_item
.last_poll
.to_min_block_num()
.unwrap()
.unique_saturated_into();
let filter_from = filter
.from_block
.and_then(|v| v.to_min_block_num())
.map(|s| s.unique_saturated_into())
.unwrap_or(last_poll);
let from_number = std::cmp::max(last_poll, filter_from);
FuturePath::Log {
filter: filter.clone(),
from_number,
current_number,
}
},
}
} else {
FuturePath::Error(internal_err(format!("Filter id {:?} does not exist.", key)))
}
} else {
FuturePath::Error(internal_err("Filter pool is not available."))
};
let client = Arc::clone(&self.client);
let backend = Arc::clone(&self.backend);
let block_data_cache = Arc::clone(&self.block_data_cache);
let max_past_logs = self.max_past_logs;
match path {
FuturePath::Error(err) => Err(err),
FuturePath::Block { last, next } => {
let mut ethereum_hashes: Vec<H256> = Vec::new();
for n in last..next {
let id = BlockId::Number(n.unique_saturated_into());
let substrate_hash = client.expect_block_hash_from_id(&id).map_err(|_| {
internal_err(format!("Expect block number from id: {}", id))
})?;
let schema =
fc_storage::onchain_storage_schema(client.as_ref(), substrate_hash);
let block = block_data_cache.current_block(schema, substrate_hash).await;
if let Some(block) = block {
ethereum_hashes.push(block.header.hash())
}
}
Ok(FilterChanges::Hashes(ethereum_hashes))
},
FuturePath::PendingTransaction { new_hashes } => Ok(FilterChanges::Hashes(new_hashes)),
FuturePath::Log {
filter,
from_number,
current_number,
} => {
let mut ret: Vec<Log> = Vec::new();
if backend.is_indexed() {
let _ = filter_range_logs_indexed(
client.as_ref(),
backend.as_ref(),
&block_data_cache,
&mut ret,
max_past_logs,
&filter,
from_number,
current_number,
)
.await?;
} else {
let _ = filter_range_logs(
client.as_ref(),
&block_data_cache,
&mut ret,
max_past_logs,
&filter,
from_number,
current_number,
)
.await?;
}
Ok(FilterChanges::Logs(ret))
},
}
}
async fn filter_logs(&self, index: Index) -> RpcResult<Vec<Log>> {
let key = U256::from(index.value());
let pool = self.filter_pool.clone();
let filter_result: RpcResult<Filter> = (|| {
let pool = pool
.lock()
.map_err(|_| internal_err("Filter pool is not available."))?;
let pool_item = pool
.get(&key)
.ok_or_else(|| internal_err(format!("Filter id {:?} does not exist.", key)))?;
match &pool_item.filter_type {
FilterType::Log(filter) => Ok(filter.clone()),
_ => Err(internal_err(format!(
"Filter id {:?} is not a Log filter.",
key
))),
}
})();
let client = Arc::clone(&self.client);
let backend = Arc::clone(&self.backend);
let block_data_cache = Arc::clone(&self.block_data_cache);
let max_past_logs = self.max_past_logs;
let filter = filter_result?;
let best_number = client.info().best_number;
let mut current_number = filter
.to_block
.and_then(|v| v.to_min_block_num())
.map(|s| s.unique_saturated_into())
.unwrap_or(best_number);
if current_number > best_number {
current_number = best_number;
}
let from_number = filter
.from_block
.and_then(|v| v.to_min_block_num())
.map(|s| s.unique_saturated_into())
.unwrap_or(best_number);
let mut ret: Vec<Log> = Vec::new();
if backend.is_indexed() {
let _ = filter_range_logs_indexed(
client.as_ref(),
backend.as_ref(),
&block_data_cache,
&mut ret,
max_past_logs,
&filter,
from_number,
current_number,
)
.await?;
} else {
let _ = filter_range_logs(
client.as_ref(),
&block_data_cache,
&mut ret,
max_past_logs,
&filter,
from_number,
current_number,
)
.await?;
}
Ok(ret)
}
fn uninstall_filter(&self, index: Index) -> RpcResult<bool> {
let key = U256::from(index.value());
let pool = self.filter_pool.clone();
let response = if let Ok(locked) = &mut pool.lock() {
if locked.remove(&key).is_some() {
Ok(true)
} else {
Err(internal_err(format!("Filter id {:?} does not exist.", key)))
}
} else {
Err(internal_err("Filter pool is not available."))
};
response
}
async fn logs(&self, filter: Filter) -> RpcResult<Vec<Log>> {
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 mut ret: Vec<Log> = Vec::new();
if let Some(hash) = filter.block_hash {
let substrate_hash = match frontier_backend_client::load_hash::<B, C>(
client.as_ref(),
backend.as_ref(),
hash,
)
.await
.map_err(|err| internal_err(format!("{:?}", err)))?
{
Some(hash) => hash,
_ => return Err(crate::err(-32000, "unknown block", None)),
};
let schema = fc_storage::onchain_storage_schema(client.as_ref(), substrate_hash);
let block = block_data_cache.current_block(schema, substrate_hash).await;
let statuses = block_data_cache
.current_transaction_statuses(schema, substrate_hash)
.await;
if let (Some(block), Some(statuses)) = (block, statuses) {
filter_block_logs(&mut ret, &filter, block, statuses);
}
} else {
let best_number = client.info().best_number;
let mut current_number = filter
.to_block
.and_then(|v| v.to_min_block_num())
.map(|s| s.unique_saturated_into())
.unwrap_or(best_number);
if current_number > best_number {
current_number = best_number;
}
let from_number = filter
.from_block
.and_then(|v| v.to_min_block_num())
.map(|s| s.unique_saturated_into())
.unwrap_or(best_number);
if backend.is_indexed() {
let _ = filter_range_logs_indexed(
client.as_ref(),
backend.as_ref(),
&block_data_cache,
&mut ret,
max_past_logs,
&filter,
from_number,
current_number,
)
.await?;
} else {
let _ = filter_range_logs(
client.as_ref(),
&block_data_cache,
&mut ret,
max_past_logs,
&filter,
from_number,
current_number,
)
.await?;
}
}
Ok(ret)
}
}
async fn filter_range_logs_indexed<B, C, BE>(
_client: &C,
backend: &(dyn fc_db::BackendReader<B> + Send + Sync),
block_data_cache: &EthBlockDataCacheTask<B>,
ret: &mut Vec<Log>,
max_past_logs: u32,
filter: &Filter,
from: NumberFor<B>,
to: NumberFor<B>,
) -> RpcResult<()>
where
B: BlockT,
C: ProvideRuntimeApi<B>,
C::Api: EthereumRuntimeRPCApi<B>,
C: HeaderBackend<B> + StorageProvider<B, BE> + 'static,
BE: Backend<B> + 'static,
{
use std::time::Instant;
let timer_start = Instant::now();
let timer_prepare = Instant::now();
let max_duration = time::Duration::from_secs(10);
let begin_request = time::Instant::now();
let topics_input = if filter.topics.is_some() {
let filtered_params = FilteredParams::new(Some(filter.clone()));
Some(filtered_params.flat_topics)
} else {
None
};
let addresses = match &filter.address {
Some(VariadicValue::Single(item)) => vec![*item],
Some(VariadicValue::Multiple(items)) => items.clone(),
_ => vec![],
};
let topics = topics_input
.unwrap_or_default()
.iter()
.map(|flat| match flat {
VariadicValue::Single(item) => vec![*item],
VariadicValue::Multiple(items) => items.clone(),
_ => vec![],
})
.collect::<Vec<Vec<Option<H256>>>>();
let time_prepare = timer_prepare.elapsed().as_millis();
let timer_fetch = Instant::now();
if let Ok(logs) = backend
.filter_logs(
UniqueSaturatedInto::<u64>::unique_saturated_into(from),
UniqueSaturatedInto::<u64>::unique_saturated_into(to),
addresses,
topics,
)
.await
{
let time_fetch = timer_fetch.elapsed().as_millis();
let timer_post = Instant::now();
use std::collections::BTreeMap;
let mut statuses_cache: BTreeMap<B::Hash, Option<Vec<TransactionStatus>>> = BTreeMap::new();
for log in logs.iter() {
let substrate_hash = log.substrate_block_hash;
let schema = log.ethereum_storage_schema;
let ethereum_block_hash = log.ethereum_block_hash;
let block_number = log.block_number;
let db_transaction_index = log.transaction_index;
let db_log_index = log.log_index;
let statuses = if let Some(statuses) = statuses_cache.get(&log.substrate_block_hash) {
statuses.clone()
} else {
let statuses = block_data_cache
.current_transaction_statuses(schema, substrate_hash)
.await;
statuses_cache.insert(log.substrate_block_hash, statuses.clone());
statuses
};
if let Some(statuses) = statuses {
let mut block_log_index: u32 = 0;
for status in statuses.iter() {
let mut transaction_log_index: u32 = 0;
let transaction_hash = status.transaction_hash;
let transaction_index = status.transaction_index;
for ethereum_log in &status.logs {
if transaction_index == db_transaction_index
&& transaction_log_index == db_log_index
{
ret.push(Log {
address: ethereum_log.address,
topics: ethereum_log.topics.clone(),
data: Bytes(ethereum_log.data.clone()),
block_hash: Some(ethereum_block_hash),
block_number: Some(U256::from(block_number)),
transaction_hash: Some(transaction_hash),
transaction_index: Some(U256::from(transaction_index)),
log_index: Some(U256::from(block_log_index)),
transaction_log_index: Some(U256::from(transaction_log_index)),
removed: false,
});
}
transaction_log_index += 1;
block_log_index += 1;
}
}
}
if ret.len() as u32 > max_past_logs {
return Err(internal_err(format!(
"query returned more than {} results",
max_past_logs
)))
}
if begin_request.elapsed() > max_duration {
return Err(internal_err(format!(
"query timeout of {} seconds exceeded",
max_duration.as_secs()
)))
}
}
let time_post = timer_post.elapsed().as_millis();
log::info!(
target: "frontier-sql",
"OUTER-TIMER fetch={}, post={}",
time_fetch,
time_post,
);
}
log::info!(
target: "frontier-sql",
"OUTER-TIMER start={}, prepare={}, all_fetch = {}",
timer_start.elapsed().as_millis(),
time_prepare,
timer_fetch.elapsed().as_millis(),
);
Ok(())
}
async fn filter_range_logs<B, C, BE>(
client: &C,
block_data_cache: &EthBlockDataCacheTask<B>,
ret: &mut Vec<Log>,
max_past_logs: u32,
filter: &Filter,
from: NumberFor<B>,
to: NumberFor<B>,
) -> RpcResult<()>
where
B: BlockT,
C: ProvideRuntimeApi<B>,
C::Api: EthereumRuntimeRPCApi<B>,
C: HeaderBackend<B> + StorageProvider<B, BE> + 'static,
BE: Backend<B> + 'static,
{
let max_duration = time::Duration::from_secs(10);
let begin_request = time::Instant::now();
let mut current_number = from;
let topics_input = if filter.topics.is_some() {
let filtered_params = FilteredParams::new(Some(filter.clone()));
Some(filtered_params.flat_topics)
} else {
None
};
let address_bloom_filter = FilteredParams::adresses_bloom_filter(&filter.address);
let topics_bloom_filter = FilteredParams::topics_bloom_filter(&topics_input);
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 = fc_storage::onchain_storage_schema(client, substrate_hash);
let block = block_data_cache.current_block(schema, substrate_hash).await;
if let Some(block) = block {
if FilteredParams::address_in_bloom(block.header.logs_bloom, &address_bloom_filter)
&& FilteredParams::topics_in_bloom(block.header.logs_bloom, &topics_bloom_filter)
{
let statuses = block_data_cache
.current_transaction_statuses(schema, substrate_hash)
.await;
if let Some(statuses) = statuses {
filter_block_logs(ret, filter, block, statuses);
}
}
}
if ret.len() as u32 > max_past_logs {
return Err(internal_err(format!(
"query returned more than {} results",
max_past_logs
)))
}
if begin_request.elapsed() > max_duration {
return Err(internal_err(format!(
"query timeout of {} seconds exceeded",
max_duration.as_secs()
)))
}
if current_number == to {
break
} else {
current_number = current_number.saturating_add(One::one());
}
}
Ok(())
}
fn filter_block_logs<'a>(
ret: &'a mut Vec<Log>,
filter: &'a Filter,
block: EthereumBlock,
transaction_statuses: Vec<TransactionStatus>,
) -> &'a Vec<Log> {
let params = FilteredParams::new(Some(filter.clone()));
let mut block_log_index: u32 = 0;
let block_hash = H256::from(keccak_256(&rlp::encode(&block.header)));
for status in transaction_statuses.iter() {
let mut transaction_log_index: u32 = 0;
let transaction_hash = status.transaction_hash;
for ethereum_log in &status.logs {
let mut log = Log {
address: ethereum_log.address,
topics: ethereum_log.topics.clone(),
data: Bytes(ethereum_log.data.clone()),
block_hash: None,
block_number: None,
transaction_hash: None,
transaction_index: None,
log_index: None,
transaction_log_index: None,
removed: false,
};
let mut add: bool = true;
match (filter.address.clone(), filter.topics.clone()) {
(Some(_), Some(_)) => {
if !params.filter_address(&log) || !params.filter_topics(&log) {
add = false;
}
},
(Some(_), _) =>
if !params.filter_address(&log) {
add = false;
},
(_, Some(_)) =>
if !params.filter_topics(&log) {
add = false;
},
_ => {},
}
if add {
log.block_hash = Some(block_hash);
log.block_number = Some(block.header.number);
log.transaction_hash = Some(transaction_hash);
log.transaction_index = Some(U256::from(status.transaction_index));
log.log_index = Some(U256::from(block_log_index));
log.transaction_log_index = Some(U256::from(transaction_log_index));
ret.push(log);
}
transaction_log_index += 1;
block_log_index += 1;
}
}
ret
}