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.