Skip to main content

EVM Quickstart

Flow EVM is an EVM-equivalent blockchain that combines the advantages of Flow, including security, low-cost gas, and native VRF with compatibility with exiting blockchain applications tools, and contracts. If it works on another EVM-equivalent blockchain, it should work on Flow EVM!

This guide is a self-contained quickstart that will walk you through deploying a contract on Flow EVM testnet with Hardhat and testing it with testnet Flowscan.

If you prefer, check out our tutorials for Remix and Foundry for information on how to deploy a contract with those platforms.

Objectives

After completing this guide, you'll be able to:

  • Fund a wallet with testnet tokens from the Flow Faucet
  • Deploy a contract on Flow EVM Testnet
  • Interact with a contract using Flowscan
  • Utilize automatically sponsored gas with the Flow Wallet on testnet and mainnet

Prerequisites

Traditional Cryptocurrency Wallet

EVM Accounts created by the Flow wallet have unique properties that allow for powerful features, but they do not have recovery phrases or private keys that can be exported in a way that's compatible with Hardhat. As a result, you'll need to use a traditional EOA and MetaMask, or the wallet of your choice, to deploy your contracts.

Deploy Your Contract

For this exercise, we'll use a Button Clicker Contract that's relatively simple, but includes several OpenZeppelin contracts. This way, we can walk through the process to configure your project to use these common imports.

info

If you really want to speedrun this tutorial, fork the Button Clicker Contract repo, run npm install, add a .env with your deploy wallet key as DEPLOY_WALLET_1, and deploy with npx hardhat ignition deploy ./ignition/modules/ClickToken.ts --network flowTestnet.

Then skip to the frontend section.

Hardhat Setup

Open a terminal window and navigate either to the folder where you wish to create your project folder, or an empty project folder. Run:


_10
npx hardhat init

Hardhat Init

Select Create a TypeScript project (with Viem)

Enter . if you ran the command from an empty folder, or enter a path.

Choose the defaults for the remaining options, then open the project in your editor.

Environment Setup

Add a .env and in it, add an environment variable called DEPLOY_WALLET_1 with your deployment wallet's private key.


_10
DEPLOY_WALLET_1=<YOUR_PRIVATE_KEY>

danger

The private key functions the same as the recovery phrase for a wallet. Anyone with the key can drain the wallet at any time! Use separate wallets for development and never commit a key to a repo.

Hardhat Config

We'll be using OpenZeppelin Contracts, so install them, then open the project in your editor:


_10
npm install --save-dev @openzeppelin/hardhat-upgrades
_10
npm install --save-dev @nomicfoundation/hardhat-ethers ethers # peer dependencies

Then install the contracts themselves:


_10
npm install --save-dev @openzeppelin/contracts

You'll also need dotenv to better protect your wallet key, so go ahead and add that too:


_10
npm install dotenv

Open hardhat.config. Below the imports, add the require statements for the contracts and dotenv:


_10
require('@openzeppelin/hardhat-upgrades');
_10
require('dotenv').config();

The default config is pretty bare. We'll need to add quite a few items. We'll do these one at a time, then provide a complete copy at the end.

First, add a networks property containing the network information for Flow Testnet and Mainnet:


_10
networks: {
_10
flow: {
_10
url: 'https://mainnet.evm.nodes.onflow.org',
_10
accounts: [process.env.DEPLOY_WALLET_1 as string],
_10
},
_10
flowTestnet: {
_10
url: 'https://testnet.evm.nodes.onflow.org',
_10
accounts: [process.env.DEPLOY_WALLET_1 as string],
_10
},
_10
},

Then, add an entry for etherscan:


_10
etherscan: {
_10
}

In it, add a property for apiKey and add keys for Flow Mainnet and Testnet. Note that the Etherscan API requires this to be here, but at the time of writing, API keys aren't actually needed. Any text can be used:


_10
apiKey: {
_10
// Is not required by blockscout. Can be any non-empty string
_10
'flow': "abc",
_10
'flowTestnet': "abc"
_10
},

Next, add customChains and the network information for Flow:


_18
customChains: [
_18
{
_18
network: 'flow',
_18
chainId: 747,
_18
urls: {
_18
apiURL: 'https://evm.flowscan.io/api',
_18
browserURL: 'https://evm.flowscan.io/',
_18
},
_18
},
_18
{
_18
network: 'flowTestnet',
_18
chainId: 545,
_18
urls: {
_18
apiURL: 'https://evm-testnet.flowscan.io/api',
_18
browserURL: 'https://evm-testnet.flowscan.io/',
_18
},
_18
},
_18
];

You should end up with:


_46
import type { HardhatUserConfig } from 'hardhat/config';
_46
import '@nomicfoundation/hardhat-toolbox-viem';
_46
_46
require('@openzeppelin/hardhat-upgrades');
_46
require('dotenv').config();
_46
_46
const config: HardhatUserConfig = {
_46
solidity: '0.8.28',
_46
networks: {
_46
flow: {
_46
url: 'https://mainnet.evm.nodes.onflow.org',
_46
accounts: [process.env.DEPLOY_WALLET_1 as string],
_46
},
_46
flowTestnet: {
_46
url: 'https://testnet.evm.nodes.onflow.org',
_46
accounts: [process.env.DEPLOY_WALLET_1 as string],
_46
},
_46
},
_46
etherscan: {
_46
apiKey: {
_46
// Is not required by blockscout. Can be any non-empty string
_46
flow: 'abc',
_46
flowTestnet: 'abc',
_46
},
_46
customChains: [
_46
{
_46
network: 'flow',
_46
chainId: 747,
_46
urls: {
_46
apiURL: 'https://evm.flowscan.io/api',
_46
browserURL: 'https://evm.flowscan.io/',
_46
},
_46
},
_46
{
_46
network: 'flowTestnet',
_46
chainId: 545,
_46
urls: {
_46
apiURL: 'https://evm-testnet.flowscan.io/api',
_46
browserURL: 'https://evm-testnet.flowscan.io/',
_46
},
_46
},
_46
],
_46
},
_46
};
_46
_46
export default config;

Contract Setup

Delete Lock.sol and add ClickToken.sol. In it add, the Button Clicker Contract.

warning

Hardhat only installs the most current version of Solidity. ^0.8.27 means that this contract requires 0.8.27 or higher. You generally don't want to include the ^ in your contracts unless you have a specific reason for doing so.

We won't go into the details of the contract for this tutorial. It's a relatively simple ERC-20 implementation that mints one token any time the mintTo function is called. Perfect for a Button Clicker game!

Deployment Setup

Delete Lock.ts from the ignition/modules folder, and add ClickToken.ts. In it, add:


_12
// This setup uses Hardhat Ignition to manage smart contract deployments.
_12
// Learn more about it at https://hardhat.org/ignition
_12
_12
import { buildModule } from '@nomicfoundation/hardhat-ignition/modules';
_12
_12
const ClickerModule = buildModule('ClickTokenModule', (m) => {
_12
const clickToken = m.contract('ClickToken');
_12
_12
return { clickToken };
_12
});
_12
_12
export default ClickerModule;

Obtain Testnet Funds

Visit the Flow Faucet and follow the instructions to add testnet funds. Compared to other networks, the Flow Faucet grants a vast amount of tokens - enough gas for millions of transactions.

Deploy the Contract

Deploy the contract with:


_10
npx hardhat ignition deploy ./ignition/modules/ClickToken.ts --network flowTestnet

You should see something similar to:


_13
✔ Confirm deploy to network flowTestnet (545)? … yes
_13
Hardhat Ignition 🚀
_13
_13
Deploying [ ClickTokenModule ]
_13
_13
Batch #1
_13
Executed ClickTokenModule#ClickToken
_13
_13
[ ClickTokenModule ] successfully deployed 🚀
_13
_13
Deployed Addresses
_13
_13
ClickTokenModule#ClickToken - 0x5Ff8221DfDD1F82fd538391D231502B4b927fbD7

Verify the Contract

Next, verify the contract with:


_10
hardhat ignition verify chain-545 --include-unrelated-contracts

You'll see something similar to:


_10
briandoyle@Mac button-clicker-contract % npx hardhat ignition verify chain-545 --include-unrelated-contracts
_10
Verifying contract "contracts/ClickToken.sol:ClickToken" for network flowTestnet...
_10
Successfully verified contract "contracts/ClickToken.sol:ClickToken" for network flowTestnet:
_10
- https://evm-testnet.flowscan.io//address/0x64366c923d5046F8417Dcd8a0Cb4a789F8722387#code

Testing the Contract

Click the link to open the contract in testnet Flowscan. Click the Connect button and connect your wallet, then navigate to the Contract tab and Read/Write contract.

read write contract

Find the mintTo function and expand the UI to mint yourself a few tokens. You can click the self button to automatically add your address without needing to copy/paste.

Once you've "earned" a few tokens, use balanceOf to see how many tokens you have. You can also use getAllScores to get a list of everyone with the tokens, and how many they have.

Testing with Free Gas

If you don't have it yet, set up the Flow Wallet, connect, and try minting some more tokens. You'll see that the wallet automatically sponsors your gas:

flow-wallet

Even better, the Flow Wallet is currently sponsoring transactions on mainnet too!

Conclusion

In this tutorial, you learned how to:

  • Fund a wallet with testnet tokens from the Flow Faucet
  • Deploy a contract on Flow EVM Testnet using Hardhat
  • Interact with a contract using Flowscan
  • Utilize automatically sponsored gas with the Flow Wallet on testnet and mainnet

You've now mastered the basics of deploying and interacting with EVM contracts on Flow. But this is just the beginning! Flow EVM's true power lies in its ability to combine the best of both worlds: EVM compatibility with Flow's native features.

In our Cross-VM Apps tutorial series, you'll learn how to supercharge your EVM applications by integrating them with Flow Cadence. You'll discover how to:

  • Build hybrid applications that seamlessly connect to both Flow EVM and Flow Cadence
  • Use Cadence's powerful features to enhance your EVM contracts
  • Enable multi-call contract writes with a single signature
  • Take advantage of Flow's native features like VRF and sponsored transactions

Ready to unlock the full potential of Flow EVM? Start with our Batched Transactions tutorial to learn how to build your first cross-VM application.