Skip to main content

Hardhat: Analyzing the test coverage of smart contracts

In this tutorial, you'll learn how to profile the test coverage of your smart contracts with Hardhat and the Solidity Coverage community plugin.


Objectives

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

  • Use the Solidity Coverage plugin to analyze the coverage of your test suite
  • Increase the coverage of your test suite

Overview

The Solidity Coverage plugin allows you to analyze and visualize the coverage of your smart contracts' test suite. This enables you to see what portions of your smart contract are being tested and what areas may have been overlooked. It's an indispensable plugin for developers seeking to fortify their testing practices and ensure robust smart contract functionality.

Setting up the Solidity Coverage plugin

The Solidity Coverage plugin is integrated into the Hardhat toolbox package, which is installed by default when you use the npx hardhat init command.

To install manually, run npm install -D solidity-coverage.

Then, import solidity-coverage in hardhat.config.ts:

import "solidity-coverage"

Once the installation completes either manually or via the default Hardhat template, the task coverage becomes available via the npx hardhat coverage command.

My first test coverage

Review the following contract and test suite (You'll recognize these if you completed the Hardhat testing lesson in our Base Learn series).

Contract:

// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.19;

contract Lock {
uint public unlockTime;
address payable public owner;

event Withdrawal(uint amount, uint when);

constructor(uint _unlockTime) payable {
require(
block.timestamp < _unlockTime,
"Unlock time should be in the future"
);

unlockTime = _unlockTime;
owner = payable(msg.sender);
}

function withdraw() public {
require(block.timestamp >= unlockTime, "You can't withdraw yet");
require(msg.sender == owner, "You aren't the owner");

emit Withdrawal(address(this).balance, block.timestamp);

owner.transfer(address(this).balance);
}
}

Lock.test.ts:

import { expect } from "chai";
import { ethers } from "hardhat";
import { time } from "@nomicfoundation/hardhat-network-helpers";
import { SignerWithAddress } from '@nomicfoundation/hardhat-ethers/signers'
import { Lock__factory, Lock} from '../typechain-types'

describe("Lock Tests", function () {
const UNLOCK_TIME = 10000;
const VALUE_LOCKED = ethers.parseEther("0.01");

let lastBlockTimeStamp: number;
let lockInstance: Lock;
let ownerSigner: SignerWithAddress
let otherUserSigner: SignerWithAddress;

before(async() => {
lastBlockTimeStamp = await time.latest()
const signers = await ethers.getSigners()
ownerSigner = signers[0]
otherUserSigner= signers[1]

const unlockTime = lastBlockTimeStamp + UNLOCK_TIME;

lockInstance = await new Lock__factory(ownerSigner).deploy(unlockTime, {
value: VALUE_LOCKED
})
})


it('should get the unlockTime value', async() => {
const unlockTime = await lockInstance.unlockTime();

expect(unlockTime).to.equal(lastBlockTimeStamp + UNLOCK_TIME)
})

it('should have the right ether balance', async() => {
const lockInstanceAddress = await lockInstance.getAddress()

const contractBalance = await ethers.provider.getBalance(lockInstanceAddress)

expect(contractBalance).to.equal(VALUE_LOCKED)
})

it('should have the right owner', async()=> {
expect(await lockInstance.owner()).to.equal(ownerSigner.address)
})

it('should not allow to withdraw before unlock time', async()=> {
await expect(lockInstance.withdraw()).to.be.revertedWith("You can't withdraw yet")
})

it('should not allow to withdraw a non owner', async()=> {
const newLastBlockTimeStamp = await time.latest()

await time.setNextBlockTimestamp(newLastBlockTimeStamp + UNLOCK_TIME)

const newInstanceUsingAnotherSigner = lockInstance.connect(otherUserSigner)

await expect(newInstanceUsingAnotherSigner.withdraw()).to.be.revertedWith("You aren't the owner")
})

it('should allow to withdraw an owner', async()=> {
const balanceBefore = await ethers.provider.getBalance(await lockInstance.getAddress());

expect(balanceBefore).to.equal(VALUE_LOCKED)

const newLastBlockTimeStamp = await time.latest()

await time.setNextBlockTimestamp(newLastBlockTimeStamp + UNLOCK_TIME)

await lockInstance.withdraw();

const balanceAfter = await ethers.provider.getBalance(await lockInstance.getAddress());
expect(balanceAfter).to.equal(0)
})
});

If you run npx hardhat coverage, you should get:

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


6 passing (195ms)

------------|----------|----------|----------|----------|----------------|
File | % Stmts | % Branch | % Funcs | % Lines |Uncovered Lines |
------------|----------|----------|----------|----------|----------------|
contracts/ | 100 | 83.33 | 100 | 100 | |
Lock.sol | 100 | 83.33 | 100 | 100 | |
------------|----------|----------|----------|----------|----------------|
All files | 100 | 83.33 | 100 | 100 | |
------------|----------|----------|----------|----------|----------------|

Which then gives you a report of the test coverage of your test suite. Notice there is a new folder called coverage, which was generated by the solidity-coverage plugin. Inside the coverage folder there is a index.html file. Open it in a browser, you'll see a report similar to the following:

Coverage report

Increasing test coverage

Although the coverage of the previous test suite is almost perfect, there is one missing branch when creating the contract. Because you have not tested the condition that the _unlockTime has to be greater than the block.timestamp:

require(
block.timestamp < _unlockTime,
"Unlock time should be in the future"
);

In order to increase the coverage, include a new test with the following:

  it('should verify the unlock time to be in the future', async () => {
const newLockInstance = new Lock__factory(ownerSigner).deploy(lastBlockTimeStamp, {
value: VALUE_LOCKED
})

await expect(newLockInstance).to.be.revertedWith("Unlock time should be in the future")
})

Then, run npx hardhat coverage and you should get:

  Lock Tests
✔ should verify the unlock time to be in the future (39ms)
✔ should get the unlockTime value
✔ should have the right ether balance
✔ should have the right owner
✔ shouldn't allow to withdraw before unlock time
✔ shouldn't allow to withdraw a non owner
✔ should allow to withdraw an owner


7 passing (198ms)

------------|----------|----------|----------|----------|----------------|
File | % Stmts | % Branch | % Funcs | % Lines |Uncovered Lines |
------------|----------|----------|----------|----------|----------------|
contracts/ | 100 | 100 | 100 | 100 | |
Lock.sol | 100 | 100 | 100 | 100 | |
------------|----------|----------|----------|----------|----------------|
All files | 100 | 100 | 100 | 100 | |
------------|----------|----------|----------|----------|----------------|

Conclusion

In this tutorial, you've learned how to profile and analyze the test coverage of your smart contracts' test suite. You learned how to visualize the coverage report and improve the coverage of the test suite by using the Solidity Coverage plugin.


See also

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.