Writing tests for your smart contracts is essential to ensure their functionality and security. Hardhat provides a robust framework for testing, allowing you to write automated tests using JavaScript or TypeScript.
Setting Up Your Hardhat Project
- First, create a new directory for your project and navigate into it:
mkdir my-hardhat-project
cd my-hardhat-project
npm init -y
npm install --save-dev hardhat
npx hardhat
Select "Create an empty hardhat.config.js" when prompted.
Creating Your Smart Contract
In the contracts
directory, create a new Solidity file, for example, Token.sol
, and define your smart contract:
pragma solidity ^0.8.0;
contract Token {
string public name = "MyToken";
string public symbol = "MTK";
uint256 public totalSupply;
mapping(address => uint256) public balanceOf;
constructor(uint256 _initialSupply) {
totalSupply = _initialSupply;
balanceOf[msg.sender] = _initialSupply;
}
function transfer(address _to, uint256 _amount) public {
require(balanceOf[msg.sender] >= _amount, "Not enough tokens");
balanceOf[msg.sender] -= _amount;
balanceOf[_to] += _amount;
}
}
Writing Tests for Your Smart Contract
Next, create a test
directory in your project root and add a new file called Token.js
to write your tests:
const { expect } = require("chai");
const { ethers } = require("hardhat");
describe("Token contract", function () {
let hardhatToken;
let owner;
let addr1;
let addr2;
beforeEach(async function () {
const Token = await ethers.getContractFactory("Token");
hardhatToken = await Token.deploy(1000);
[owner, addr1, addr2] = await ethers.getSigners();
});
it("Should assign the total supply of tokens to the owner", async function () {
const ownerBalance = await hardhatToken.balanceOf(owner.address);
expect(await hardhatToken.totalSupply()).to.equal(ownerBalance);
});
it("Should transfer tokens between accounts", async function () {
await hardhatToken.transfer(addr1.address, 50);
expect(await hardhatToken.balanceOf(addr1.address)).to.equal(50);
await hardhatToken.connect(addr1).transfer(addr2.address, 50);
expect(await hardhatToken.balanceOf(addr2.address)).to.equal(50);
});
it("Should fail if sender doesn't have enough tokens", async function () {
const initialOwnerBalance = await hardhatToken.balanceOf(owner.address);
await expect(hardhatToken.connect(addr1).transfer(owner.address, 1)).to.be.revertedWith("Not enough tokens");
expect(await hardhatToken.balanceOf(owner.address)).to.equal(initialOwnerBalance);
});
});
Running Your Tests
To run your tests, execute the following command in your terminal:
npx hardhat test
You should see output indicating whether your tests passed or failed. A successful output will look something like this:
Token contract
✓ Should assign the total supply of tokens to the owner
✓ Should transfer tokens between accounts
✓ Should fail if sender doesn't have enough tokens
3 passing (Xms)
Conclusion
Testing your smart contracts in Hardhat is straightforward and essential for ensuring their reliability. By following the steps outlined above, you can create a robust testing suite for your contracts.