Skip to main content

Interacting with COAs from Cadence

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.

access(all)
contract EVM {
//...
access(all)
resource CadenceOwnedAccount: Addressable {
/// The EVM address of the cadence owned account
/// -> could be used to query balance, code, nonce, etc.
access(all)
view fun address(): EVM.EVMAddress

/// Get balance of the cadence owned account
/// This balance
access(all)
view fun balance(): EVM.Balance

/// Deposits the given vault into the cadence owned account's balance
access(all)
fun deposit(from: @FlowToken.Vault)

/// The EVM address of the cadence owned account behind an entitlement, acting as proof of access
access(EVM.Owner | EVM.Validate)
view fun protectedAddress(): EVM.EVMAddress

/// Withdraws the balance from the cadence owned account's balance
/// Note that amounts smaller than 10nF (10e-8) can't be withdrawn
/// given that Flow Token Vaults use UFix64s to store balances.
/// If the given balance conversion to UFix64 results in
/// rounding error, this function would fail.
access(EVM.Owner | EVM.Withdraw)
fun withdraw(balance: EVM.Balance): @FlowToken.Vault

/// Deploys a contract to the EVM environment.
/// Returns the address of the newly deployed contract
access(EVM.Owner | EVM.Deploy)
fun deploy(
code: [UInt8],
gasLimit: UInt64,
value: Balance
): EVM.EVMAddress

/// Calls a function with the given data.
/// The execution is limited by the given amount of gas
access(EVM.Owner | EVM.Call)
fun call(
to: EVMAddress,
data: [UInt8],
gasLimit: UInt64,
value: Balance
): EVM.Result
}

// Create a new CadenceOwnedAccount resource
access(all)
fun createCadenceOwnedAccount(): @EVM.CadenceOwnedAccount
// ...
}

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):

// This assumes you are working in the in the Flow CLI, FCL, or another tool that supports this syntax
// The contract address should be configured in your project's `flow.json` file
import "EVM"
// ...

However, if you wish to use manual address imports instead, you can use the following format:

// Must use the correct address based on the network you are interacting with
import EVM from 0x1234
// ...

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:

import "EVM"

// Note that this is a simplified example & will not handle cases where the COA already exists
transaction() {
prepare(signer: auth(SaveValue, IssueStorageCapabilityController, PublishCapability) &Account) {
let storagePath = /storage/evm
let publicPath = /public/evm

// Create account & save to storage
let coa: @EVM.CadenceOwnedAccount <- EVM.createCadenceOwnedAccount()
signer.storage.save(<-coa, to: storagePath)

// Publish a public capability to the COA
let cap = signer.capabilities.storage.issue<&EVM.CadenceOwnedAccount>(storagePath)
signer.capabilities.publish(cap, at: publicPath)
}
}

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:

import "EVM"

access(all)
fun main(address: Address): EVM.EVMAddress {
// Get the desired Flow account holding the COA in storage
let account = getAuthAccount<auth(Storage) &Account>(address)

// Borrow a reference to the COA from the storage location we saved it to
let coa = account.storage.borrow<&EVM.CadenceOwnedAccount>(
from: /storage/evm
) ?? panic("Could not borrow reference to the COA")

// Return the EVM address of the COA
return coa.address()
}

If you'd prefer the hex representation of the address, you instead return using the EVMAddress.toString() function:

return coa.address().toString()

The above will return the EVM address as a string; however note that Cadence does not prefix hex strings with 0x.

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:

import "EVM"

access(all)
fun main(address: Address): EVM.Balance {
// Get the desired Flow account holding the COA in storage
let account = getAuthAccount<auth(Storage) &Account>(address)

// Borrow a reference to the COA from the storage location we saved it to
let coa = account.storage.borrow<&EVM.CadenceOwnedAccount>(
from: /storage/evm
) ?? panic("Could not borrow reference to the COA")

// Get the current balance of this COA
return coa.balance()
}

You can also easily get the UFix64 FLOW balance of any EVM address with this script:

import "EVM"

access(all)
fun main(addressHex: String): UFix64 {
let addr = EVM.addressFromString(addressHex)
return addr.balance().inFLOW()
}

The above script is helpful if you already know the COA address and can provide the hex representation directly.

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:

import "EVM"
import "FungibleToken"
import "FlowToken"

transaction(amount: UFix64) {
let coa: &EVM.CadenceOwnedAccount
let sentVault: @FlowToken.Vault

prepare(signer: auth(BorrowValue) &Account) {
// Borrow the public capability to the COA from the desired account
// This script could be modified to deposit into any account with a `EVM.CadenceOwnedAccount` capability
self.coa = signer.capabilities.borrow<&EVM.CadenceOwnedAccount>(/public/evm)
?? panic("Could not borrow reference to the COA")

// Withdraw the balance from the COA, we will use this later to deposit into the receiving account
let vaultRef = signer.storage.borrow<auth(FungibleToken.Withdraw) &FlowToken.Vault>(
from: /storage/flowTokenVault
) ?? panic("Could not borrow reference to the owner's Vault")
self.sentVault <- vaultRef.withdraw(amount: amount) as! @FlowToken.Vault
}

execute {
// Deposit the withdrawn tokens into the COA
self.coa.deposit(from: <-self.sentVault)
}
}
info

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:

import "EVM"
import "FungibleToken"
import "FlowToken"

transaction(amount: UFix64) {
let sentVault: @FlowToken.Vault
let receiver: &{FungibleToken.Receiver}

prepare(signer: auth(BorrowValue) &Account) {
// Borrow a reference to the COA from the storage location we saved it to with the `EVM.Withdraw` entitlement
let coa = signer.storage.borrow<auth(EVM.Withdraw) &EVM.CadenceOwnedAccount>(
from: /storage/evm
) ?? panic("Could not borrow reference to the COA")

// We must create a `EVM.Balance` struct to represent the amount of Flow tokens to withdraw
let withdrawBalance = EVM.Balance(attoflow: 0)
withdrawBalance.setFLOW(flow: amount)

// Withdraw the balance from the COA, we will use this later to deposit into the receiving account
self.sentVault <- coa.withdraw(balance: withdrawBalance) as! @FlowToken.Vault

// Borrow the public capability to the receiving account (in this case the signer's own Vault)
// This script could be modified to deposit into any account with a `FungibleToken.Receiver` capability
self.receiver = signer.capabilities.borrow<&{FungibleToken.Receiver}>(/public/flowTokenReceiver)!
}

execute {
// Deposit the withdrawn tokens into the receiving vault
self.receiver.deposit(from: <-self.sentVault)
}
}
info

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 use the signer's COA to call a contract method with the defined signature and args at a given EVM address, executing with the provided gas limit and value:

import "EVM"

transaction(evmContractHex: String, signature: String, args: [AnyStruct], gasLimit: UInt64, flowValue: UFix64) {
let coa: auth(EVM.Call) &EVM.CadenceOwnedAccount

prepare(signer: auth(BorrowValue) &Account) {
// Borrow an entitled reference to the COA from the storage location we saved it to
self.coa = signer.storage.borrow<auth(EVM.Call) &EVM.CadenceOwnedAccount>(
from: /storage/evm
) ?? panic("Could not borrow reference to the COA")
}

execute {
// Deserialize the EVM address from the hex string
let contractAddress = EVM.addressFromString(evmContractHex)
// Construct the calldata from the signature and arguments
let calldata = EVM.encodeABIWithSignature(
signature,
args
)
// Define the value as EVM.Balance struct
let value = EVM.Balance(attoflow: 0)
value.setFLOW(flow: flowValue)
// Call the contract at the given EVM address with the given data, gas limit, and value
// These values could be configured through the transaction arguments or other means
// however, for simplicity, we will hardcode them here
let result: EVM.Result = self.coa.call(
to: contractAddress,
data: calldata,
gasLimit: gasLimit,
value: value
)

// Revert the transaction if the call was not successful
// Note: a failing EVM call will not automatically revert the Cadence transaction
// and it is up to the developer to use this result however it suits their application
assert(
result.status == EVM.Status.successful,
message: "EVM call failed"
)
}
}
info

Notice that the calldata is encoded in the scope of the transaction. While developers can encode the calldata outside the scope of the transaction and pass the encoded data as an argument, doing so compromises the human-readability of Cadence transactions.

It's encouraged to either define transactions for each COA call and encoded the hardcoded EVM signature and arguments, or to pass in the human-readable arguments and signature and encode the calldata within the transaction. This ensures a more interpretable and therefore transparent transaction.

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:

import "EVM"

transaction(bytecode: String) {
let coa: auth(EVM.Deploy) &EVM.CadenceOwnedAccount

prepare(signer: auth(BorrowValue) &Account) {
// Borrow an entitled reference to the COA from the storage location we saved it to
self.coa = signer.storage.borrow<auth(EVM.Deploy) &EVM.CadenceOwnedAccount>(
from: /storage/evm
) ?? panic("Could not borrow reference to the COA")
}

execute {
// Deploy the contract with the given compiled bytecode, gas limit, and value
self.coa.deploy(
code: bytecode.decodeHex(),
gasLimit: 15_000_000, // can be adjusted as needed, hard coded here for simplicity
value: EVM.Balance(attoflow: 0)
)
}
}

More Information

For more information about Cadence Owned Accounts, see Flow EVM Accounts.

Other useful snippets for interacting with COAs can be found here.