mirror of
https://github.com/paradigmxyz/reth.git
synced 2026-02-16 18:05:47 -05:00
133 lines
5.9 KiB
Rust
133 lines
5.9 KiB
Rust
use crate::BlockHashReader;
|
|
use alloy_eips::{BlockHashOrNumber, BlockId, BlockNumberOrTag};
|
|
use alloy_primitives::{BlockNumber, B256};
|
|
use reth_chainspec::ChainInfo;
|
|
use reth_storage_errors::provider::{ProviderError, ProviderResult};
|
|
|
|
/// Client trait for getting important block numbers (such as the latest block number), converting
|
|
/// block hashes to numbers, and fetching a block hash from its block number.
|
|
///
|
|
/// This trait also supports fetching block hashes and block numbers from a [BlockHashOrNumber].
|
|
#[auto_impl::auto_impl(&, Arc)]
|
|
pub trait BlockNumReader: BlockHashReader + Send + Sync {
|
|
/// Returns the current info for the chain.
|
|
fn chain_info(&self) -> ProviderResult<ChainInfo>;
|
|
|
|
/// Returns the best block number in the chain.
|
|
fn best_block_number(&self) -> ProviderResult<BlockNumber>;
|
|
|
|
/// Returns the last block number associated with the last canonical header in the database.
|
|
fn last_block_number(&self) -> ProviderResult<BlockNumber>;
|
|
|
|
/// Gets the `BlockNumber` for the given hash. Returns `None` if no block with this hash exists.
|
|
fn block_number(&self, hash: B256) -> ProviderResult<Option<BlockNumber>>;
|
|
|
|
/// Gets the block number for the given `BlockHashOrNumber`. Returns `None` if no block with
|
|
/// this hash exists. If the `BlockHashOrNumber` is a `Number`, it is returned as is.
|
|
fn convert_hash_or_number(&self, id: BlockHashOrNumber) -> ProviderResult<Option<BlockNumber>> {
|
|
match id {
|
|
BlockHashOrNumber::Hash(hash) => self.block_number(hash),
|
|
BlockHashOrNumber::Number(num) => Ok(Some(num)),
|
|
}
|
|
}
|
|
|
|
/// Gets the block hash for the given `BlockHashOrNumber`. Returns `None` if no block with this
|
|
/// number exists. If the `BlockHashOrNumber` is a `Hash`, it is returned as is.
|
|
fn convert_number(&self, id: BlockHashOrNumber) -> ProviderResult<Option<B256>> {
|
|
match id {
|
|
BlockHashOrNumber::Hash(hash) => Ok(Some(hash)),
|
|
BlockHashOrNumber::Number(num) => self.block_hash(num),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Client trait for transforming [`BlockId`] into block numbers or hashes.
|
|
///
|
|
/// Types that implement this trait must be able to resolve all variants of [`BlockNumberOrTag`] to
|
|
/// block numbers or hashes. Automatic implementations for resolving [`BlockNumberOrTag`] variants
|
|
/// are provided if the type implements the `pending_block_num_hash`, `finalized_block_num`, and
|
|
/// `safe_block_num` methods.
|
|
///
|
|
/// The resulting block numbers can be converted to hashes using the underlying [BlockNumReader]
|
|
/// methods, and vice versa.
|
|
#[auto_impl::auto_impl(&, Arc)]
|
|
pub trait BlockIdReader: BlockNumReader + Send + Sync {
|
|
/// Converts the `BlockNumberOrTag` variants to a block number.
|
|
fn convert_block_number(&self, num: BlockNumberOrTag) -> ProviderResult<Option<BlockNumber>> {
|
|
let num = match num {
|
|
BlockNumberOrTag::Latest => self.best_block_number()?,
|
|
BlockNumberOrTag::Earliest => 0,
|
|
BlockNumberOrTag::Pending => {
|
|
return self
|
|
.pending_block_num_hash()
|
|
.map(|res_opt| res_opt.map(|num_hash| num_hash.number))
|
|
}
|
|
BlockNumberOrTag::Number(num) => num,
|
|
BlockNumberOrTag::Finalized => {
|
|
self.finalized_block_number()?.ok_or(ProviderError::FinalizedBlockNotFound)?
|
|
}
|
|
BlockNumberOrTag::Safe => {
|
|
self.safe_block_number()?.ok_or(ProviderError::SafeBlockNotFound)?
|
|
}
|
|
};
|
|
Ok(Some(num))
|
|
}
|
|
|
|
/// Get the hash of the block by matching the given id.
|
|
fn block_hash_for_id(&self, block_id: BlockId) -> ProviderResult<Option<B256>> {
|
|
match block_id {
|
|
BlockId::Hash(hash) => Ok(Some(hash.into())),
|
|
BlockId::Number(num) => match num {
|
|
BlockNumberOrTag::Latest => Ok(Some(self.chain_info()?.best_hash)),
|
|
BlockNumberOrTag::Pending => self
|
|
.pending_block_num_hash()
|
|
.map(|res_opt| res_opt.map(|num_hash| num_hash.hash)),
|
|
BlockNumberOrTag::Finalized => self.finalized_block_hash(),
|
|
BlockNumberOrTag::Safe => self.safe_block_hash(),
|
|
BlockNumberOrTag::Earliest => self.block_hash(0),
|
|
BlockNumberOrTag::Number(num) => self.block_hash(num),
|
|
},
|
|
}
|
|
}
|
|
|
|
/// Get the number of the block by matching the given id.
|
|
fn block_number_for_id(&self, block_id: BlockId) -> ProviderResult<Option<BlockNumber>> {
|
|
match block_id {
|
|
BlockId::Hash(hash) => self.block_number(hash.into()),
|
|
BlockId::Number(num) => self.convert_block_number(num),
|
|
}
|
|
}
|
|
|
|
/// Get the current pending block number and hash.
|
|
fn pending_block_num_hash(&self) -> ProviderResult<Option<alloy_eips::BlockNumHash>>;
|
|
|
|
/// Get the current safe block number and hash.
|
|
fn safe_block_num_hash(&self) -> ProviderResult<Option<alloy_eips::BlockNumHash>>;
|
|
|
|
/// Get the current finalized block number and hash.
|
|
fn finalized_block_num_hash(&self) -> ProviderResult<Option<alloy_eips::BlockNumHash>>;
|
|
|
|
/// Get the safe block number.
|
|
fn safe_block_number(&self) -> ProviderResult<Option<BlockNumber>> {
|
|
self.safe_block_num_hash().map(|res_opt| res_opt.map(|num_hash| num_hash.number))
|
|
}
|
|
|
|
/// Get the finalized block number.
|
|
fn finalized_block_number(&self) -> ProviderResult<Option<BlockNumber>> {
|
|
self.finalized_block_num_hash().map(|res_opt| res_opt.map(|num_hash| num_hash.number))
|
|
}
|
|
|
|
/// Get the safe block hash.
|
|
fn safe_block_hash(&self) -> ProviderResult<Option<B256>> {
|
|
self.safe_block_num_hash().map(|res_opt| res_opt.map(|num_hash| num_hash.hash))
|
|
}
|
|
|
|
/// Get the finalized block hash.
|
|
fn finalized_block_hash(&self) -> ProviderResult<Option<B256>> {
|
|
self.finalized_block_num_hash().map(|res_opt| res_opt.map(|num_hash| num_hash.hash))
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
fn _object_safe(_: Box<dyn BlockIdReader>) {}
|