Testing cross-contract interactions is crucial in ensuring that your smart contracts work together as intended. In this guide, we will explore how to set up and execute tests for contracts that interact with each other using Truffle.
1. Set Up Your Truffle Project
First, ensure you have a Truffle project set up. If you haven't created one yet, you can initialize a new Truffle project by running the following commands:
mkdir MyTruffleProject
cd MyTruffleProject
truffle init
2. Create Your Contracts
For this example, let's create two contracts: Token
and TokenSale
. The TokenSale
contract will interact with the Token
contract.
Token Contract:
pragma solidity ^0.8.0;
contract Token {
string public name = "MyToken";
mapping(address => uint256) public balances;
function mint(address to, uint256 amount) public {
balances[to] += amount;
}
function transfer(address to, uint256 amount) public {
require(balances[msg.sender] >= amount, "Insufficient balance");
balances[msg.sender] -= amount;
balances[to] += amount;
}
}
TokenSale Contract:
pragma solidity ^0.8.0;
import "./Token.sol";
contract TokenSale {
Token public token;
uint256 public price;
constructor(Token _token, uint256 _price) {
token = _token;
price = _price;
}
function buyTokens(uint256 amount) public payable {
require(msg.value == amount * price, "Incorrect Ether value");
token.mint(msg.sender, amount);
}
}
3. Write Tests for Cross-Contract Interactions
Next, we will write tests to verify the interactions between the Token
and TokenSale
contracts. Create a new file in the test
directory called TokenSale.test.js
.
Example Test Code:
const Token = artifacts.require("Token");
const TokenSale = artifacts.require("TokenSale");
contract("TokenSale", (accounts) => {
let token;
let tokenSale;
before(async () => {
token = await Token.new();
tokenSale = await TokenSale.new(token.address, web3.utils.toWei("0.01", "ether"));
});
it("should mint tokens when purchased", async () => {
await tokenSale.buyTokens(10, { from: accounts[1], value: web3.utils.toWei("0.1", "ether") });
const balance = await token.balances(accounts[1]);
assert.equal(balance.toString(), "10", "Tokens were not minted correctly");
});
it("should not allow purchasing with incorrect Ether value", async () => {
try {
await tokenSale.buyTokens(10, { from: accounts[1], value: web3.utils.toWei("0.05", "ether") });
assert.fail("Expected error not received");
} catch (error) {
assert(error.message.includes("Incorrect Ether value"), "Error message does not match");
}
});
it("should allow token transfer between accounts", async () => {
await token.transfer(accounts[2], 5, { from: accounts[1] });
const balance1 = await token.balances(accounts[1]);
const balance2 = await token.balances(accounts[2]);
assert.equal(balance1.toString(), "5", "Sender's balance is incorrect after transfer");
assert.equal(balance2.toString(), "5", "Receiver's balance is incorrect after transfer");
});
});
4. Run Your Tests
To execute your tests, run the following command in your terminal:
truffle test
This will compile your contracts and run the tests you have written, providing feedback on the success or failure of each test case.
Conclusion
Testing cross-contract interactions in Truffle is essential for ensuring that your smart contracts function correctly together. By following the steps outlined in this guide, you can create contracts, write tests, and verify that interactions between contracts behave as expected.