Configuring Foundry with Hedera Localnet/Testnet: A Step-by-Step Guide

Developers building smart contracts on Hedera often use the Hedera JSON-RPC Relay to enable EVM tools like Foundry. In this post, we'll walk through how to set up Foundry to work with the Hiero Local Node, allowing for local deployment, debugging, and testing of smart contracts without using testnet resources.

You can take a look at the complete code in the Hedera-Code-Snippets repository.

This guide shows you how to configure Foundry to deploy, interact with, and test Solidity smart contracts on:

  • Hedera Localnet (via the Hiero Local Node + JSON-RPC Relay)

  • Hedera Testnet (via Hashio JSON-RPC Relay)

You’ll set up environment variables, configure foundry.toml, write a simple contract and a Foundry deployment script, and learn how to switch between Localnet and Testnet with a single flag.

If you’re looking for an end-to-end “Hello, ERC-20” tutorial, see: “Getting started with Foundry” (deploy, interact, verify on Hashscan). This guide focuses on environment configuration and workflow for Localnet/Testnet.


What you will accomplish

  • Configure Foundry to talk to the Hedera JSON-RPC Relay (Localnet and Testnet)

  • Deploy and interact with a sample contract using forge script and cast

  • Verify contracts on Hashscan (Testnet)


Prerequisites

  • Foundry installed (forge, cast, anvil, chisel):

    • curl -L https://foundry.paradigm.xyz | bash

    • foundryup

  • ECDSA account and private key

  • Basic Solidity / CLI familiarity


Table of Contents


Option A: Run Hedera Localnet (Hiero Local Node)

Hedera provides a local node configuration that includes a mirror node, a consensus node, and the JSON-RPC relay. You can run it via npm.

Clone, install, and run the node:

git clone https://github.com/hiero-ledger/hiero-local-node.git
cd hiero-local-node
npm install
npm run start

Once all the containers have started, the Hiero Local Node is up and running. This includes a Consensus Node, Mirror Node and explorer, JSON RPC Relay, Block Node, Grafana UI, and Prometheus UI.

  • On startup, the Local Node prints funded ECDSA accounts to the console logs. You can use one of these for local development. Make sure to only use accounts listed under "Accounts list (Alias ECDSA keys)"

  • The JSON-RPC Relay is typically on port 7546. So, the URL would be http://localhost:7546


Option B: Use Hedera Testnet (Hashio)

Hashio is a community relay node suitable for development/testing:

  • RPC URL: https://testnet.hashio.io/api

Note: For production, prefer a commercial-grade JSON-RPC relay or host your own Hiero JSON-RPC Relay.


Step 1: Initialize a Foundry project

forge init foundry-hello-world
cd foundry-hello-world

This creates src, script, test, and lib.

Project Structure

The Foundry project initialization creates the following file structure:

foundry.toml
lib
└── forge-std
src
└── Counter.sol
script
└── Counter.s.sol
test
└── Counter.t.sol

Here's a quick overview of these files and directories:

  • foundry.toml: You can configure Foundry's behavior using this file such as defining RPC URLs

  • src: Serves as the default for storing your smart contract source code

  • script: This is where you store Solidity scripts for deploying contracts and performing other on-chain operations

  • test: Serves as the dedicated location for Solidity-based unit and integration tests for your smart contracts


Step 2: Add environment variables

Create an .env for your RPC URL and private key.

touch .env

Put the following into your environment file.

.env
HEDERA_RPC_URL=your-rpc-url
HEDERA_PRIVATE_KEY=0x-your-private-key

Now, let's also load these to the terminal:

source .env

Step 3: Configure "foundry.toml"

Foundry uses the foundry.toml file for configuration. Open it and add profiles for the Hedera RPC endpoint.

foundry.toml
[profile.default]
src = "src"
out = "out"
libs = ["lib"]
remappings = [
  "forge-std/=lib/forge-std/src/"
]

# Add this section for Hedera Testnet
[rpc_endpoints]
hedera = "${HEDERA_RPC_URL}"

Step 4: Add a simple contract

Use a tiny Counter contract to focus on configuration rather than external dependencies for this exercise.

Compile with:

forge build

Step 5: Update the deploy script

We are going to update our deploy script a little bit so we can use our private key instead of passing it as a flag every time:

script/Counter.s.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.28;

import {Script, console} from "forge-std/Script.sol";
import {Counter} from "../src/Counter.sol";

contract CounterScript is Script {
    Counter public counter;

    function run() external returns (address) {
        // Load the private key from the .env file
        uint256 deployerPrivateKey = vm.envUint("HEDERA_PRIVATE_KEY");

        // Start broadcasting transactions with the loaded private key
        vm.startBroadcast(deployerPrivateKey);

        // Deploy the contract
        counter = new Counter();

        // Stop broadcasting
        vm.stopBroadcast();

        console.log("Counter Contract deployed to:", address(counter));

        return address(counter);
    }
}

Step 6: Deploy the contract

Now, execute the script to deploy your contract:

forge script script/Counter.s.sol:CounterScript --rpc-url hedera --broadcast

After a few moments, you will see the address of your newly deployed contract:

[⠊] Compiling...
No files changed, compilation skipped
Script ran successfully.

== Return ==
0: address 0x061A1DdE963792192eA823C2F57285111812630b

== Logs ==
  Counter Contract deployed to: 0x061A1DdE963792192eA823C2F57285111812630b

## Setting up 1 EVM.

==========================

Chain 296

Estimated gas price: 720.000000001 gwei

Estimated total gas used for script: 203856

Estimated amount required: 0.146776320000203856 ETH

==========================

##### 296
✅  [Success] Hash: 0x279d667c3a31bb0956449d819bdd4b99619a25b8387a0765f28c9407d158069b
Contract Address: 0x061A1DdE963792192eA823C2F57285111812630b
Block: 25089799
Paid: 0.0554489 ETH (163085 gas * 340 gwei)

✅ Sequence #1 on 296 | Total Paid: 0.0554489 ETH (163085 gas * avg 340 gwei)
                                                                                                             

==========================

ONCHAIN EXECUTION COMPLETE & SUCCESSFUL

Now, go ahead and update your .env values to point to another Hedera Network(Localnet or Testnet) and try again.


Step 7: Run tests

You can also run the test suite that's included as part of test/ directory:

forge test

Step 8: Verifying the Contract on Hashscan

Verifying your smart contract makes its source code publicly available on Hashscan.

Note: Please note that verification of the contract can not be done on Localnet and this process is relevant only for Hedera Testnet/Mainnet. Also, make sure to replace <your-contract-address> with your own contract address you got after the deployment above.

Run the following command, using the variables you set earlier.

forge verify-contract <your-contract-address> src/Counter.sol:Counter \
    --chain-id 296 \
    --verifier sourcify \
    --verifier-url "https://server-verify.hashscan.io/"

After running the command, you should see a success message.

Submitting verification for [Counter] "0x061A1DdE963792192eA823C2F57285111812630b".
Contract successfully verified

Congratulations! 🎉 You have successfully deployed, interacted with, and verified a smart contract on the Hedera Testnet using Foundry. Feel free to reach out in Discord!


Interact using cast

Set helpers:

export CONTRACT_ADDRESS=<your-contract-address>
export MY_ADDRESS=$(cast wallet address $HEDERA_PRIVATE_KEY)

Read:

cast call $CONTRACT_ADDRESS "number()(uint256)" --rpc-url hedera

Write:

cast send $CONTRACT_ADDRESS "setNumber(uint256)" 42 \
  --private-key $HEDERA_PRIVATE_KEY \
  --rpc-url hedera

cast send $CONTRACT_ADDRESS "increment()" \
  --private-key $HEDERA_PRIVATE_KEY \
  --rpc-url hedera

Read again:

cast call $CONTRACT_ADDRESS "number()(uint256)" --rpc-url hedera

You should get an output of 43 since the counter is at 43 now after having added 42 + 1.


Further Learning & Next Steps

Want to take your local development setup even further? Here are some excellent tutorials to help you dive deeper into smart contract development on Hedera using Foundry:

  1. How to Mint and Burn an ERC-721 Token (Part 1) Learn how to create a basic ERC-721 NFT, mint it, and burn it on Hedera.

  2. How to Write Tests in Solidity (Part 2) Learn how to start writing tests in Foundry using Solidity

  3. How to Fork the Hedera Network for Local Testing Learn how to fork hedera network(testnet/mainnet) locally so you can start testing against the forked network

Last updated

Was this helpful?