Interacting with COAs from Cadence
For Cadence 0.42 go to Legacy Docs
Cadence Owned Accounts (COAs) are EVM accounts owned by a Cadence resource and are used to interact with Flow EVM from Cadence.
COAs expose two interfaces for interaction: one on the Cadence side and one on the EVM side. In this guide, we will focus on how to interact with COAs with Cadence.
In this guide we will walk through some basic examples creating and interacting with a COA in Cadence. Your specific usage of the COA resource will depend on your own application's requirements (e.g. the COA resource may not live directly in /storage/evm
as in these examples, but may instead be a part of a more complex resource structure).
COA Interface​
To begin, we can take a look at a simplified version of the EVM
contract, highlighting parts specific to COAs.
You can learn more about the EVM
contract here and the full contract code can be found on GitHub.
_56access(all)_56contract EVM {_56 //..._56 access(all)_56 resource CadenceOwnedAccount: Addressable {_56 /// The EVM address of the cadence owned account_56 /// -> could be used to query balance, code, nonce, etc._56 access(all)_56 view fun address(): EVM.EVMAddress_56_56 /// Get balance of the cadence owned account_56 /// This balance_56 access(all)_56 view fun balance(): EVM.Balance_56_56 /// Deposits the given vault into the cadence owned account's balance_56 access(all)_56 fun deposit(from: @FlowToken.Vault)_56_56 /// The EVM address of the cadence owned account behind an entitlement, acting as proof of access_56 access(EVM.Owner | EVM.Validate)_56 view fun protectedAddress(): EVM.EVMAddress_56_56 /// Withdraws the balance from the cadence owned account's balance_56 /// Note that amounts smaller than 10nF (10e-8) can't be withdrawn_56 /// given that Flow Token Vaults use UFix64s to store balances._56 /// If the given balance conversion to UFix64 results in_56 /// rounding error, this function would fail._56 access(EVM.Owner | EVM.Withdraw)_56 fun withdraw(balance: EVM.Balance): @FlowToken.Vault_56_56 /// Deploys a contract to the EVM environment._56 /// Returns the address of the newly deployed contract_56 access(EVM.Owner | EVM.Deploy)_56 fun deploy(_56 code: [UInt8],_56 gasLimit: UInt64,_56 value: Balance_56 ): EVM.EVMAddress_56_56 /// Calls a function with the given data._56 /// The execution is limited by the given amount of gas_56 access(EVM.Owner | EVM.Call)_56 fun call(_56 to: EVMAddress,_56 data: [UInt8],_56 gasLimit: UInt64,_56 value: Balance_56 ): EVM.Result_56 }_56_56 // Create a new CadenceOwnedAccount resource_56 access(all)_56 fun createCadenceOwnedAccount(): @EVM.CadenceOwnedAccount_56 // ..._56}
Importing the EVM Contract​
The CadenceOwnedAccount
resource is a part of the EVM
system contract, so to use any of these functions, you will need to begin by importing the EVM
contract into your Cadence code.
To import the EVM
contract into your Cadence code using the simple import syntax, you can use the following format (learn more about configuring contracts in flow.json
here):
_10// This assumes you are working in the in the Flow CLI, FCL, or another tool that supports this syntax_10// The contract address should be configured in your project's `flow.json` file_10import "EVM"_10// ...
However, if you wish to use manual address imports instead, you can use the following format:
_10// Must use the correct address based on the network you are interacting with_10import EVM from 0x1234_10// ...
To find the deployment addresses of the EVM
contract, you can refer to the EVM contract documentation.
Creating a COA​
To create a COA, we can use the createCadenceOwnedAccount
function from the EVM
contract. This function takes no arguments and returns a new CadenceOwnedAccount
resource which represents this newly created EVM account.
For example, we can create this COA in a transaction, saving it to the user's storage and publishing a public capability to its reference:
_17import "EVM"_17_17// Note that this is a simplified example & will not handle cases where the COA already exists_17transaction() {_17 prepare(signer: auth(SaveValue) &Account) {_17 let storagePath = /storage/evm_17 let publicPath = /public/evm_17_17 // Create account & save to storage_17 let coa: @EVM.CadenceOwnedAccount <- EVM.createCadenceOwnedAccount()_17 signer.storage.save(<-coa, to: storagePath)_17_17 // Publish a public capability to the COA_17 let cap = signer.capabilities.storage.issue<&EVM.CadenceOwnedAccount>(storagePath)_17 signer.capabilities.link<&EVM.CadenceOwnedAccount>(cap, at: publicPath)_17 }_17}
Getting the EVM Address of a COA​
To get the EVM address of a COA, you can use the address
function from the EVM
contract. This function returns the EVM address of the COA as an EVM.Address
struct. This struct is used to represent addresses within Flow EVM and can also be used to query the balance, code, nonce, etc. of an account.
For our example, we could query the address of the COA we just created with the following script:
_15import "EVM"_15_15access(all)_15fun main(address: Address): EVM.EVMAddress {_15 // Get the desired Flow account holding the COA in storage_15 let account: = getAuthAccount<auth(Storage) &Account>(address)_15_15 // Borrow a reference to the COA from the storage location we saved it to_15 let coa = account.storage.borrow<&EVM.CadenceOwnedAccount>(_15 from: /storage/evm_15 ) ?? panic("Could not borrow reference to the COA")_15_15 // Return the EVM address of the COA_15 return coa.address()_15}
Getting the Flow Balance of a COA​
Like any other Flow EVM or Cadence account, COAs possess a balance of FLOW tokens. To get the current balance of our COA, we can use the COA's balance
function. It will return a EVM.Balance
struct for the account - these are used to represent balances within Flow EVM.
This script will query the current balance of our newly created COA:
_15import "EVM"_15_15access(all)_15fun main(address: Address): EVM.Balance {_15 // Get the desired Flow account holding the COA in storage_15 let account: = getAuthAccount<auth(Storage) &Account>(address)_15_15 // Borrow a reference to the COA from the storage location we saved it to_15 let coa = account.storage.borrow<&EVM.CadenceOwnedAccount>(_15 from: /storage/evm_15 ) ?? panic("Could not borrow reference to the COA")_15_15 // Get the current balance of this COA_15 return coa.balance()_15}
Depositing and Withdrawing Flow Tokens​
Tokens can be seamlessly transferred between the Flow EVM and Cadence environment using the deposit
and withdraw
functions provided by the COA resource. Anybody with a valid reference to a COA may deposit Flow tokens into a it, however only someone with the Owner
or Withdraw
entitlements can withdraw tokens.
Depositing Flow Tokens​
The deposit
function takes a FlowToken.Vault
resource as an argument, representing the tokens to deposit. It will transfer the tokens from the vault into the COA's balance.
This transaction will withdraw Flow tokens from a user's Cadence vault and deposit them into their COA:
_27import "EVM"_27import "FungibleToken"_27import "FlowToken"_27_27transaction(amount: UFix64) {_27 let coa: &EVM.CadenceOwnedAccount_27 let sentVault: @FlowToken.Vault_27_27 prepare(signer: auth(Capabilities, Storage) &Account) {_27 // Borrow the public capability to the COA from the desired account_27 // This script could be modified to deposit into any account with a `EVM.CadenceOwnedAccount` capability_27 self.coa = signer.capabilities.borrow<&EVM.CadenceOwnedAccount>(_27 from: /public/evm_27 ) ?? panic("Could not borrow reference to the COA")_27_27 // Withdraw the balance from the COA, we will use this later to deposit into the receiving account_27 let vaultRef = signer.storage.borrow<auth(FungibleToken.Withdraw) &FlowToken.Vault>(_27 from: /storage/flowTokenVault_27 ) ?? panic("Could not borrow reference to the owner's Vault")_27 self.sentVault <- vaultRef.withdraw(amount: amount) as! @FlowToken.Vault_27 }_27_27 execute {_27 // Deposit the withdrawn tokens into the COA_27 coa.deposit(from: <-self.sentVault)_27 }_27}
This is a basic example which only transfers tokens between a single user's COA & Flow account. It can be easily modified to transfer these tokens between any arbitrary accounts.
You can also deposit tokens directly into other types of EVM accounts using the EVM.EVMAddress.deposit
function. See the EVM contract documentation for more information.
Withdrawing Flow Tokens​
The withdraw
function takes a EVM.Balance
struct as an argument, representing the amount of Flow tokens to withdraw, and returns a FlowToken.Vault
resource with the withdrawn tokens.
We can run the following transaction to withdraw Flow tokens from a user's COA and deposit them into their Flow vault:
_31import "EVM"_31import "FungibleToken"_31import "FlowToken"_31_31transaction(amount: UFix64) {_31 let sentVault: @FlowToken.Vault_31 let receiver: &{FungibleToken.Receiver}_31_31 prepare(signer: auth(Storage, EVM.Withdraw) &Account) {_31 // Borrow a reference to the COA from the storage location we saved it to with the `EVM.Withdraw` entitlement_31 let coa = signer.storage.borrow<auth(EVM.Withdraw) &EVM.CadenceOwnedAccount>(_31 from: /storage/evm_31 ) ?? panic("Could not borrow reference to the COA")_31_31 // We must create a `EVM.Balance` struct to represent the amount of Flow tokens to withdraw_31 let withdrawBalance = EVM.Balance(attoflow: 0)_31 withdrawBalance.setFLOW(flow: amount)_31_31 // Withdraw the balance from the COA, we will use this later to deposit into the receiving account_31 self.sentVault <- coa.withdraw(balance: withdrawBalance) as! @FlowToken.Vault_31_31 // Borrow the public capability to the receiving account (in this case the signer's own Vault)_31 // This script could be modified to deposit into any account with a `FungibleToken.Receiver` capability_31 self.receiver = signer.capabilities.borrow<&{FungibleToken.Receiver}>(/public/flowTokenReceiver)!_31 }_31_31 execute {_31 // Deposit the withdrawn tokens into the receiving vault_31 receiver.deposit(from: <-self.sentVault)_31 }_31}
This is a basic example which only transfers tokens between a single user's COA & Flow account. It can be easily modified to transfer these tokens between any arbitrary accounts.
Direct Calls to Flow EVM​
To interact with smart contracts on the EVM, you can use the call
function provided by the COA resource. This function takes the EVM address of the contract you want to call, the data you want to send, the gas limit, and the value you want to send. It will return a EVM.Result
struct with the result of the call - you will need to handle this result in your Cadence code.
This transaction will call a contract at a given EVM address with a hardcoded data payload, gas limit, and value using the signer's COA:
_32import "EVM"_32_32transaction() {_32 let coa: auth(EVM.Call) &EVM.CadenceOwnedAccount_32_32 prepare(signer: auth(Storage) &Account) {_32 // Borrow an entitled reference to the COA from the storage location we saved it to_32 self.coa = signer.storage.borrow<auth(EVM.Call) &EVM.CadenceOwnedAccount>(_32 from: /storage/evm_32 ) ?? panic("Could not borrow reference to the COA")_32 }_32_32 execute {_32 // Call the contract at the given EVM address with the given data, gas limit, and value_32 // These values could be configured through the transaction arguments or other means_32 // however, for simplicity, we will hardcode them here_32 let result: EVM.Result = coa.call(_32 to: 0x1234567890123456789012345678901234567890, // INSERT EVM ADDRESS HERE_32 data: [0x01, 0x02, 0x03], // INSERT DATA HERE_32 gasLimit: 15000000, // INSERT GAS LIMIT HERE (attoflow)_32 value: 0.0 // INSERT VALUE HERE_32 )_32_32 // Revert the transaction if the call was not successful_32 // Note: a failing EVM call will not automatically revert the Cadence transaction_32 // and it is up to the developer to use this result however it suits their application_32 assert(_32 result.status == EVM.Status.successful,_32 message: "EVM call failed"_32 )_32 }_32}
Deploying a Contract to Flow EVM​
To deploy a contract to the EVM, you can use the deploy
function provided by the COA resource. This function takes the contract code, gas limit, and value you want to send. It will return the EVM address of the newly deployed contract.
This transaction will deploy a contract with the given code using the signer's COA:
_21import "EVM"_21_21transaction(code: String) {_21 let coa: auth(EVM.Deploy) &EVM.CadenceOwnedAccount_21_21 prepare(signer: auth(Storage) &Account) {_21 // Borrow an entitled reference to the COA from the storage location we saved it to_21 self.coa = signer.storage.borrow<auth(EVM.Deploy) &EVM.CadenceOwnedAccount>(_21 from: /storage/evm_21 ) ?? panic("Could not borrow reference to the COA")_21 }_21_21 execute {_21 // Deploy the contract with the given code, gas limit, and value_21 self.coa.deploy(_21 code: code.decodeHex(),_21 gasLimit: 15000000, // can be adjusted as needed, maxed for simplicity_21 value: EVM.Balance(attoflow: 0)_21 )_21 }_21}
More Information​
For more information about Cadence Owned Accounts, see Flow EVM Accounts.
Other useful snippets for interacting with COAs can be found here.