In Solidity, there are 3 types of variables: local, state, and global. Local variables are not stored on the blockchain, while state variables are (and incur a much higher cost as a result). This is true of Arbitrum Stylus Rust smart contracts as well, although how they're defined is quite different.
In Rust, local variables are just ordinary variables you assign with let
or let mut
statements. Local variables are far cheaper than state variables, even on the EVM, however, Stylus local variables are more than 100x cheaper to allocate in memory than their Solidity equivalents.
Unlike Solidity, Rust was not built inherently with the blockchain in mind. It is a general purpose programming language. We therefore define specific storage types to explicitly denote values intended to be stored permanently as part of the contract's state. State variables cost the same to store as their Solidity equivalents.
Global variables in Solidity, such as msg.sender
and block.timestamp
, are available as function calls pulled in from the stylus_sdk
with their Rust equivalents being msg::sender()
and block::timestamp()
, respectively. These variables provide information about the blockchain or the active transaction.
1// Only run this as a WASM if the export-abi feature is not set.
2#![cfg_attr(not(any(feature = "export-abi", test)), no_main)]
3extern crate alloc;
4
5use stylus_sdk::alloy_primitives::{U16, U256};
6use stylus_sdk::prelude::*;
7use stylus_sdk::storage::{StorageAddress, StorageBool, StorageU256, StorageString};
8use stylus_sdk::{block, console, msg};
9
10#[storage]
11#[entrypoint]
12pub struct Contract {
13 initialized: StorageBool,
14 owner: StorageAddress,
15 max_supply: StorageU256,
16 base_uri: StorageString,
17}
18
19#[public]
20impl Contract {
21 // State variables are initialized in an `init` function.
22 pub fn init(&mut self) -> Result<(), Vec<u8>> {
23 // We check if contract has been initialized before.
24 // We return if so, we initialize if not.
25 let initialized = self.initialized.get();
26 if initialized {
27 return Ok(());
28 }
29 self.initialized.set(true);
30
31 // We set the contract owner to the caller,
32 // which we get from the global msg module
33 self.owner.set(msg::sender());
34 self.max_supply.set(U256::from(10_000));
35
36 // We set the base URI for the contract
37 // Unlike other variables, string needs to use .set_str() to set the value
38 self.base_uri.set_str("https://stylus-by-example.org/".to_string());
39
40 Ok(())
41 }
42
43 pub fn do_something(&self) -> Result<(), Vec<u8>> {
44 // Local variables are not saved to the blockchain
45 // 16-bit Rust integer
46 let _i = 456_u16;
47 // 16-bit int inferred from U16 Alloy primitive
48 let _j = U16::from(123);
49
50 // Here are some global variables
51 let _timestamp = block::timestamp();
52 let _amount = msg::value();
53 let _base_uri = self.base_uri.get_string();
54
55 console!("Local variables: {_i}, {_j}");
56 console!("Global variables: {_timestamp}, {_amount}");
57 console!("Base uri: {_base_uri}");
58
59 Ok(())
60 }
61}
1// Only run this as a WASM if the export-abi feature is not set.
2#![cfg_attr(not(any(feature = "export-abi", test)), no_main)]
3extern crate alloc;
4
5use stylus_sdk::alloy_primitives::{U16, U256};
6use stylus_sdk::prelude::*;
7use stylus_sdk::storage::{StorageAddress, StorageBool, StorageU256, StorageString};
8use stylus_sdk::{block, console, msg};
9
10#[storage]
11#[entrypoint]
12pub struct Contract {
13 initialized: StorageBool,
14 owner: StorageAddress,
15 max_supply: StorageU256,
16 base_uri: StorageString,
17}
18
19#[public]
20impl Contract {
21 // State variables are initialized in an `init` function.
22 pub fn init(&mut self) -> Result<(), Vec<u8>> {
23 // We check if contract has been initialized before.
24 // We return if so, we initialize if not.
25 let initialized = self.initialized.get();
26 if initialized {
27 return Ok(());
28 }
29 self.initialized.set(true);
30
31 // We set the contract owner to the caller,
32 // which we get from the global msg module
33 self.owner.set(msg::sender());
34 self.max_supply.set(U256::from(10_000));
35
36 // We set the base URI for the contract
37 // Unlike other variables, string needs to use .set_str() to set the value
38 self.base_uri.set_str("https://stylus-by-example.org/".to_string());
39
40 Ok(())
41 }
42
43 pub fn do_something(&self) -> Result<(), Vec<u8>> {
44 // Local variables are not saved to the blockchain
45 // 16-bit Rust integer
46 let _i = 456_u16;
47 // 16-bit int inferred from U16 Alloy primitive
48 let _j = U16::from(123);
49
50 // Here are some global variables
51 let _timestamp = block::timestamp();
52 let _amount = msg::value();
53 let _base_uri = self.base_uri.get_string();
54
55 console!("Local variables: {_i}, {_j}");
56 console!("Global variables: {_timestamp}, {_amount}");
57 console!("Base uri: {_base_uri}");
58
59 Ok(())
60 }
61}
1[package]
2name = "stylus_variable_example"
3version = "0.1.7"
4edition = "2021"
5license = "MIT OR Apache-2.0"
6keywords = ["arbitrum", "ethereum", "stylus", "alloy"]
7
8[dependencies]
9alloy-primitives = "=0.7.6"
10alloy-sol-types = "=0.7.6"
11mini-alloc = "0.4.2"
12stylus-sdk = "0.6.0"
13hex = "0.4.3"
14
15[dev-dependencies]
16tokio = { version = "1.12.0", features = ["full"] }
17ethers = "2.0"
18eyre = "0.6.8"
19
20[features]
21export-abi = ["stylus-sdk/export-abi"]
22
23[lib]
24crate-type = ["lib", "cdylib"]
25
26[profile.release]
27codegen-units = 1
28strip = true
29lto = true
30panic = "abort"
31opt-level = "s"
1[package]
2name = "stylus_variable_example"
3version = "0.1.7"
4edition = "2021"
5license = "MIT OR Apache-2.0"
6keywords = ["arbitrum", "ethereum", "stylus", "alloy"]
7
8[dependencies]
9alloy-primitives = "=0.7.6"
10alloy-sol-types = "=0.7.6"
11mini-alloc = "0.4.2"
12stylus-sdk = "0.6.0"
13hex = "0.4.3"
14
15[dev-dependencies]
16tokio = { version = "1.12.0", features = ["full"] }
17ethers = "2.0"
18eyre = "0.6.8"
19
20[features]
21export-abi = ["stylus-sdk/export-abi"]
22
23[lib]
24crate-type = ["lib", "cdylib"]
25
26[profile.release]
27codegen-units = 1
28strip = true
29lto = true
30panic = "abort"
31opt-level = "s"