Day 21. hardhat – configuration (hardhat.config.ts) –

The configuration of Hardhat is defined in a file named hardhat.config.ts. This file includes settings for networks, the Solidity compiler, accounts, and other aspects relevant to Hardhat’s operation.

TOC

Hardhat Configuration

The following is an example of hardhat.config.ts used in this guide.

TypeScript
/// 1. Import necessary dependency libraries.
import "@typechain/hardhat"            // Core hardhat library
import "@nomiclabs/hardhat-waffle"     // Library for writing tests, used along with chai
import "@nomiclabs/hardhat-etherscan"  // Library for uploading and verifying source code on Etherscan after deployment
import "@nomiclabs/hardhat-ethers"     // Library to facilitate testing and deployment of contracts. Example usage: ethers object in deployment scripts
import "hardhat-gas-reporter"          // Library to visualize the gas consumption of contracts
import "dotenv/config"                 // Library to easily use environment variables defined in .env
import "hardhat-deploy"                // Library used for deploying. Example usage: deploy function and getNamedAccounts function in deployment scripts。
import "solidity-coverage"             // Library to visualize test coverage
import { HardhatUserConfig } from "hardhat/config"

/// 2. Definition of Constants (Defined as Environment Variables in External Files)
// Below, environment variables are retrieved as constants. These constants are used in the settings (HardhatUserConfig).
const SEPOLIA_RPC_URL = process.env.SEPOLIA_RPC_URL || "https://eth-sepolia.g.alchemy.com/v2/YOUR-API-KEY"
const PRIVATE_KEY = process.env.PRIVATE_KEY || "0x123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0"
const ETHERSCAN_API_KEY = process.env.ETHERSCAN_API_KEY || ""
const COINMARKETCAP_API_KEY = process.env.COINMARKETCAP_API_KEY || ""

/// 3. HardhatUserConfig. Actual Settings
const config: HardhatUserConfig = {
    defaultNetwork: "hardhat",
    networks: { // Define the environments to connect to
        hardhat: {
            chainId: 31337,
            // gasPrice: 130000000000,
        },
        sepolia: {
            url: SEPOLIA_RPC_URL,
            accounts: [PRIVATE_KEY],
            chainId: 11155111,
        },
    },
    solidity: { // Define the version of the Solidity compiler
        compilers: [
            {
                version: "0.8.8",
            },
            {
                version: "0.6.6",
            },
        ],
    },
    etherscan: { // Define the API key for requesting source code verification (verify) on Etherscan
        apiKey: ETHERSCAN_API_KEY,
    },
    gasReporter: { // Gas reporter (a tool for visualizing gas costs)
        enabled: true,
        currency: "USD",
        outputFile: "gas-report.txt",
        noColors: true,
        coinmarketcap: COINMARKETCAP_API_KEY,
    },
    namedAccounts: { // Define account information (such as your wallet information) used in deployment and testing
        deployer: {
            default: 0, // here this will by default take the first account as deployer
            1: 0, // similarly on mainnet it will take the first account as deployer. Note though that depending on how hardhat network are configured, the account 0 on one network can be different than on another
        },
        user:{
            default: 1, 
        },
    },
}

export default config

The configuration file primarily consists of three parts, and the third section will be discussed further. The details for sections one and two are as mentioned in the comments. In section two, constants are defined and then used in specific settings. Alternatively, settings can directly embed values, for example, apiKey:'${process.env.ETHERSCAN_API_KEY}'.

  1. Section to import necessary libraries
  2. Section defining constants (loading environment variables as constants)
  3. Section with actual settings

HardhatUserConfig

Networks

The network settings example is as follows.

TypeScript
    defaultNetwork: "hardhat",
    networks: { // Define the environments connecting to
        hardhat: {
            chainId: 31337,
            // gasPrice: 130000000000,
        },
        sepolia: {
            url: SEPOLIA_RPC_URL,
            accounts: [PRIVATE_KEY],
            chainId: 11155111,
        },
    },

This defines the environment for deploying and executing scripts. There are blockchain networks such as mainnet (production), testnet (test, Sepolia network in the example), and hardhat (local virtual blockchain network). Each network has an identifier called chainId, which is specified accordingly: 31337 for localnet (hardhat), 11155111 for Sepolia, and 1 for mainnet.

In the accounts section, the private key from a local digital wallet like Metamask is configured (strictly speaking, it’s defined in the .env file and read as a constant).

To issue transactions to the blockchain network (such as deploying a smart contract), it’s necessary to connect from the local terminal to the blockchain network. This is done through nodes called RPC (Remote Procedure Call). Imagine it as a communication path connecting the local and blockchain networks. More specifically, it uses JSON-RPC, exchanging information in JSON format. Infrastructure services providing such RPC services include Alchemy, Infura, QuickNode, Moralis, etc. The environment variable SEPULIA_RPC_URL is an individual RPC node created through Alchemy’s service.

To set up an RPC node, follow these steps on Alchemy’s website.

Select Apps -> Create new app.
Select Apps -> Create new app.
Enter Name and Description, choose Network, and click Create App.
Click on API Key.
Confirm the HTTPS URL of the created node.

solidity

TypeScript
    solidity: { // Define the version of the Solidity compiler
        compilers: [
            {
                version: "0.8.8",
            },
            {
                version: "0.6.6",
            },
        ],
    },

In this section, the version of Solidity (compiler version) is specified. It is mentioned in conjunction with the version specified in the actual Solidity source.

etherscan

TypeScript
    etherscan: { // Define the API key for requesting source code verification (verify) on Etherscan
        apiKey: ETHERSCAN_API_KEY,
    },

This section involves setting up the Etherscan API key. Etherscan, as explained in Day 6, is a block explorer service used to view information on Ethereum.

When smart contracts are deployed to the blockchain, they are stored as bytecode. However, it is not possible to directly obtain the original Solidity source code from the bytecode. This is due to the difficulty of reverse-compiling the code, and even if possible, the original source code’s comments and variable names cannot be restored.

Therefore, to maintain the transparency of smart contracts, developers need to publish the source code and verify (prove) that the deployed bytecode was compiled from that source code. This allows others to verify the contract’s operations and assess its reliability.

Etherscan provides a feature to verify contracts, and the Etherscan API key is registered in the deployment script to automate verification in the script (the API key enables automated verification within the script).

Obtaining an Etherscan API key can be done by creating an account on Etherscan and then generating it via API Keys > +Add, as shown in the diagram.

gas-reporter

TypeScript
    gasReporter: { // Gas reporter (a tool for visualizing gas costs)
        enabled: true,
        currency: "USD",
        outputFile: "gas-report.txt",
        noColors: true,
        coinmarketcap: COINMARKETCAP_API_KEY,
    },

This section holds the settings for gas-reporter. gas-reporter is a tool for reporting the gas consumption of smart contracts. It measures the gas usage of each function and transaction during Hardhat tests and generates a consolidated report.

The following image is an actual example of a gas-reporter run. It lists gas values (minimum, maximum, average) for each function. Such visualization helps identify which parts of the code consume more gas, aiding in improvements towards lower consumption (detailed in Day 25).

namedAccounts

TypeScript
    namedAccounts: { // Define account information (such as your wallet information) used in deployment and testing
        deployer: {
            default: 0, 
            1: 0, 
        },
        user:{
            default: 1, 
        },

In this section, you can define accounts to be used in deployment and test scripts.

In the example above, an account named deployer is defined. It specifies that by default, the 0th account from the accounts list is used, and for chain ID 1 (mainnet), the 0th account is used as well. Instead of the number 1, network names like ‘mainnet’ can also be used. Similarly, an account named ‘user’ is defined, setting the default as the 0th account.

The accounts list, as mentioned in the networks section, is set for each blockchain network and is generated from private keys or mnemonics.

Let's share this post !

Author of this article

After joining IBM in 2004, the author gained extensive experience in developing and maintaining distributed systems, primarily web-based, as an engineer and PM. Later, he founded his own company, designing and developing mobile applications and backend services. He is currently leading a Tech team at a venture company specializing in robo-advisory.

Comments

To comment

CAPTCHA


TOC