reth_rpc_eth_types/
pending_block.rs1use std::{sync::Arc, time::Instant};
6
7use crate::block::BlockAndReceipts;
8use alloy_consensus::BlockHeader;
9use alloy_eips::{BlockId, BlockNumberOrTag};
10use alloy_primitives::{BlockHash, TxHash, B256};
11use derive_more::Constructor;
12use reth_chain_state::{BlockState, ExecutedBlock};
13use reth_ethereum_primitives::Receipt;
14use reth_evm::{ConfigureEvm, EvmEnvFor};
15use reth_primitives_traits::{
16 Block, BlockTy, IndexedTx, NodePrimitives, ReceiptTy, RecoveredBlock, SealedHeader,
17};
18use reth_rpc_convert::{RpcConvert, RpcTypes};
19
20#[derive(Debug, Clone, Constructor)]
22pub struct PendingBlockEnv<Evm: ConfigureEvm> {
23 pub evm_env: EvmEnvFor<Evm>,
25 pub origin: PendingBlockEnvOrigin<BlockTy<Evm::Primitives>, ReceiptTy<Evm::Primitives>>,
27}
28
29#[derive(Clone, Debug)]
31pub enum PendingBlockEnvOrigin<B: Block = reth_ethereum_primitives::Block, R = Receipt> {
32 ActualPending(Arc<RecoveredBlock<B>>, Arc<Vec<R>>),
34 DerivedFromLatest(SealedHeader<B::Header>),
41}
42
43impl<B: Block, R> PendingBlockEnvOrigin<B, R> {
44 pub const fn is_actual_pending(&self) -> bool {
46 matches!(self, Self::ActualPending(_, _))
47 }
48
49 pub fn into_actual_pending(self) -> Option<Arc<RecoveredBlock<B>>> {
51 match self {
52 Self::ActualPending(block, _) => Some(block),
53 _ => None,
54 }
55 }
56
57 pub fn state_block_id(&self) -> BlockId {
62 match self {
63 Self::ActualPending(_, _) => BlockNumberOrTag::Pending.into(),
64 Self::DerivedFromLatest(latest) => BlockId::Hash(latest.hash().into()),
65 }
66 }
67
68 pub fn build_target_hash(&self) -> B256 {
74 match self {
75 Self::ActualPending(block, _) => block.header().parent_hash(),
76 Self::DerivedFromLatest(latest) => latest.hash(),
77 }
78 }
79}
80
81pub type PendingBlockAndReceipts<N> = BlockAndReceipts<N>;
84
85#[derive(Debug, Clone, Constructor)]
87pub struct PendingBlock<N: NodePrimitives> {
88 pub expires_at: Instant,
90 pub receipts: Arc<Vec<ReceiptTy<N>>>,
92 pub executed_block: ExecutedBlock<N>,
94}
95
96impl<N: NodePrimitives> PendingBlock<N> {
97 pub fn with_executed_block(expires_at: Instant, executed_block: ExecutedBlock<N>) -> Self {
100 Self {
101 expires_at,
102 receipts: Arc::new(executed_block.execution_output.receipts.clone()),
103 executed_block,
104 }
105 }
106
107 pub const fn block(&self) -> &Arc<RecoveredBlock<BlockTy<N>>> {
109 &self.executed_block.recovered_block
110 }
111
112 pub fn into_block_and_receipts(self) -> PendingBlockAndReceipts<N> {
115 BlockAndReceipts { block: self.executed_block.recovered_block, receipts: self.receipts }
116 }
117
118 pub fn to_block_and_receipts(&self) -> PendingBlockAndReceipts<N> {
121 BlockAndReceipts {
122 block: self.executed_block.recovered_block.clone(),
123 receipts: self.receipts.clone(),
124 }
125 }
126
127 pub fn parent_hash(&self) -> BlockHash {
129 self.executed_block.recovered_block().parent_hash()
130 }
131
132 pub fn find_transaction_and_receipt_by_hash(
136 &self,
137 tx_hash: TxHash,
138 ) -> Option<(IndexedTx<'_, N::Block>, &N::Receipt)> {
139 let indexed_tx = self.executed_block.recovered_block().find_indexed(tx_hash)?;
140 let receipt = self.receipts.get(indexed_tx.index())?;
141 Some((indexed_tx, receipt))
142 }
143
144 pub fn find_and_convert_transaction_receipt<C>(
149 &self,
150 tx_hash: TxHash,
151 converter: &C,
152 ) -> Option<Result<<C::Network as RpcTypes>::Receipt, C::Error>>
153 where
154 C: RpcConvert<Primitives = N>,
155 {
156 self.to_block_and_receipts().find_and_convert_transaction_receipt(tx_hash, converter)
157 }
158}
159
160impl<N: NodePrimitives> From<PendingBlock<N>> for BlockState<N> {
161 fn from(pending_block: PendingBlock<N>) -> Self {
162 Self::new(pending_block.executed_block)
163 }
164}