Skip to main content

foundry_common/transactions/
receipt.rs

1use alloy_network::{AnyNetwork, AnyTransactionReceipt, Network, TransactionResponse};
2use alloy_primitives::Address;
3use alloy_provider::{
4    Provider,
5    network::{ReceiptResponse, TransactionBuilder},
6};
7use alloy_rpc_types::{BlockId, TransactionReceipt};
8use eyre::Result;
9use foundry_common_fmt::{UIfmt, UIfmtReceiptExt, get_pretty_receipt_attr};
10#[cfg(feature = "optimism")]
11use op_alloy_rpc_types::OpTransactionReceipt;
12use serde::{Deserialize, Serialize};
13use tempo_alloy::rpc::TempoTransactionReceipt;
14
15/// Helper trait providing `contract_address` setter for generic `ReceiptResponse`
16pub trait FoundryReceiptResponse {
17    /// Sets address of the created contract, or `None` if the transaction was not a deployment.
18    fn set_contract_address(&mut self, contract_address: Address);
19}
20
21impl FoundryReceiptResponse for TransactionReceipt {
22    fn set_contract_address(&mut self, contract_address: Address) {
23        self.contract_address = Some(contract_address);
24    }
25}
26
27#[cfg(feature = "optimism")]
28impl FoundryReceiptResponse for OpTransactionReceipt {
29    fn set_contract_address(&mut self, contract_address: Address) {
30        self.inner.contract_address = Some(contract_address);
31    }
32}
33
34impl FoundryReceiptResponse for TempoTransactionReceipt {
35    fn set_contract_address(&mut self, contract_address: Address) {
36        self.contract_address = Some(contract_address);
37    }
38}
39
40/// Helper type to carry a transaction along with an optional revert reason
41#[derive(Clone, Debug, Serialize, Deserialize)]
42pub struct TransactionReceiptWithRevertReason<N: Network> {
43    /// The underlying transaction receipt
44    #[serde(flatten)]
45    pub receipt: N::ReceiptResponse,
46
47    /// The revert reason string if the transaction status is failed
48    #[serde(skip_serializing_if = "Option::is_none", rename = "revertReason")]
49    pub revert_reason: Option<String>,
50}
51
52impl<N: Network> TransactionReceiptWithRevertReason<N>
53where
54    N::TxEnvelope: Clone,
55    N::ReceiptResponse: UIfmtReceiptExt,
56{
57    /// Updates the revert reason field using `eth_call` and returns an Err variant if the revert
58    /// reason was not successfully updated
59    pub async fn update_revert_reason(&mut self, provider: &dyn Provider<N>) -> Result<()> {
60        self.revert_reason = self.fetch_revert_reason(provider).await?;
61        Ok(())
62    }
63
64    async fn fetch_revert_reason(&self, provider: &dyn Provider<N>) -> Result<Option<String>> {
65        // If the transaction succeeded, there is no revert reason to fetch
66        if self.receipt.status() {
67            return Ok(None);
68        }
69
70        let transaction = provider
71            .get_transaction_by_hash(self.receipt.transaction_hash())
72            .await
73            .map_err(|err| eyre::eyre!("unable to fetch transaction: {err}"))?
74            .ok_or_else(|| eyre::eyre!("transaction not found"))?;
75
76        if let Some(block_hash) = self.receipt.block_hash() {
77            let mut call_request: N::TransactionRequest = transaction.as_ref().clone().into();
78            call_request.set_from(transaction.from());
79            match provider.call(call_request).block(BlockId::Hash(block_hash.into())).await {
80                Err(e) => return Ok(extract_revert_reason(e.to_string())),
81                Ok(_) => eyre::bail!("no revert reason as transaction succeeded"),
82            }
83        }
84        eyre::bail!("unable to fetch block_hash")
85    }
86}
87
88impl From<AnyTransactionReceipt> for TransactionReceiptWithRevertReason<AnyNetwork> {
89    fn from(receipt: AnyTransactionReceipt) -> Self {
90        Self { receipt, revert_reason: None }
91    }
92}
93
94impl From<TransactionReceiptWithRevertReason<AnyNetwork>> for AnyTransactionReceipt {
95    fn from(receipt_with_reason: TransactionReceiptWithRevertReason<AnyNetwork>) -> Self {
96        receipt_with_reason.receipt
97    }
98}
99
100impl<N: Network> UIfmt for TransactionReceiptWithRevertReason<N>
101where
102    N::ReceiptResponse: UIfmt,
103{
104    fn pretty(&self) -> String {
105        if let Some(revert_reason) = &self.revert_reason {
106            format!(
107                "{}
108revertReason         {}",
109                self.receipt.pretty(),
110                revert_reason
111            )
112        } else {
113            self.receipt.pretty()
114        }
115    }
116}
117
118fn extract_revert_reason<S: AsRef<str>>(error_string: S) -> Option<String> {
119    let message_substr = "execution reverted: ";
120    error_string
121        .as_ref()
122        .find(message_substr)
123        .map(|index| error_string.as_ref().split_at(index + message_substr.len()).1.to_string())
124}
125
126/// Returns the `UiFmt::pretty()` formatted attribute of the transaction receipt with revert reason
127pub fn get_pretty_receipt_w_reason_attr<N>(
128    receipt: &TransactionReceiptWithRevertReason<N>,
129    attr: &str,
130) -> Option<String>
131where
132    N: Network,
133    N::ReceiptResponse: UIfmtReceiptExt,
134{
135    // Handle revert reason first, then delegate to the receipt formatting function
136    if matches!(attr, "revertReason" | "revert_reason") {
137        return Some(receipt.revert_reason.pretty());
138    }
139    get_pretty_receipt_attr::<N>(&receipt.receipt, attr)
140}
141
142#[cfg(test)]
143mod tests {
144    use super::*;
145
146    #[test]
147    fn test_extract_revert_reason() {
148        let error_string_1 = "server returned an error response: error code 3: execution reverted: Transaction too old";
149        let error_string_2 = "server returned an error response: error code 3: Invalid signature";
150
151        assert_eq!(extract_revert_reason(error_string_1), Some("Transaction too old".to_string()));
152        assert_eq!(extract_revert_reason(error_string_2), None);
153    }
154}