Reentrancy attacks are a common vulnerability in smart contracts, where a malicious contract can call back into the original contract before the first execution is complete. This can lead to unintended consequences, such as draining funds. In this guide, we will explore how to use Hardhat to test for reentrancy attacks and ensure the security of your smart contracts.
1. Setting Up Your Hardhat Project
First, ensure that you have a Hardhat project set up. If you haven't done this yet, follow these steps:
mkdir my-hardhat-project
cd my-hardhat-project
npm init -y
npm install --save-dev hardhat
npx hardhat
Follow the prompts to create a basic sample project.
2. Writing a Vulnerable Contract
Next, create a simple vulnerable contract that we will test against reentrancy attacks. Create a new file in the contracts
directory named Vulnerable.sol
:
pragma solidity ^0.8.0;
contract Vulnerable {
mapping(address => uint256) public balances;
constructor() {
balances[msg.sender] = 1000; // Initial balance for the deployer
}
function withdraw(uint256 amount) public {
require(balances[msg.sender] >= amount, "Insufficient funds");
balances[msg.sender] -= amount;
payable(msg.sender).transfer(amount); // External call
}
receive() external payable {}
}
3. Creating a Malicious Contract
Now, let's create a malicious contract that will exploit the reentrancy vulnerability in the Vulnerable
contract. Create a new file in the contracts
directory named Attacker.sol
:
pragma solidity ^0.8.0;
import "./Vulnerable.sol";
contract Attacker {
Vulnerable public vulnerable;
constructor(address _vulnerable) {
vulnerable = Vulnerable(_vulnerable);
}
function attack() public {
vulnerable.withdraw(100); // Start the attack
}
receive() external payable {
if (address(vulnerable).balance >= 100) {
vulnerable.withdraw(100); // Re-enter the withdraw function
}
}
}
4. Writing Tests for the Vulnerability
Now that we have our vulnerable and malicious contracts, we can write tests to check for the reentrancy attack. Create a new test file in the test
directory named reentrancy-test.js
:
const { expect } = require("chai");
const { ethers } = require("hardhat");
describe("Reentrancy Attack Test", function () {
let vulnerable;
let attacker;
beforeEach(async function () {
const Vulnerable = await ethers.getContractFactory("Vulnerable");
vulnerable = await Vulnerable.deploy();
await vulnerable.deployed();
const Attacker = await ethers.getContractFactory("Attacker");
attacker = await Attacker.deploy(vulnerable.address);
await attacker.deployed();
// Fund the vulnerable contract
await vulnerable.sendTransaction({ value: ethers.utils.parseEther("1") });
});
it("should prevent reentrancy attack", async function () {
const initialBalance = await ethers.provider.getBalance(vulnerable.address);
// Try to attack
await expect(attacker.attack()).to.be.revertedWith("Insufficient funds");
const finalBalance = await ethers.provider.getBalance(vulnerable.address);
expect(finalBalance).to.equal(initialBalance); // Balance should remain unchanged
});
});
5. Running the Tests
To run the tests, execute the following command in your terminal:
npx hardhat test
This will compile your contracts and run the tests. You should see output indicating whether the tests passed or failed.
6. Mitigating Reentrancy Vulnerabilities
To mitigate reentrancy vulnerabilities, consider using the Checks-Effects-Interactions pattern or employing the ReentrancyGuard from OpenZeppelin:
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
contract Secure is ReentrancyGuard {
mapping(address => uint256) public balances;
constructor() {
balances[msg.sender] = 1000; // Initial balance for the deployer
}
function withdraw(uint256 amount) public nonReentrant {
require(balances[msg.sender] >= amount, "Insufficient funds");
balances[msg.sender] -= amount;
payable(msg.sender).transfer(amount); // External call
}
receive() external payable {}
}
Conclusion
Testing for reentrancy attacks using Hardhat is crucial for ensuring the security of your smart contracts. By setting up a vulnerable contract, creating a malicious contract, and writing tests to simulate the attack, you can identify and address vulnerabilities effectively. Always implement best practices and consider using established libraries like OpenZeppelin to enhance the security of your contracts.