Skip to main content

Deploying Smart Contracts

In this article, you'll learn how to deploy smart contracts to multiple Blockchain networks using Hardhat and Hardhat deploy.


Objectives

By the end of this lesson, you should be able to:

  • Deploy a smart contract to the Base Sepolia Testnet with hardhat-deploy
  • Deploy a smart contract to the Sepolia Testnet with hardhat-deploy
  • Use BaseScan to view a deployed smart contract

Overview

Hardhat capabilities enable developers to deploy smart contracts easily to any Blockchain by simply creating tasks or scripts. However, due to the Hardhat architecture that enables its extension by creating plugins, you can rely on existing solutions developed by the community.

Hardhat deploy is a community-developed plugin that enables the deployment of your smart contracts in a simple way.

Setting up Hardhat deploy

To install:

  1. Run npm install -D hardhat-deploy. Then, import hardhat-deploy in hardhat.config.ts:
import 'hardhat-deploy';
  1. Create a folder called deploy and inside it create a new file called 001_deploy_lock.ts.

  2. Include the following:

import { HardhatRuntimeEnvironment } from 'hardhat/types';
import { DeployFunction } from 'hardhat-deploy/types';

const func: DeployFunction = async function (hre: HardhatRuntimeEnvironment) {
// code here
};
export default func;
  1. Modify the tsconfig.json file to look like:
{
"compilerOptions": {
"target": "es2020",
"module": "commonjs",
"esModuleInterop": true,
"forceConsistentCasingInFileNames": true,
"strict": true,
"skipLibCheck": true,
"resolveJsonModule": true
},
"include": ["./hardhat.config.ts", "./scripts", "./deploy", "./test"]
}
  1. Before implementing the deploy functionality, configure a deployer account in the hardhat.config.ts file. Hardhat deployment includes a way to name accounts in the config file.

  2. Run the following, which adds an alias to the account 0 of your environment:

const config: HardhatUserConfig = {
solidity: '0.8.23',
namedAccounts: {
deployer: 0,
},
};
  1. Implement the deploy function by including the following in the 001_deploy_lock.ts file:
import { HardhatRuntimeEnvironment } from 'hardhat/types';
import { DeployFunction } from 'hardhat-deploy/types';
import { ethers } from 'hardhat';

const func: DeployFunction = async function (hre: HardhatRuntimeEnvironment) {
const { deploy } = hre.deployments;
// We can now use deployer
const { deployer } = await hre.getNamedAccounts();

// The value we want to lock
const VALUE_LOCKED = hre.ethers.parseEther('0.01');

// The unlock time after deployment
const UNLOCK_TIME = 10000;

// We use ethers to get the current time stamp
const blockNumber = await ethers.provider.getBlockNumber();
const lastBlockTimeStamp = (await ethers.provider.getBlock(blockNumber))?.timestamp as number;

// We say we want to deploy our Lock contract using the deployer
// account and passing the value and arguments.
await deploy('Lock', {
from: deployer,
args: [lastBlockTimeStamp + UNLOCK_TIME],
value: VALUE_LOCKED.toString(),
});
};

export default func;

// This tag will help us in the next section to trigger this deployment file programmatically
func.tags = ['DeployAll'];

Testing your deployment

The easiest way to test your deployment is by modifying the test.

Go to Lock.ts and include in the imports the following:

import { ethers, deployments } from 'hardhat';

deployments will allow you to execute the deployment files from your test.

Change the before function to look like the following:

before(async () => {
lastBlockTimeStamp = await time.latest();

const signers = await ethers.getSigners();
ownerSigner = signers[0];
otherUserSigner = signers[1];

await deployments.fixture(['DeployAll']);
const lockDeployment = await deployments.get('Lock');

lockInstance = Lock__factory.connect(lockDeployment.address, ownerSigner);
});

Notice how you execute deployments.fixture and pass a tag that matches the one you specified in the deployment file (001_deploy_lock.ts).

The deployment file is then executed and you can then reuse that functionality and simply consume the address of the newly-deployed contract by using:

const lockDeployment = await deployments.get('Lock');

Reuse Lock__factory but use the connect function and pass the address of the newly-created contract plus a signer. Then, run npx hardhat test and you should get the same result:

  Lock
✔ should get the unlockTime value
✔ should have the right ether balance
✔ should have the right owner
✔ shouldn't allow to withdraw before unlock time (51ms)
✔ shouldn't allow to withdraw a non owner
✔ should allow to withdraw an owner

6 passing (2s)

Deploying to a test network

Deploying to a real test network involves configuring the network parameters in the hardhat config file. You need to include parameters such as:

  • The JSON RPC URL
  • The account you want to use
  • Real test ether or the native Blockchain token for gas costs

Include the following in the hardhat.config.ts file:

const config: HardhatUserConfig = {
solidity: '0.8.18',
namedAccounts: {
deployer: 0,
},
networks: {
base_sepolia: {
url: 'https://sepolia.base.org',
accounts: {
mnemonic: process.env.MNEMONIC ?? '',
},
},
sepolia: {
url: `https://eth-sepolia.g.alchemy.com/v2/${process.env.ALCHEMY_SEPOLIA_KEY ?? ''}`,
accounts: {
mnemonic: process.env.MNEMONIC ?? '',
},
},
},
};

You've configured 2 networks:

  • base_sepolia
  • sepolia

You also need to create a .env file with the following variables:

MNEMONIC="<REPLACE WITH YOUR MNEMONIC>"
ALCHEMY_SEPOLIA_KEY=<REPLACE WITH YOUR API KEY>

In order to ensure the environment variables are loaded, you need to install another package called dotenv:

npm install -D dotenv

Then, include the following in the hardhat.config.ts file:

import dotenv from 'dotenv';

dotenv.config();

Deploy to base with the following command:

npx hardhat deploy --network base_sepolia

After you run the command, a deployments folder appears with a newly-created deployment for base_sepolia:

New deployment

If you want to deploy to another network, change the network name as follows:

npx hardhat deploy --network sepolia
info

Be aware that you must have the correct environment variables for the JSON RPC URLs. For example, for Sepolia use ALCHEMY_SEPOLIA_KEY.

Conclusion

In this lesson, you've learned how to deploy smart contracts using Hardhat and Hardhat-deploy. You have configured hardhat to easily deploy to multiple networks and you created deployment files to abstract this task.


See also

Solidity Docs

We use cookies and similar technologies on our websites to enhance and tailor your experience, analyze our traffic, and for security and marketing. You can choose not to allow some type of cookies by clicking . For more information see our Cookie Policy.