Cairo Academy: Understanding an Escrow Contract
This chapter explores an Escrow contract implemented in Cairo. This contract demonstrates a secure token exchange mechanism between two parties, providing a trustless way to facilitate transactions on StarkNet.
Purpose and Functionality
The provided Cairo code defines an escrow contract with the following capabilities:
- Token Escrow: Lock tokens from sender until conditions are met
- Time-Locked Refunds: Allow sender to reclaim tokens after timeout
- Secure Execution: Recipient can execute the trade when ready
- Status Examination: View escrow details at any time
Key Components
Data Structures
#![allow(unused)] fn main() { #[derive(Copy, Drop, Hash)] struct EscrowId { sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress } #[derive(Copy, Default, Drop, Serde, starknet::Store)] pub struct EscrowDetails { pub in_amount: u256, pub out_token: ContractAddress, pub out_amount: u256, pub created_at: u64, } }
Core Functionalities
-
Escrow Creation:
- Sender locks input tokens into the contract
- Specifies recipient and expected output tokens
- Records creation timestamp for refund timing
-
Escrow Execution:
- Recipient can execute the trade when ready
- Atomic swap of input and output tokens
- Only the specified recipient can execute
-
Refund Mechanism:
- Sender can reclaim tokens after 7 days
- Prevents indefinite locking of funds
- Only available after timeout period
-
Status Checking:
- Anyone can examine escrow details
- View token amounts and timestamps
- No state modification
Interface Definition
#![allow(unused)] fn main() { #[starknet::interface] pub trait IEscrow<TContractState> { fn examine( self: @TContractState, sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress ) -> EscrowDetails; fn enter( ref self: TContractState, recipient: ContractAddress, in_token: ContractAddress, in_amount: u256, out_token: ContractAddress, out_amount: u256 ); fn exit(ref self: TContractState, sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress); fn execute(ref self: TContractState, sender: ContractAddress, in_token: ContractAddress); } }
Event System
#![allow(unused)] fn main() { #[event] enum Event { EscrowCreated: EscrowCreatedEvent, EscrowExecuted: EscrowExecutedEvent, EscrowRefunded: EscrowRefundedEvent } }
Security Features
- Token transfers use StarkNet's native token standards
- Time-locked refunds prevent indefinite locking
- Only specified recipient can execute
- Only original sender can refund
- Atomic swap ensures both sides complete or none do
Usage Example
- Creating an Escrow:
#![allow(unused)] fn main() { // Alice wants to trade 100 TOKA for 200 TOKB with Bob escrow.enter( bob_address, token_a_address, 100, token_b_address, 200 ); }
- Executing the Trade:
#![allow(unused)] fn main() { // When Bob is ready, he executes the escrow escrow.execute(alice_address, token_a_address); }
- Refunding:
#![allow(unused)] fn main() { // If Bob doesn't complete within 7 days, Alice can refund escrow.exit(alice_address, bob_address, token_a_address); }
Full Implementation
#![allow(unused)] fn main() { use core::num::traits::Zero; use starknet::ContractAddress; #[derive(Copy, Drop, Hash)] struct EscrowId { sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress } #[derive(Copy, Default, Drop, Serde, starknet::Store)] pub struct EscrowDetails { pub in_amount: u256, pub out_token: ContractAddress, pub out_amount: u256, pub created_at: u64, } #[starknet::interface] pub trait IEscrow<TContractState> { fn examine( self: @TContractState, sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress ) -> EscrowDetails; fn enter( ref self: TContractState, recipient: ContractAddress, in_token: ContractAddress, in_amount: u256, out_token: ContractAddress, out_amount: u256 ); fn exit(ref self: TContractState, sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress); fn execute(ref self: TContractState, sender: ContractAddress, in_token: ContractAddress); } #[starknet::interface] trait IERC20<TContractState> { fn transfer(ref self: TContractState, to: ContractAddress, amount: u256); fn transfer_from( ref self: TContractState, from: ContractAddress, to: ContractAddress, amount: u256 ); } impl ContractAddressDefault of Default<ContractAddress> { fn default() -> ContractAddress { Zero::zero() } } impl EscrowDetailsZero of Zero<EscrowDetails> { fn zero() -> EscrowDetails { EscrowDetails { in_amount: Zero::zero(), out_token: Zero::zero(), out_amount: Zero::zero(), created_at: 0 } } fn is_zero(self: @EscrowDetails) -> bool { self.in_amount.is_zero() && self.out_token.is_zero() && self.out_amount.is_zero() } fn is_non_zero(self: @EscrowDetails) -> bool { !self.is_zero() } } #[starknet::contract] mod escrow { use core::num::traits::Zero; use starknet::{ContractAddress, get_caller_address, get_contract_address, get_block_timestamp}; use super::{IEscrow, IERC20Dispatcher, IERC20DispatcherTrait}; use super::{ContractAddressDefault, EscrowDetails, EscrowId}; #[storage] struct Storage { escrows: LegacyMap<EscrowId, EscrowDetails> } #[event] enum Event { EscrowCreated: EscrowCreatedEvent, EscrowExecuted: EscrowExecutedEvent, EscrowRefunded: EscrowRefundedEvent } #[derive(Drop, Serde)] struct EscrowCreatedEvent { sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress, in_amount: u256, out_token: ContractAddress, out_amount: u256 } #[derive(Drop, Serde)] struct EscrowExecutedEvent { sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress, in_amount: u256, out_token: ContractAddress, out_amount: u256 } #[derive(Drop, Serde)] struct EscrowRefundedEvent { sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress, in_amount: u256 } #[abi(embed_v0)] impl IEscrowImpl of IEscrow<ContractState> { fn examine( self: @ContractState, sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress ) -> EscrowDetails { self.escrows.read(EscrowId { sender, recipient, in_token }) } fn enter( ref self: ContractState, recipient: ContractAddress, in_token: ContractAddress, in_amount: u256, out_token: ContractAddress, out_amount: u256 ) { let caller: ContractAddress = get_caller_address(); let escrow_id = EscrowId { sender: caller, recipient, in_token }; let escrow_details: EscrowDetails = self.escrows.read(escrow_id); assert!(escrow_details.is_zero(), "escrow already exists"); assert!(in_amount > Zero::zero(), "amount must be positive"); assert!(out_amount > Zero::zero(), "amount must be positive"); let created_at = get_block_timestamp(); self.escrows.write(escrow_id, EscrowDetails { in_amount, out_token, out_amount, created_at }); IERC20Dispatcher { contract_address: in_token } .transfer_from(caller, get_contract_address(), in_amount); self.emit(EscrowCreatedEvent { sender: caller, recipient, in_token, in_amount, out_token, out_amount }); } fn exit(ref self: ContractState, sender: ContractAddress, recipient: ContractAddress, in_token: ContractAddress) { let caller: ContractAddress = get_caller_address(); let escrow_id = EscrowId { sender: caller, recipient, in_token }; let escrow_details = self.escrows.read(escrow_id); assert!(escrow_details.is_non_zero(), "escrow does not exist"); assert!(caller == sender, "only sender can refund"); let current_time = get_block_timestamp(); assert!(current_time > escrow_details.created_at + 7 * 24 * 60 * 60, "escrow cannot be refunded yet"); self.escrows.write(escrow_id, Default::default()); IERC20Dispatcher { contract_address: in_token }.transfer(caller, escrow_details.in_amount); self.emit(EscrowRefundedEvent { sender: caller, recipient, in_token, in_amount: escrow_details.in_amount }); } fn execute(ref self: ContractState, sender: ContractAddress, in_token: ContractAddress) { let caller: ContractAddress = get_caller_address(); let escrow_id = EscrowId { sender, recipient: caller, in_token }; let escrow_details: EscrowDetails = self.escrows.read(escrow_id); assert!(escrow_details.is_non_zero(), "escrow does not exist"); assert!(caller == escrow_id.recipient, "only recipient can execute"); self.escrows.write(escrow_id, Default::default()); // Transfer locked tokens to recipient IERC20Dispatcher { contract_address: escrow_id.in_token } .transfer(caller, escrow_details.in_amount); // Transfer expected tokens from recipient to sender IERC20Dispatcher { contract_address: escrow_details.out_token } .transfer_from(caller, sender, escrow_details.out_amount); self.emit(EscrowExecutedEvent { sender, recipient: caller, in_token, in_amount: escrow_details.in_amount, out_token: escrow_details.out_token, out_amount: escrow_details.out_amount }); } } } }
This contract serves as a foundation for building secure token exchange systems on StarkNet, demonstrating key concepts like atomic swaps and time-locked transactions.