In Rust Stylus contracts, error handling is a crucial aspect of writing robust and reliable smart contracts. Rust differentiates between recoverable and unrecoverable errors. Recoverable errors are represented using the Result
type, which can either be Ok
, indicating success, or Err
, indicating failure. This allows developers to manage errors gracefully and maintain control over the flow of execution. Unrecoverable errors are handled with the panic!
macro, which stops execution, unwinds the stack, and returns a dataless error.
In Stylus contracts, error types are often explicitly defined, providing clear and structured ways to handle different failure scenarios. This structured approach promotes better error management, ensuring that contracts are secure, maintainable, and behave predictably under various conditions. Similar to Solidity and EVM, errors in Stylus will undo all changes made to the state during a transaction by reverting the transaction. Thus, there are two main types of errors in Rust Stylus contracts:
#[derive(SolidityError)]
alloy_sol_types::SolError
Error handling: Rust book
Recoverable errors are represented using the Result
type, which can either be Ok
, indicating success, or Err
, indicating failure. The Stylus SDK provides tools to define custom error types and manage recoverable errors effectively.
Here's a simplified Rust Stylus contract demonstrating how to define and handle recoverable errors:
1#![cfg_attr(not(feature = "export-abi"), no_main)]
2extern crate alloc;
3
4#[global_allocator]
5static ALLOC: mini_alloc::MiniAlloc = mini_alloc::MiniAlloc::INIT;
6
7use alloy_sol_types::sol;
8use stylus_sdk::{abi::Bytes, alloy_primitives::{Address, U256}, call::RawCall, prelude::*};
9
10#[solidity_storage]
11#[entrypoint]
12pub struct MultiCall;
13
14// Declare events and Solidity error types
15sol! {
16 error ArraySizeNotMatch();
17 error CallFailed(uint256 call_index);
18}
19
20#[derive(SolidityError)]
21pub enum MultiCallErrors {
22 ArraySizeNotMatch(ArraySizeNotMatch),
23 CallFailed(CallFailed),
24}
25
26#[external]
27impl MultiCall {
28 pub fn multicall(
29 &self,
30 addresses: Vec<Address>,
31 data: Vec<Bytes>,
32 ) -> Result<Vec<Bytes>, MultiCallErrors> {
33 let addr_len = addresses.len();
34 let data_len = data.len();
35 let mut results: Vec<Bytes> = Vec::new();
36 if addr_len != data_len {
37 return Err(MultiCallErrors::ArraySizeNotMatch(ArraySizeNotMatch {}));
38 }
39 for i in 0..addr_len {
40 let result: Result<Vec<u8>, Vec<u8>> =
41 RawCall::new().call(addresses[i], data[i].to_vec().as_slice());
42 let data = match result {
43 Ok(data) => data,
44 Err(_data) => return Err(MultiCallErrors::CallFailed(CallFailed { call_index: U256::from(i) })),
45 };
46 results.push(data.into())
47 }
48 Ok(results)
49 }
50}
1#![cfg_attr(not(feature = "export-abi"), no_main)]
2extern crate alloc;
3
4#[global_allocator]
5static ALLOC: mini_alloc::MiniAlloc = mini_alloc::MiniAlloc::INIT;
6
7use alloy_sol_types::sol;
8use stylus_sdk::{abi::Bytes, alloy_primitives::{Address, U256}, call::RawCall, prelude::*};
9
10#[solidity_storage]
11#[entrypoint]
12pub struct MultiCall;
13
14// Declare events and Solidity error types
15sol! {
16 error ArraySizeNotMatch();
17 error CallFailed(uint256 call_index);
18}
19
20#[derive(SolidityError)]
21pub enum MultiCallErrors {
22 ArraySizeNotMatch(ArraySizeNotMatch),
23 CallFailed(CallFailed),
24}
25
26#[external]
27impl MultiCall {
28 pub fn multicall(
29 &self,
30 addresses: Vec<Address>,
31 data: Vec<Bytes>,
32 ) -> Result<Vec<Bytes>, MultiCallErrors> {
33 let addr_len = addresses.len();
34 let data_len = data.len();
35 let mut results: Vec<Bytes> = Vec::new();
36 if addr_len != data_len {
37 return Err(MultiCallErrors::ArraySizeNotMatch(ArraySizeNotMatch {}));
38 }
39 for i in 0..addr_len {
40 let result: Result<Vec<u8>, Vec<u8>> =
41 RawCall::new().call(addresses[i], data[i].to_vec().as_slice());
42 let data = match result {
43 Ok(data) => data,
44 Err(_data) => return Err(MultiCallErrors::CallFailed(CallFailed { call_index: U256::from(i) })),
45 };
46 results.push(data.into())
47 }
48 Ok(results)
49 }
50}
SolidityError
Derive Macro: The #[derive(SolidityError)]
attribute is used for the MultiCallErrors
enum, automatically implementing the necessary traits for error handling.ArraySizeNotMatch
and CallFailed
is declared in MultiCallErrors
enum. CallFailed
error includes a call_index
parameter to indicate which call failed.multicall
function returns ArraySizeNotMatch
if the size of addresses and data vectors are not equal.multicall
function returns a CallFailed
error with the index of the failed call if any call fails. Note that we're using match to check if the result of the call is an error or a return data. We'll describe match pattern in the further sections.Here are various ways to handle such errors in the multicall
function, which calls multiple addresses and panics in different scenarios:
panic!
Directly panics if the call fails, including the index of the failed call.
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice());
3 let data = match result {
4 Ok(data) => data,
5 Err(_data) => panic!("Call to address {:?} failed at index {}", addresses[i], i),
6 };
7 results.push(data.into());
8}
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice());
3 let data = match result {
4 Ok(data) => data,
5 Err(_data) => panic!("Call to address {:?} failed at index {}", addresses[i], i),
6 };
7 results.push(data.into());
8}
Handling Call Failure with panic!
: The function panics if any call fails and the transaction will be reverted without any data.
unwrap
Uses unwrap
to handle the result, panicking if the call fails.
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice()).unwrap();
3 results.push(result.into());
4}
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice()).unwrap();
3 results.push(result.into());
4}
Handling Call Failure with unwrap
: The function uses unwrap
to panic if any call fails, including the index of the failed call.
match
Uses a match
statement to handle the result of call
, panicking if the call fails.
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice());
3 let data = match result {
4 Ok(data) => data,
5 Err(_data) => return Err(MultiCallErrors::CallFailed(CallFailed { call_index: U256::from(i) })),
6 };
7 results.push(data.into());
8}
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice());
3 let data = match result {
4 Ok(data) => data,
5 Err(_data) => return Err(MultiCallErrors::CallFailed(CallFailed { call_index: U256::from(i) })),
6 };
7 results.push(data.into());
8}
Handling Call Failure with match
: The function uses a match
statement to handle the result of call
, returning error if any call fails.
?
OperatorUses the ?
operator to propagate the error if the call fails, including the index of the failed call.
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice())
3 .map_err(|_| MultiCallErrors::CallFailed(CallFailed { call_index: U256::from(i) }))?;
4 results.push(result.into());
5}
1for i in 0..addr_len {
2 let result = RawCall::new().call(addresses[i], data[i].to_vec().as_slice())
3 .map_err(|_| MultiCallErrors::CallFailed(CallFailed { call_index: U256::from(i) }))?;
4 results.push(result.into());
5}
Handling Call Failure with ?
Operator: The function uses the ?
operator to propagate the error if any call fails, including the index of the failed call.
Each method demonstrates a different way to handle unrecoverable errors in the multicall
function of a Rust Stylus contract, providing a comprehensive approach to error management.
Note that as mentioned above, it is strongly recommended to use custom error handling instead of unrecoverable error handling.
The main code can be found at the top of the page in the recoverable error example section.
1[package]
2name = "stylus-multicall-contract"
3version = "0.1.5"
4edition = "2021"
5
6[dependencies]
7alloy-primitives = "0.3.1"
8alloy-sol-types = "0.3.1"
9mini-alloc = "0.4.2"
10stylus-sdk = "0.5.0"
11hex = "0.4.3"
12
13[dev-dependencies]
14tokio = { version = "1.12.0", features = ["full"] }
15ethers = "2.0"
16eyre = "0.6.8"
17
18[features]
19export-abi = ["stylus-sdk/export-abi"]
20
21[[bin]]
22name = "stylus-multicall-contract"
23path = "src/main.rs"
24
25[lib]
26crate-type = ["lib", "cdylib"]
27
28[profile.release]
29codegen-units = 1
30strip = true
31lto = true
32panic = "abort"
33opt-level = "s"
1[package]
2name = "stylus-multicall-contract"
3version = "0.1.5"
4edition = "2021"
5
6[dependencies]
7alloy-primitives = "0.3.1"
8alloy-sol-types = "0.3.1"
9mini-alloc = "0.4.2"
10stylus-sdk = "0.5.0"
11hex = "0.4.3"
12
13[dev-dependencies]
14tokio = { version = "1.12.0", features = ["full"] }
15ethers = "2.0"
16eyre = "0.6.8"
17
18[features]
19export-abi = ["stylus-sdk/export-abi"]
20
21[[bin]]
22name = "stylus-multicall-contract"
23path = "src/main.rs"
24
25[lib]
26crate-type = ["lib", "cdylib"]
27
28[profile.release]
29codegen-units = 1
30strip = true
31lto = true
32panic = "abort"
33opt-level = "s"