Arbitrum Stylus logo

Stylus by Example

Errors

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:

  • Recoverable Errors: The Stylus SDK provides features that make using recoverable errors in Rust Stylus contracts convenient. This type of error handling is strongly recommended for Stylus contracts.
  • Unrecoverable Errors: These can be defined similarly to Rust code but are not recommended for smart contracts if recoverable errors can be used instead.

Learn More

Recoverable Errors

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.

Example: Recoverable Errors

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}
  • Using SolidityError Derive Macro: The #[derive(SolidityError)] attribute is used for the MultiCallErrors enum, automatically implementing the necessary traits for error handling.
  • Defining Errors: Custom errors ArraySizeNotMatch and CallFailed is declared in MultiCallErrors enum. CallFailed error includes a call_index parameter to indicate which call failed.
  • ArraySizeNotMatch Error Handling: The multicall function returns ArraySizeNotMatch if the size of addresses and data vectors are not equal.
  • CallFailed Error Handling: The 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.

Unrecoverable Errors

Here are various ways to handle such errors in the multicall function, which calls multiple addresses and panics in different scenarios:

Using 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.

Using 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.

Using 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.

Using the ? Operator

Uses 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.

Boilerplate

src/main.rs

The main code can be found at the top of the page in the recoverable error example section.

Cargo.toml

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"