reth_primitives_traits/transaction/
signed.rs1use crate::{InMemorySize, MaybeCompact, MaybeSerde, MaybeSerdeBincodeCompat};
4use alloc::fmt;
5use alloy_consensus::{
6 transaction::{Recovered, RlpEcdsaEncodableTx, SignerRecoverable, TxHashRef},
7 EthereumTxEnvelope, SignableTransaction,
8};
9use alloy_eips::eip2718::{Decodable2718, Encodable2718};
10use alloy_primitives::{keccak256, Address, Signature, B256};
11use alloy_rlp::{Decodable, Encodable};
12use core::hash::Hash;
13
14pub use alloy_consensus::crypto::RecoveryError;
15
16pub trait FullSignedTx: SignedTransaction + MaybeCompact + MaybeSerdeBincodeCompat {}
18impl<T> FullSignedTx for T where T: SignedTransaction + MaybeCompact + MaybeSerdeBincodeCompat {}
19
20#[auto_impl::auto_impl(&, Arc)]
31pub trait SignedTransaction:
32 Send
33 + Sync
34 + Unpin
35 + Clone
36 + fmt::Debug
37 + PartialEq
38 + Eq
39 + Hash
40 + Encodable
41 + Decodable
42 + Encodable2718
43 + Decodable2718
44 + alloy_consensus::Transaction
45 + MaybeSerde
46 + InMemorySize
47 + SignerRecoverable
48 + TxHashRef
49{
50 fn is_broadcastable_in_full(&self) -> bool {
56 !self.is_eip4844()
58 }
59
60 fn try_recover(&self) -> Result<Address, RecoveryError> {
64 self.recover_signer()
65 }
66
67 fn try_recover_unchecked(&self) -> Result<Address, RecoveryError> {
72 self.recover_signer_unchecked()
73 }
74
75 fn recalculate_hash(&self) -> B256 {
78 keccak256(self.encoded_2718())
79 }
80
81 #[auto_impl(keep_default_for(&, Arc))]
83 fn try_clone_into_recovered(&self) -> Result<Recovered<Self>, RecoveryError> {
84 self.recover_signer().map(|signer| Recovered::new_unchecked(self.clone(), signer))
85 }
86
87 #[auto_impl(keep_default_for(&, Arc))]
89 fn try_clone_into_recovered_unchecked(&self) -> Result<Recovered<Self>, RecoveryError> {
90 self.recover_signer_unchecked().map(|signer| Recovered::new_unchecked(self.clone(), signer))
91 }
92
93 #[auto_impl(keep_default_for(&, Arc))]
98 fn try_into_recovered(self) -> Result<Recovered<Self>, Self> {
99 match self.recover_signer() {
100 Ok(signer) => Ok(Recovered::new_unchecked(self, signer)),
101 Err(_) => Err(self),
102 }
103 }
104
105 #[deprecated(note = "Use try_into_recovered_unchecked instead")]
110 #[auto_impl(keep_default_for(&, Arc))]
111 fn into_recovered_unchecked(self) -> Result<Recovered<Self>, RecoveryError> {
112 self.recover_signer_unchecked().map(|signer| Recovered::new_unchecked(self, signer))
113 }
114
115 #[auto_impl(keep_default_for(&, Arc))]
119 fn with_signer(self, signer: Address) -> Recovered<Self> {
120 Recovered::new_unchecked(self, signer)
121 }
122
123 #[auto_impl(keep_default_for(&, Arc))]
127 fn with_signer_ref(&self, signer: Address) -> Recovered<&Self> {
128 Recovered::new_unchecked(self, signer)
129 }
130}
131
132impl<T> SignedTransaction for EthereumTxEnvelope<T>
133where
134 T: RlpEcdsaEncodableTx + SignableTransaction<Signature> + Unpin,
135 Self: Clone + PartialEq + Eq + Decodable + Decodable2718 + MaybeSerde + InMemorySize,
136{
137}
138
139#[cfg(feature = "op")]
140mod op {
141 use super::*;
142 use op_alloy_consensus::{OpPooledTransaction, OpTxEnvelope};
143
144 impl SignedTransaction for OpPooledTransaction {}
145
146 impl SignedTransaction for OpTxEnvelope {}
147}