Testing the Solidity Smart Contract, step-by-step
Truffle comes with a framework to automate tests, based on Mocha.
We will write a javascript test file that accesses our lottery contract, and interacts with it over an Ethereum local test network, that runs every time we run our tests.
First create a new file under the test directory, by right-clicking it, selecting new, and then file.
data:image/s3,"s3://crabby-images/6b5e5/6b5e5496a901ff36a45fbc6296c87c7579519825" alt="Screenshot from 2018-12-14 14-58-08 (1).png 803"
We will name it lottery.js
data:image/s3,"s3://crabby-images/11680/11680166120f5a0d40ac7d64f76157e976fd25a4" alt="Screenshot from 2018-12-14 15-00-33 (1).png 502"
At the beginning of our lottery.js test file, we will use the method:
artifacts.require(“ContractName”), this method returns an abstraction of our contract (ABI), that was created in the compilation process.
We assign the object returned by this method to a constant called Lottery, that we can use to interact with the contract.
data:image/s3,"s3://crabby-images/78c4b/78c4b50359122bc521fc7dd2e05a24c128d46c94" alt="Screenshot from 2018-12-14 15-10-06.png 629"
We then initialize a variable lottery, that we will later assign to an instance of the Lottery contract.
data:image/s3,"s3://crabby-images/33ec1/33ec13e69fb0cb4ccf287f3f8f66e5146dd4f5a5" alt="Screenshot from 2018-12-14 15-15-33.png 152"
This is a good time, to explain 3 functions generally used when testing with truffle.
Function | What it does |
---|---|
before() | The before function runs before the testing begins and it can be used to set the adequate variables to be used in each test. |
contract() | It works as describe() in mocha, and it names and groups a set of tests under a common group. It includes a block of it() functions. |
it() | It includes single units of tests, focused on testing specific aspects in a contract. |
Back in our test file, we will instantiate our contract, like this:
lottery = await Lottery.new()
This returns a promise, so we need to wrap it in an asynchronous function that allows that.
In this case, we want to make use of the before function, to instantiate the contract before all the tests are run.
data:image/s3,"s3://crabby-images/fa026/fa02694b3afba7ebaeb8e7137800ab237b89edca" alt="Screenshot from 2018-12-14 15-36-51.png 474"
Now we want to will create a contract() function named Lottery Tests that will group all the test functions it() under a themed block.
data:image/s3,"s3://crabby-images/a36b6/a36b6fe89df200e56e5ee0c214b531f7c23bf5d9" alt="Screenshot from 2018-12-14 15-41-25.png 567"
Note that we are also passing an object called accounts.
This is an array of ethereum addresses, generated by Truffle.
We can reference them to interact with our contract.
- After creating our testing block, we can begin the creation of our first test.
When writing our contract we created a variable of type address called manager, that would be assigned to the address of the deployer of the contract:
data:image/s3,"s3://crabby-images/e4cd8/e4cd83aebec4993f3c79cc0d33ebf4c50b12e96e" alt="Screenshot from 2018-12-06 16-14-31 (1).png 299"
We should test this behavior using the it() function.
First we want to call the instance of our contract and read, the address of the variable manager. To do so we use the syntax <contract_instance> .<public_variable>.call() as such:
When Truffle, deploys a contract, it uses the first address from the accounts array as default.
So the deployer address should be:
We can check if the manager_address and deployer_address are the same, by using the method assert.equal()
This method takes 2 parameters, to be compared for equality, and a third optional string parameter, to be shown in case the test fails.
We can then assert if both addresses are equal to each other:
data:image/s3,"s3://crabby-images/e1361/e1361ee0df2da8554bbefe34393e3bac8aaf5b4f" alt="Screenshot from 2018-12-14 16-14-46.png 887"
Our first test should look like this:
data:image/s3,"s3://crabby-images/a0f68/a0f68a840eacb85215502e767347b90609a2382c" alt="unnamed (7).png 512"
- Another test we should write is one that checks the process of a player entering the lottery.
data:image/s3,"s3://crabby-images/1103f/1103f9c71876a74508ace7fa864288e40c2819d5" alt="Screenshot from 2018-12-14 16-32-35.png 585"
The new_player to enter will be selected from the 2nd element of the accounts array.
data:image/s3,"s3://crabby-images/dc0f6/dc0f652504ca8559006d1fb9f767e58242ba19e1" alt="Screenshot from 2018-12-14 16-32-43.png 370"
Back in our lottery.sol contract. We created the enter function, that required, the new player, to send more than 0.1 ether to enter the lottery.
data:image/s3,"s3://crabby-images/399f2/399f2a3fdbd606d4b8b912ed04e69aea55e2cf18" alt="Screenshot from 2018-12-07 16-15-45 (1).png 573"
Now in our lottery.js test file, we can trigger this function directly sending an object with 2 parameters in it:
data:image/s3,"s3://crabby-images/7c52b/7c52b83f0cb108a0ca7201fd5fb90a3fc9f36e48" alt="Screenshot from 2018-12-14 16-32-54.png 763"
- from: which is the address creating the transaction
- value: a string representing value in wei
- Wei is the minimum unit of ether; 1 eth = 1e18 in scientific notation; 0.11 eth = 11e16 we
We then call the players(array of addresses), with a string ‘0’, representing the first object of the array of players.
data:image/s3,"s3://crabby-images/1e399/1e3994fd99922b4ba52476ac043c7451c6c029b4" alt="Screenshot from 2018-12-14 16-49-37.png 547"
Finally, we assert that the player variable is equal to new_player:
data:image/s3,"s3://crabby-images/a8937/a89373b332c1c449bf062d8487ff4eb75c7dfce0" alt="Screenshot from 2018-12-14 16-49-47.png 789"
This is how our new test should look:
data:image/s3,"s3://crabby-images/a9af7/a9af724f337237a40439ae3a782bbf98b1d86c79" alt="Screenshot from 2018-12-14 16-50-09.png 862"
- We could test if the lottery has a balance, now that the player entered the contest.
data:image/s3,"s3://crabby-images/ee467/ee46787a64921876d2ce59b4f704bcbb4e77b8b3" alt="Screenshot from 2018-12-14 16-58-12.png 501"
First, we must find out, what is the address that references our contract in our local-test-ethereum network. We can do it by reading the address property of the instance of our lottery contract.
In order to find out the balance of our lottery contract, we make use of the method web3.eth.getBalance(), and we pass the lottery.address as an attribute.
This method will return a BigNumber, which should be transformed to .toNumber(), to give the balance available in our lottery contract in Wei.
data:image/s3,"s3://crabby-images/df48a/df48a03f66f307690aeabf7555db3a9679f2efc3" alt="Screenshot from 2018-12-15 01-36-20.png 815"
Finally we, use the method web3.toWei(), to calculate the amount of wei equivalent to 0.11 ether. Which we sent with our player from accounts[1] .
We assert that both the balance and the 0.11 Ether in wei are equal.
data:image/s3,"s3://crabby-images/4622e/4622e51da6518dec0cff153cdb7a68b16b4a49cf" alt="Screenshot from 2018-12-15 01-36-26.png 597"
The test should look like this:
data:image/s3,"s3://crabby-images/48672/48672dd679b1e27895d94eb9101a02ca58f34be1" alt="Screenshot from 2018-12-15 01-36-49.png 883"
- We should test that a player cannot call the selectWinner function.
According to our contract, the selectWinner function uses the modifier managerOnly, which restricts any address, but that belonging to the manager, from triggering this function.
data:image/s3,"s3://crabby-images/7c753/7c7536999b86931614c4d645c79de1196aeca6d9" alt="Screenshot from 2018-12-07 16-14-09 (1).png 490"
To test this, we can trigger this function, using the address from one of the players accounts[1].
When the function runs, the EVM will throw an error. And we will assert for the existence of this error using a try - catch javascript syntax.
data:image/s3,"s3://crabby-images/e17d8/e17d8ecb1fed5d7533a753f2b12a39f4ba89aace" alt="Screenshot from 2018-12-14 17-01-48.png 663"
data:image/s3,"s3://crabby-images/c1daa/c1daafe79c1d3d8cffbe9768200e3dd092d6d9b1" alt="Screenshot from 2018-12-14 17-01-20.png 787"
The final test looks as such:
- We should test that the manager, can call the selectWinner function.
data:image/s3,"s3://crabby-images/73081/73081843646455b84ab7a869741574df25e2cac4" alt="Screenshot from 2018-12-14 16-59-54.png 614"
First we read the current balance of the contract and assign it to the balance variable
data:image/s3,"s3://crabby-images/0711a/0711a1312e8f5bf3360d001f4bf529c1272f2e0a" alt="Screenshot from 2018-12-14 17-00-07.png 749"
We then call the selectWinner function with the address of the manager
data:image/s3,"s3://crabby-images/a4f45/a4f45dd558c53ad20bbd5b232a03a934e8d2ec03" alt="Screenshot from 2018-12-14 17-00-01.png 594"
We finally assert, that the contract balance is ‘0’
data:image/s3,"s3://crabby-images/7ba36/7ba36ff67c2f4883706912e54ba10e0eadeac41e" alt="Screenshot from 2018-12-14 17-00-12.png 334"
The last test should look like this:
data:image/s3,"s3://crabby-images/f86cf/f86cf48163011ad941d29d451f5da01ca8591048" alt="Screenshot from 2018-12-14 16-59-34.png 845"
The final lottery.js file, with all the tests included, should be like this:
data:image/s3,"s3://crabby-images/fe48d/fe48d51e4a20ddcea7527ea904aaa50fe0ccffcb" alt="2018-12-24.png 813"
Running the Tests
While in our terminal, and making sure we are located in our lotteryapp directory, let’s run the next command to start our tests.
$ truffle test
If all the test pass successfully, we should see an output similar to:
data:image/s3,"s3://crabby-images/06894/06894475ac65e56bed44f0fb382364128a8f427b" alt="Screenshot from 2018-12-15 02-01-07 (1).png 562"
Now that these tests have passed, we can be more confident in the deployment of our contract.
Updated about 6 years ago