Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat/improve fee history performance #5182

Merged
merged 43 commits into from
Nov 27, 2023
Merged
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
d025d82
add structures which implements cache for base_fee_per_gas and used_r…
allnil Oct 23, 2023
a4b9195
prepare blueprint for cache integration for fees values
allnil Oct 24, 2023
558b3c4
add BlockFees type to collect fees data for particular block in cache…
allnil Oct 25, 2023
54204b4
format and clean
allnil Oct 25, 2023
2f40820
add comment
allnil Oct 25, 2023
acbbd76
leave comment regarding collected values
allnil Oct 25, 2023
43a4d4a
delete unused structs
allnil Oct 25, 2023
70dfb48
remove unused imports
allnil Oct 25, 2023
e690f42
remove fee history cache from eth_state, prepare separate type and ca…
allnil Oct 26, 2023
a9b0740
fix tests for single block
allnil Oct 27, 2023
dae2964
add parent_hash in mock data for tests
allnil Oct 27, 2023
52d4156
BTreeMap wrapped with RwLock, prepare task which listens to new canon…
allnil Oct 30, 2023
93ed179
add atomic bounds, refactor cache and provider
allnil Oct 31, 2023
1ae2aae
handle empty block intervals and fee history
allnil Nov 1, 2023
b3c2d8d
prepare ground for futher refactoring, remove provider from FeeHistor…
allnil Nov 3, 2023
e4ade22
refactor logic to use iterator over blocks in on_new_blocks function,…
allnil Nov 4, 2023
1d2f455
Merge branch 'main' of github.com:allnil/reth into feat/improve_fee_h…
allnil Nov 4, 2023
0cde2a1
merge
allnil Nov 4, 2023
912fd0f
prepare type to store fee data
allnil Nov 4, 2023
2805a7a
add annotation for clippy to ignore too many arguments on new fn
allnil Nov 4, 2023
30d0b4c
merge with main
allnil Nov 6, 2023
bbe4483
Merge branch 'main' of github.com:allnil/reth into feat/improve_fee_h…
allnil Nov 14, 2023
398e885
refactor, precalculate percentiles, rewards
allnil Nov 15, 2023
7d072b5
merge
allnil Nov 15, 2023
e213d13
add mock for block_range, update tests
allnil Nov 16, 2023
811a3c8
Merge branch 'main' of github.com:allnil/reth into feat/improve_fee_h…
allnil Nov 16, 2023
08a25fc
add simple test on rewards, calculate rewards on the fly
allnil Nov 16, 2023
ff3aed8
remove test because of transactions, for now
allnil Nov 16, 2023
1bc833a
merge with main
allnil Nov 16, 2023
2f61a8c
Update crates/rpc/rpc/src/eth/api/mod.rs
allnil Nov 20, 2023
66bb10d
merge and fmt
allnil Nov 20, 2023
30cd2ce
Merge branch 'feat/improve_fee_history_performance' of github.com:all…
allnil Nov 20, 2023
b131559
fix docs, add resolution in config, improve naming, move from blocks …
allnil Nov 20, 2023
b904b37
apply nightly rules
allnil Nov 20, 2023
3d4efe7
set proper and idiomatic semicolons
allnil Nov 20, 2023
8e9328c
move types adjacent to fee history to separate file
allnil Nov 22, 2023
9660e55
Merge branch 'main' into feat/improve_fee_history_performance
mattsse Nov 24, 2023
83f9c88
chore: touchups
mattsse Nov 24, 2023
3cbd463
refactor
mattsse Nov 24, 2023
7dc0e83
wip: caching
mattsse Nov 24, 2023
710540e
Merge branch 'main' into feat/improve_fee_history_performance
mattsse Nov 27, 2023
87bd429
cleanup
mattsse Nov 27, 2023
ef27a2f
cleanuo
mattsse Nov 27, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion crates/net/downloaders/src/test_utils/bodies_client.rs
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,7 @@ impl BodiesClient for TestBodiesClient {

Box::pin(async move {
if should_respond_empty {
return Ok((PeerId::default(), vec![]).into());
return Ok((PeerId::default(), vec![]).into())
}

if should_delay {
Expand Down
7 changes: 6 additions & 1 deletion crates/rpc/rpc-builder/src/auth.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,10 @@ use reth_provider::{
StateProviderFactory,
};
use reth_rpc::{
eth::{cache::EthStateCache, gas_oracle::GasPriceOracle, EthFilterConfig},
eth::{
cache::EthStateCache, gas_oracle::GasPriceOracle, EthFilterConfig, FeeHistoryCache,
FeeHistoryCacheConfig,
},
AuthLayer, BlockingTaskPool, Claims, EngineEthApi, EthApi, EthFilter,
EthSubscriptionIdProvider, JwtAuthValidator, JwtSecret,
};
Expand Down Expand Up @@ -58,6 +61,7 @@ where
let eth_cache =
EthStateCache::spawn_with(provider.clone(), Default::default(), executor.clone());
let gas_oracle = GasPriceOracle::new(provider.clone(), Default::default(), eth_cache.clone());
let fee_history_cache = FeeHistoryCache::new(FeeHistoryCacheConfig::default());
let eth_api = EthApi::with_spawner(
provider.clone(),
pool.clone(),
Expand All @@ -67,6 +71,7 @@ where
EthConfig::default().rpc_gas_cap,
Box::new(executor.clone()),
BlockingTaskPool::build().expect("failed to build tracing pool"),
fee_history_cache,
);
let config = EthFilterConfig::default()
.max_logs_per_response(DEFAULT_MAX_LOGS_PER_RESPONSE)
Expand Down
5 changes: 4 additions & 1 deletion crates/rpc/rpc-builder/src/eth.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ use reth_rpc::{
eth::{
cache::{EthStateCache, EthStateCacheConfig},
gas_oracle::GasPriceOracleConfig,
EthFilterConfig, RPC_DEFAULT_GAS_CAP,
EthFilterConfig, FeeHistoryCacheConfig, RPC_DEFAULT_GAS_CAP,
},
BlockingTaskPool, EthApi, EthFilter, EthPubSub,
};
Expand Down Expand Up @@ -46,6 +46,8 @@ pub struct EthConfig {
///
/// Sets TTL for stale filters
pub stale_filter_ttl: std::time::Duration,
/// Settings for the fee history cache
pub fee_history_cache: FeeHistoryCacheConfig,
}

impl EthConfig {
Expand All @@ -71,6 +73,7 @@ impl Default for EthConfig {
max_logs_per_response: DEFAULT_MAX_LOGS_PER_RESPONSE,
rpc_gas_cap: RPC_DEFAULT_GAS_CAP.into(),
stale_filter_ttl: DEFAULT_STALE_FILTER_TTL,
fee_history_cache: FeeHistoryCacheConfig::default(),
}
}
}
Expand Down
16 changes: 16 additions & 0 deletions crates/rpc/rpc-builder/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -152,7 +152,9 @@ use reth_provider::{
use reth_rpc::{
eth::{
cache::{cache_new_blocks_task, EthStateCache},
fee_history_cache_new_blocks_task,
gas_oracle::GasPriceOracle,
FeeHistoryCache,
},
AdminApi, AuthLayer, BlockingTaskGuard, BlockingTaskPool, Claims, DebugApi, EngineEthApi,
EthApi, EthFilter, EthPubSub, EthSubscriptionIdProvider, JwtAuthValidator, JwtSecret, NetApi,
Expand Down Expand Up @@ -1063,13 +1065,26 @@ where
);
let new_canonical_blocks = self.events.canonical_state_stream();
let c = cache.clone();

let fee_history_cache = FeeHistoryCache::new(self.config.eth.fee_history_cache.clone());
self.executor.spawn_critical(
"cache canonical blocks task",
Box::pin(async move {
cache_new_blocks_task(c, new_canonical_blocks).await;
}),
);

let new_canonical_blocks = self.events.canonical_state_stream();
let fhc = fee_history_cache.clone();
let provider_clone = self.provider.clone();
self.executor.spawn_critical(
"cache canonical blocks for fee history task",
Box::pin(async move {
fee_history_cache_new_blocks_task(fhc, new_canonical_blocks, provider_clone)
.await;
}),
);

let executor = Box::new(self.executor.clone());
let blocking_task_pool =
BlockingTaskPool::build().expect("failed to build tracing pool");
Expand All @@ -1082,6 +1097,7 @@ where
self.config.eth.rpc_gas_cap,
executor.clone(),
blocking_task_pool.clone(),
fee_history_cache,
);
let filter = EthFilter::new(
self.provider.clone(),
Expand Down
6 changes: 3 additions & 3 deletions crates/rpc/rpc-engine-api/src/engine_api.rs
Original file line number Diff line number Diff line change
Expand Up @@ -262,12 +262,12 @@ where
self.inner.task_spawner.spawn_blocking(Box::pin(async move {
if count > MAX_PAYLOAD_BODIES_LIMIT {
tx.send(Err(EngineApiError::PayloadRequestTooLarge { len: count })).ok();
return
return;
allnil marked this conversation as resolved.
Show resolved Hide resolved
}

if start == 0 || count == 0 {
tx.send(Err(EngineApiError::InvalidBodiesRange { start, count })).ok();
return
return;
}

let mut result = Vec::with_capacity(count as usize);
Expand All @@ -291,7 +291,7 @@ where
}
Err(err) => {
tx.send(Err(EngineApiError::Internal(Box::new(err)))).ok();
return
return;
}
};
}
Expand Down
58 changes: 36 additions & 22 deletions crates/rpc/rpc/src/eth/api/fees.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,15 +4,16 @@ use crate::{
eth::error::{EthApiError, EthResult},
EthApi,
};

use reth_network_api::NetworkInfo;
use reth_primitives::{
basefee::calculate_next_block_base_fee, BlockNumberOrTag, SealedHeader, U256,
};
use reth_primitives::{basefee::calculate_next_block_base_fee, BlockNumberOrTag, U256};
use reth_provider::{BlockReaderIdExt, ChainSpecProvider, EvmEnvProvider, StateProviderFactory};
use reth_rpc_types::{FeeHistory, TxGasAndReward};
use reth_transaction_pool::TransactionPool;
use tracing::debug;

use super::FeeHistoryEntry;

impl<Provider, Pool, Network> EthApi<Provider, Pool, Network>
where
Pool: TransactionPool + Clone + 'static,
Expand Down Expand Up @@ -75,9 +76,9 @@ where
block_count = end_block_plus;
}

// If reward percentiles were specified, we need to validate that they are monotonically
// If reward percentiles were specified, we
// need to validate that they are monotonically
// increasing and 0 <= p <= 100
//
// Note: The types used ensure that the percentiles are never < 0
if let Some(percentiles) = &reward_percentiles {
if percentiles.windows(2).any(|w| w[0] > w[1] || w[0] > 100.) {
Expand All @@ -90,37 +91,48 @@ where
// Treat a request for 1 block as a request for `newest_block..=newest_block`,
// otherwise `newest_block - 2
// SAFETY: We ensured that block count is capped

let start_block = end_block_plus - block_count;
let headers = self.provider().sealed_headers_range(start_block..=end_block)?;
if headers.len() != block_count as usize {
return Err(EthApiError::InvalidBlockRange)

let mut fee_entries = self.fee_history_cache().get_history(start_block, end_block).await?;
if fee_entries.is_empty() {
fee_entries = self
.provider()
.sealed_headers_range(start_block..=end_block)?
.into_iter()
.map(|header| FeeHistoryEntry::from(&header))
.collect();
}

if fee_entries.len() != block_count as usize {
return Err(EthApiError::InvalidBlockRange)
}
// Collect base fees, gas usage ratios and (optionally) reward percentile data
let mut base_fee_per_gas: Vec<U256> = Vec::new();
let mut gas_used_ratio: Vec<f64> = Vec::new();
let mut rewards: Vec<Vec<U256>> = Vec::new();
for header in &headers {
base_fee_per_gas
.push(U256::try_from(header.base_fee_per_gas.unwrap_or_default()).unwrap());
gas_used_ratio.push(header.gas_used as f64 / header.gas_limit as f64);

// Percentiles were specified, so we need to collect reward percentile ino
for entry in &fee_entries {
base_fee_per_gas.push(U256::try_from(entry.base_fee_per_gas).unwrap());
gas_used_ratio.push(entry.gas_used_ratio);
if let Some(percentiles) = &reward_percentiles {
rewards.push(self.calculate_reward_percentiles(percentiles, header).await?);
rewards.push(self.calculate_reward_percentiles(percentiles, entry).await?);
}
}

// The spec states that `base_fee_per_gas` "[..] includes the next block after the newest of
// the returned range, because this value can be derived from the newest block"
//
// The unwrap is safe since we checked earlier that we got at least 1 header.
let last_header = headers.last().unwrap();

let last_entry = fee_entries.last().unwrap();

let chain_spec = self.provider().chain_spec();

base_fee_per_gas.push(U256::from(calculate_next_block_base_fee(
last_header.gas_used,
last_header.gas_limit,
last_header.base_fee_per_gas.unwrap_or_default(),
last_entry.gas_used,
last_entry.gas_limit,
last_entry.base_fee_per_gas,
chain_spec.base_fee_params,
)));

Expand All @@ -140,11 +152,11 @@ where
async fn calculate_reward_percentiles(
&self,
percentiles: &[f64],
header: &SealedHeader,
fee_entry: &FeeHistoryEntry,
) -> Result<Vec<U256>, EthApiError> {
let (transactions, receipts) = self
.cache()
.get_transactions_and_receipts(header.hash)
.get_transactions_and_receipts(fee_entry.header_hash)
.await?
.ok_or(EthApiError::InvalidBlockRange)?;

Expand All @@ -163,7 +175,9 @@ where

Some(TxGasAndReward {
gas_used,
reward: tx.effective_tip_per_gas(header.base_fee_per_gas).unwrap_or_default(),
reward: tx
.effective_tip_per_gas(Some(fee_entry.base_fee_per_gas))
.unwrap_or_default(),
})
})
.collect::<Vec<_>>();
Expand All @@ -186,7 +200,7 @@ where
continue
}

let threshold = (header.gas_used as f64 * percentile / 100.) as u64;
let threshold = (fee_entry.gas_used as f64 * percentile / 100.) as u64;
while cumulative_gas_used < threshold && tx_index < transactions.len() - 1 {
tx_index += 1;
cumulative_gas_used += transactions[tx_index].gas_used;
Expand Down
Loading
Loading