Table of contents generated with markdown-toc
- First install Truffle. See documentation here
- Create a workplace for this project
The following commands initializes a truffle project in the $HOME/tut directory.
$ cd # navigate to $HOME folder
$ mkdir tut && cd $_ # make a tut folder and navigate into it.
$ truffle init
See output
$ truffle init
Starting init...
================
> Copying project files to /home/amal/work/webinars/tut
Init successful, sweet!
We now have the basic ingredients for a Truffle project.
File | description |
---|---|
Migrations.sol | a migration contract |
1_initial_migration.js | a migration script |
truffle-config.js | Truffle's configuration file |
Run truffle test
to check the system.
truffle test
will execute all the test we specified and report the results.
$ truffle test
See output
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations
π There's not much to report because we have not yet written any tests.
We want to write a Smart Contract with a simple API: set its value, and read its value. To do this we need to have the following.
- a smart contract,
SimpleStorage
- some way to define the behavior of
SimpleStorage
- the ability to verify
SimpleStorage
behaves the way we defined
We will use aspects of Behavior driven development (BDD), and Test driven development (TDD) methodologies to work through this tutorial. We will define new behavior in truffle test files and implement contract logic to achieve that behavior.
This approach lets us to examine the system in small testable pieces while increasing our knowledge and intuition of Truffle, Solidity and the ecosystem.
Let's create the truffle test-file for SimpleStorage, invoke truffle test, and examine the output.
truffle create test <ContractName>
scaffolds a truffle test! This is where we define the behavior of a contract.
See output
$ truffle create test SimpleStorage
$ truffle test
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Artifacts written to /tmp/test--15904-aWUmTcuhKTB5
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Error: Could not find artifacts for SimpleStorage from any sources
at Resolver.require (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/build/webpack:/packages/resolver/dist/lib/resolver.js:35:1)
at TestResolver.require (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/build/webpack:/packages/core/lib/testing/TestResolver.js:24:1)
at Object.require (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/build/webpack:/packages/core/lib/test.js:278:1)
at Object.<anonymous> (/home/amal/work/webinars/tut/test/simple_storage.js:1:33)
at Module._compile (internal/modules/cjs/loader.js:778:30)
at Object.Module._extensions..js (internal/modules/cjs/loader.js:789:10)
at Module.load (internal/modules/cjs/loader.js:653:32)
at tryModuleLoad (internal/modules/cjs/loader.js:593:12)
at Function.Module._load (internal/modules/cjs/loader.js:585:3)
at Module.require (internal/modules/cjs/loader.js:692:17)
at require (internal/modules/cjs/helpers.js:25:18)
at /home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/node_modules/mocha/lib/mocha.js:390:36
at Array.forEach (<anonymous>)
at Mocha.loadFiles (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/node_modules/mocha/lib/mocha.js:387:14)
at Mocha.run (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/node_modules/mocha/lib/mocha.js:961:10)
at resolve (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/build/webpack:/packages/core/lib/test.js:159:1)
at new Promise (<anonymous>)
at Object.run (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/build/webpack:/packages/core/lib/test.js:158:1)
at process._tickCallback (internal/process/next_tick.js:68:7)
Truffle v5.1.43 (core: 5.1.43)
Node v10.20.1
β οΈ π This error means SimpleStorage is missing in action!
Truffle test mocks! It compiles and deploys the contracts we write to a local
test blockchain every time we run truffle test
. Our only behavior, the
expectation that SimpleStorage
is deployed, fails because we have not yet
introduced this contract to our project.
Now that we've defined the SimpleStorage behavior, let's add the contract to our project and rerun the test.
truffle create contract <ContractName>
adds an empty contract to our project! This is where we will implement our desired contract behavior.
$ truffle create contract SimpleStorage
$ truffle test
Examine test output
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Compiling ./contracts/SimpleStorage.sol
> Artifacts written to /tmp/test--16564-yHf6bkBqCImA
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Contract: SimpleStorage
Error: SimpleStorage has not been deployed to detected network (network/artifact mismatch)
Error: SimpleStorage has not been deployed to detected network (network/artifact mismatch)
1) should assert true
> No events were emitted
0 passing (42ms)
1 failing
1) Contract: SimpleStorage
should assert true:
Uncaught Error: the string "abort(Error: SimpleStorage has not been deployed to detected network (network/artifact mismatch)). Build with -s ASSERTIONS=1 for more info." was thrown, throw an Error :)
at process.emit (/home/amal/.nvm/versions/node/v10.20.1/lib/node_modules/truffle/build/webpack:/node_modules/source-map-support/source-map-support.js:495:1)
at process._fatalException (internal/bootstrap/node.js:497:27)
β οΈ π This error is informative. It tells us thatSimpleStorage
has not been deployed to a network.
The insights and feedback from tests can increase our knowledge and increase
confidence in our Smart Contract evolution. As contract developers we
want to focus on implementing our contract logic and not have to wrestle with
complicated testchain business. Here's what truffle test
does at a high
level to facilitate that focus:
- starts a local test chain
- compiles all contracts it knows about
- deploys those contracts
- calls all the tests in the system and displays the results
Recall that Truffle scaffolded an initial migration script. Review it to see how it deploys a Contract
Truffle test will execute all the deployment scripts in lexical order. We will have to write one for our SimpleStorage contract.
π a migration script is the place to define a Smart Contract's deployment logic. These scripts are processed in lexical order which explains their odd filenames.
Create the SimpleStorage deployment migrations/2_deploy_simple_storage.js
and then execute another test
const SimpleStorage = artifacts.require("SimpleStorage");
module.exports = function(deployer) {
deployer.deploy(SimpleStorage);
}
Execute Truffle test
$ truffle test
Examine test output
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Compiling ./contracts/SimpleStorage.sol
> Artifacts written to /tmp/test--17726-UpvQCi3AszsJ
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Contract: SimpleStorage
β should assert true
1 passing (34ms)
π Congratulations! If you see a similar output then you have:
- a smart contract,
SimpleStorage
- a way to define the behavior of
SimpleStorage
- a way to validate its behavior is correct
π In other words, we've built a system to iterate, with focus, to the Contract behavior we want. In the test file we describe the desired contract behavior, implement said contract and test/observe/iterate to the desired outcome.
Lets focus on the business logic of our smart contract now that we have the test, contract and migration pices in place
Our SimpleStorage contract should have:
- a state,
storedData
. This is the location to store an integer value - its
storedData
value at deployment be zero - a function
getStoredData
, to retrieve the currentstoredData
value. - a function
setStoredData
, to set thestoredData
value.
Add the following code to the test file and execute truffle test.
contract("SimpleStorage", (/* accounts */) => {
describe("Initial deployment", async () => {
it("should assert true", async function () {
await SimpleStorage.deployed();
assert.isTrue(true);
});
it("was deployed and it's intial value is 0", async () => {
// get subject
const ssInstance = await SimpleStorage.deployed();
// verify it starts with zero
const storedData = await ssInstance.getStoredData.call();
assert.equal(storedData, 0, `Initial state should be zero`);
});
});
});
Since we defined a new behavior we expect the test to fail. Lets see how the system reports this expected failure.
π This is a great way to learn about new systems. We expect the test to fail and examining the errors will build our intuition about the system.
$ truffle test
See test result
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Compiling ./contracts/SimpleStorage.sol
> Artifacts written to /tmp/test--40553-kTukJ514YAII
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Contract: SimpleStorage
Initial deployment
β should assert true
1) was deployed and it's intial value is 0
> No events were emitted
1 passing (50ms)
1 failing
1) Contract: SimpleStorage
Initial deployment
was deployed and it's intial value is 0:
TypeError: Cannot read property 'call' of undefined
at Context.it (test/simple_storage.js:14:57)
at process._tickCallback (internal/process/next_tick.js:68:7)
π The error indicates something is undefined, and includes a file, line and column location Take a look at the file and try to deduce what's happening.
In the test file we ask truffle for SimpleStorage's deployed instance and then try to invoke a method on that instance. We have yet to define an API for SimpleStorage
update SimpleStorage.sol to
- declare a public uint256 named storedData
- add a public view named getStoredData that returns the contract's state storedData
contract SimpleStorage {
uint256 public storedData;
function getStoredData() public view returns (uint256) {
return storedData;
}
}
βTry to predict what will happen before running the test. Think about what we've observed so far. What will the change to the solidity file add to the system? How can it fail?
$ truffle test
See output
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Compiling ./contracts/SimpleStorage.sol
> Artifacts written to /tmp/test--41672-tvxa6sdLx1V6
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Contract: SimpleStorage
Initial deployment
β should assert true
β was deployed and it's intial value is 0
2 passing (63ms)
π SimpleStorage has:
- a state,
storedData
. This is the location to store an integer value - its
storedData
value at deployment be zero - a function
getStoredData
, to retrieve the currentstoredData
value. - a function
setStoredData
, to set thestoredData
value.
βSomething to ponder for later: we didn't initialize the state, yet its initial value is zero. Why do you think that is? If you can't figure out yourself, read this section from the Solidity documentation.
There's one bit of feature missing. Lets implement!
- a function
setStoredData
, to set thestoredData
value.
Add the following describe
block to our test and run truffle test.
describe("Functionality", () => {
it("should store the value 42", async () => {
// get subject
const ssInstance = await SimpleStorage.deployed();
// change the subject
await ssInstance.setStoredData(42, { from: accounts[0] });
// verify we changed the subject
const storedData = await ssInstance.getStoredData.call();
assert.equal(storedData, 42, `${storedData} was not stored!`);
});
});
β Try to predict the test outcome.
$ truffle test
see output
$ truffle test
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Compiling ./contracts/SimpleStorage.sol
> Artifacts written to /tmp/test--42579-z0xosuySymKs
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Contract: SimpleStorage
Initial deployment
β should assert true
β was deployed and it's intial value is 0
Functionality
1) should store the value 42
> No events were emitted
2 passing (82ms)
1 failing
1) Contract: SimpleStorage
Functionality
should store the value 42:
ReferenceError: accounts is not defined
at Context.it (test/simple_storage.js:25:50)
at process._tickCallback (internal/process/next_tick.js:68:7)
π
ReferenceError: accounts is not defined
This error indicates that we haven't included accounts in our test. When truffle runs its tests it creates a local testnet and creates 10 funded accounts that use for testing. This is part of the identity concept mentioned in the earlier part of the webinar. This is another piece of the eco system we don't have to worry about as we build up our smart contract.
Let's make sure we have accounts in our tests. Uncomment accounts
so test
entry point reads as:
contract("SimpleStorage", function (accounts) {
Run truffle test
$ truffle test
β What do you think will happen now? Will the test pass?
see output
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Compiling ./contracts/SimpleStorage.sol
> Artifacts written to /tmp/test--43198-xp1rxfykLBu7
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Contract: SimpleStorage
Initial deployment
β should assert true
β was deployed and it's intial value is 0
Functionality
1) should store the value 42
> No events were emitted
2 passing (81ms)
1 failing
1) Contract: SimpleStorage
Functionality
should store the value 42:
TypeError: ssInstance.setStoredData is not a function
at Context.it (test/simple_storage.js:25:24)
at process._tickCallback (internal/process/next_tick.js:68:7)
π Did you predict what would happen? We haven't implemented the contract behavior.
The following line in the output indicates that setStoredData
is not a
function. We should fix that.
TypeError: ssInstance.setStoredData is not a function
Add the following function to the SimpleStorage contract
function setStoredData(uint256 x) public {
storedData = x;
}
Lets run truffle test once again
$ truffle test
see results
Compiling your contracts...
===========================
> Compiling ./contracts/Migrations.sol
> Compiling ./contracts/SimpleStorage.sol
> Artifacts written to /tmp/test--44360-9r4fMMWnmb6y
> Compiled successfully using:
- solc: 0.5.16+commit.9c3226ce.Emscripten.clang
Contract: SimpleStorage
Initial deployment
β should assert true
β was deployed and it's intial value is 0
Functionality
β should store the value 42 (55ms)
3 passing (125ms)
π β¨ Congratulations! You did it! I hope this exercise was helpful and recommend you continue exploring.
Truffle with a BDD and TDD workflow is a powerful way to develop and test Smart contracts. It keeps you focused on the activities of contract development and testing by mocking the blockchain, managing identities and converts your tests and assertsions into transactions for you.
See you in the next episode where we'll continue our exploration with Ganache and Sandboxes.