anvil_zksync_core/node/
error.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
use alloy::hex::ToHexExt;
use anvil_zksync_common::resolver::decode_function_selector;
use async_trait::async_trait;
use zksync_error::anvil_zksync::halt::HaltError;
use zksync_error::anvil_zksync::revert::RevertError;
use zksync_multivm::interface::{Halt, VmRevertReason};

#[derive(thiserror::Error, Debug)]
pub enum LoadStateError {
    #[error("loading state into a node with existing state is not allowed (please create an issue if you have a valid use case)")]
    HasExistingState,
    #[error("loading empty state (no blocks) is not allowed")]
    EmptyState,
    #[error("failed to decompress state: {0}")]
    FailedDecompress(std::io::Error),
    #[error("failed to deserialize state: {0}")]
    FailedDeserialize(serde_json::Error),
    #[error("unknown state version `{0}`")]
    UnknownStateVersion(u8),
    #[error(transparent)]
    Other(#[from] anyhow::Error),
}

async fn handle_vm_revert_reason(reason: &VmRevertReason) -> (String, &[u8]) {
    match reason {
        VmRevertReason::General { msg, data } => (msg.to_string(), data),
        VmRevertReason::InnerTxError => ("Inner transaction error".to_string(), &[]),
        VmRevertReason::VmError => ("VM Error".to_string(), &[]),
        VmRevertReason::Unknown {
            function_selector,
            data,
        } => {
            if function_selector.is_empty() {
                ("Error: no function selector available".to_string(), &[])
            } else {
                let hex_selector = function_selector.encode_hex();
                match decode_function_selector(&hex_selector).await {
                    Ok(Some(decoded_name)) => (decoded_name, data),
                    Ok(None) => (
                        format!("Error with function selector: 0x{hex_selector}"),
                        data,
                    ),
                    Err(e) => (
                        format!(
                            "Error with function selector: 0x{hex_selector}. Decode failure: {e}"
                        ),
                        data,
                    ),
                }
            }
        }
        _ => ("".to_string(), &[]),
    }
}

/// Extracts the VmRevertReason from a Halt, if available.
fn revert_reason(halt: &Halt) -> Option<&VmRevertReason> {
    match halt {
        Halt::ValidationFailed(reason)
        | Halt::PaymasterValidationFailed(reason)
        | Halt::PrePaymasterPreparationFailed(reason)
        | Halt::PayForTxFailed(reason)
        | Halt::FailedToMarkFactoryDependencies(reason)
        | Halt::FailedToChargeFee(reason)
        | Halt::Unknown(reason) => Some(reason),
        _ => None,
    }
}

#[async_trait]
pub trait ToRevertReason {
    async fn to_revert_reason(self) -> RevertError;
}

#[async_trait]
impl ToRevertReason for VmRevertReason {
    async fn to_revert_reason(self) -> RevertError {
        let (message, data) = handle_vm_revert_reason(&self).await;

        match self {
            VmRevertReason::General { .. } => RevertError::General {
                msg: message,
                data: data.encode_hex().into(),
            },
            VmRevertReason::InnerTxError => RevertError::InnerTxError,
            VmRevertReason::VmError => RevertError::VmError,
            VmRevertReason::Unknown { .. } => RevertError::Unknown {
                function_selector: message.encode_hex(),
                data: data.encode_hex(),
            },
            _ => RevertError::Unknown {
                function_selector: message.encode_hex(),
                data: data.encode_hex(),
            },
        }
    }
}

#[async_trait]
pub trait ToHaltError {
    async fn to_halt_error(self) -> HaltError;
}

#[async_trait]
impl ToHaltError for Halt {
    async fn to_halt_error(self) -> HaltError {
        let (msg_opt, data_opt) = if let Some(reason) = revert_reason(&self) {
            let (msg, data) = handle_vm_revert_reason(reason).await;
            (Some(msg), Some(data.encode_hex()))
        } else {
            (None, None)
        };

        let msg = msg_opt.unwrap_or_else(|| "Unknown revert reason".into());
        let data = data_opt.unwrap_or_default();

        match self {
            Halt::ValidationFailed(_) => HaltError::ValidationFailed { msg, data },
            Halt::PaymasterValidationFailed(_) => {
                HaltError::PaymasterValidationFailed { msg, data }
            }
            Halt::PrePaymasterPreparationFailed(_) => {
                HaltError::PrePaymasterPreparationFailed { msg, data }
            }
            Halt::PayForTxFailed(_) => HaltError::PayForTxFailed { msg, data },
            Halt::FailedToMarkFactoryDependencies(_) => {
                HaltError::FailedToMarkFactoryDependencies { msg, data }
            }
            Halt::FailedToChargeFee(_) => HaltError::FailedToChargeFee { msg, data },
            Halt::Unknown(_) => HaltError::Unknown { msg, data },
            // Other variants that do not include a VmRevertReason:
            Halt::UnexpectedVMBehavior(msg) => HaltError::UnexpectedVMBehavior { problem: msg },
            Halt::FailedToSetL2Block(msg) => HaltError::FailedToSetL2Block { msg },
            Halt::FailedToAppendTransactionToL2Block(msg) => {
                HaltError::FailedToAppendTransactionToL2Block { msg }
            }
            Halt::TracerCustom(msg) => HaltError::TracerCustom { msg },
            Halt::FromIsNotAnAccount => HaltError::FromIsNotAnAccount,
            Halt::InnerTxError => HaltError::InnerTxError,
            Halt::BootloaderOutOfGas => HaltError::BootloaderOutOfGas,
            Halt::ValidationOutOfGas => HaltError::ValidationOutOfGas,
            Halt::TooBigGasLimit => HaltError::TooBigGasLimit,
            Halt::NotEnoughGasProvided => HaltError::NotEnoughGasProvided,
            Halt::MissingInvocationLimitReached => HaltError::MissingInvocationLimitReached,
            Halt::VMPanic => HaltError::VMPanic,
            Halt::FailedToPublishCompressedBytecodes => {
                HaltError::FailedToPublishCompressedBytecodes
            }
            Halt::FailedBlockTimestampAssertion => HaltError::FailedBlockTimestampAssertion,
        }
    }
}