We have three main ways to send Ether in Rust Stylus: using the transfer_eth
method, using low level call
method, and sending value while calling an external contract.
It's important to note that the transfer_eth
method in Rust Stylus invokes the recipient contract, which may subsequently call other contracts. All the gas is supplied to the recipient, which it may burn. Conversely, the transfer method in Solidity is capped at 2300 gas. In Rust Stylus, you can cap the gas by using the low-level call method with a specified gas. An example of this is provided in the code on bottom of the page.
These two methods are exactly equivalent under the hood:
1transfer_eth(recipient, value)?;
2
3call(Call::new().value(value), recipient, &[])?;
1transfer_eth(recipient, value)?;
2
3call(Call::new().value(value), recipient, &[])?;
Externally Owned Account (EOA) Addresses: Directly send Ether to an EOA address.
Solidity Smart Contracts with Receive Function (No Calldata): Send Ether to a Solidity smart contract that has a receive
function without providing any calldata.
Solidity Smart Contracts with Fallback Function (With Calldata): Send Ether to a Solidity smart contract that has a fallback
function by providing the necessary calldata.
Smart Contracts with Payable Methods (both Solidity and Stylus): Send Ether to smart contracts that have defined payable methods.
Below you can find examples for each of these methods and how to define them in a Rust Stylus smart contract using the Stylus SDK:
src/main.rs
1#![no_main]
2extern crate alloc;
3
4#[global_allocator]
5static ALLOC: mini_alloc::MiniAlloc = mini_alloc::MiniAlloc::INIT;
6use alloc::vec;
7use alloc::vec::Vec;
8
9use alloy_primitives::Address;
10use stylus_sdk::{
11 abi::Bytes,
12 call::{call, transfer_eth, Call},
13 msg::{self},
14 prelude::*,
15};
16
17sol_interface! {
18 interface ITarget {
19 function receiveEther() external payable;
20 }
21}
22
23#[solidity_storage]
24#[entrypoint]
25pub struct SendEther {}
26
27#[external]
28impl SendEther {
29 // Transfer Ether using the transfer_eth method
30 // This can be used to send Ether to an EOA or a Solidity smart contract that has a receive() function implemented
31 #[payable]
32 pub fn send_via_transfer(to: Address) -> Result<(), Vec<u8>> {
33 transfer_eth(to, msg::value())?;
34 Ok(())
35 }
36
37 // Transfer Ether using a low-level call
38 // This can be used to send Ether to an EOA or a Solidity smart contract that has a receive() function implemented
39 #[payable]
40 pub fn send_via_call(&mut self, to: Address) -> Result<(), Vec<u8>> {
41 call(Call::new_in(self).value(msg::value()), to, &[])?;
42 Ok(())
43 }
44
45 // Transfer Ether using a low-level call with a specified gas limit
46 // This can be used to send Ether to an EOA or a Solidity smart contract that has a receive() function implemented
47 #[payable]
48 pub fn send_via_call_gas_limit(&mut self, to: Address, gas_amount: u64) -> Result<(), Vec<u8>> {
49 call(
50 Call::new_in(self).value(msg::value()).gas(gas_amount),
51 to,
52 &[],
53 )?;
54 Ok(())
55 }
56
57 // Transfer Ether using a low-level call with calldata
58 // This can be used to call a Solidity smart contract's fallback function and send Ether along with calldata
59 #[payable]
60 pub fn send_via_call_with_call_data(
61 &mut self,
62 to: Address,
63 data: Bytes,
64 ) -> Result<(), Vec<u8>> {
65 call(Call::new_in(self).value(msg::value()), to, data.as_slice())?;
66 Ok(())
67 }
68
69 // Transfer Ether to another smart contract via a payable method on the target contract
70 // The target contract can be either a Solidity smart contract or a Stylus contract that has a receiveEther function, which is a payable function
71 #[payable]
72 pub fn send_to_stylus_contract(&mut self, to: Address) -> Result<(), Vec<u8>> {
73 let target = ITarget::new(to);
74 let config = Call::new_in(self).value(msg::value());
75 target.receive_ether(config)?;
76 Ok(())
77 }
78}
1#![no_main]
2extern crate alloc;
3
4#[global_allocator]
5static ALLOC: mini_alloc::MiniAlloc = mini_alloc::MiniAlloc::INIT;
6use alloc::vec;
7use alloc::vec::Vec;
8
9use alloy_primitives::Address;
10use stylus_sdk::{
11 abi::Bytes,
12 call::{call, transfer_eth, Call},
13 msg::{self},
14 prelude::*,
15};
16
17sol_interface! {
18 interface ITarget {
19 function receiveEther() external payable;
20 }
21}
22
23#[solidity_storage]
24#[entrypoint]
25pub struct SendEther {}
26
27#[external]
28impl SendEther {
29 // Transfer Ether using the transfer_eth method
30 // This can be used to send Ether to an EOA or a Solidity smart contract that has a receive() function implemented
31 #[payable]
32 pub fn send_via_transfer(to: Address) -> Result<(), Vec<u8>> {
33 transfer_eth(to, msg::value())?;
34 Ok(())
35 }
36
37 // Transfer Ether using a low-level call
38 // This can be used to send Ether to an EOA or a Solidity smart contract that has a receive() function implemented
39 #[payable]
40 pub fn send_via_call(&mut self, to: Address) -> Result<(), Vec<u8>> {
41 call(Call::new_in(self).value(msg::value()), to, &[])?;
42 Ok(())
43 }
44
45 // Transfer Ether using a low-level call with a specified gas limit
46 // This can be used to send Ether to an EOA or a Solidity smart contract that has a receive() function implemented
47 #[payable]
48 pub fn send_via_call_gas_limit(&mut self, to: Address, gas_amount: u64) -> Result<(), Vec<u8>> {
49 call(
50 Call::new_in(self).value(msg::value()).gas(gas_amount),
51 to,
52 &[],
53 )?;
54 Ok(())
55 }
56
57 // Transfer Ether using a low-level call with calldata
58 // This can be used to call a Solidity smart contract's fallback function and send Ether along with calldata
59 #[payable]
60 pub fn send_via_call_with_call_data(
61 &mut self,
62 to: Address,
63 data: Bytes,
64 ) -> Result<(), Vec<u8>> {
65 call(Call::new_in(self).value(msg::value()), to, data.as_slice())?;
66 Ok(())
67 }
68
69 // Transfer Ether to another smart contract via a payable method on the target contract
70 // The target contract can be either a Solidity smart contract or a Stylus contract that has a receiveEther function, which is a payable function
71 #[payable]
72 pub fn send_to_stylus_contract(&mut self, to: Address) -> Result<(), Vec<u8>> {
73 let target = ITarget::new(to);
74 let config = Call::new_in(self).value(msg::value());
75 target.receive_ether(config)?;
76 Ok(())
77 }
78}
Cargo.toml
1[package]
2name = "stylus-sending-ether"
3version = "0.1.7"
4edition = "2021"
5keywords = ["arbitrum", "ethereum", "stylus", "alloy"]
6description = "Stylus sending ether example"
7
8[dependencies]
9alloy-primitives = "0.7.6"
10alloy-sol-types = "0.7.6"
11mini-alloc = "0.4.2"
12stylus-sdk = "0.5.2"
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[[bin]]
24name = "stylus-multi-call"
25path = "src/main.rs"
26
27[lib]
28crate-type = ["lib", "cdylib"]
29
30[profile.release]
31codegen-units = 1
32strip = true
33lto = true
34panic = "abort"
35opt-level = "s"
1[package]
2name = "stylus-sending-ether"
3version = "0.1.7"
4edition = "2021"
5keywords = ["arbitrum", "ethereum", "stylus", "alloy"]
6description = "Stylus sending ether example"
7
8[dependencies]
9alloy-primitives = "0.7.6"
10alloy-sol-types = "0.7.6"
11mini-alloc = "0.4.2"
12stylus-sdk = "0.5.2"
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[[bin]]
24name = "stylus-multi-call"
25path = "src/main.rs"
26
27[lib]
28crate-type = ["lib", "cdylib"]
29
30[profile.release]
31codegen-units = 1
32strip = true
33lto = true
34panic = "abort"
35opt-level = "s"