From 23a16fc670646a4a31b9397406f9f20ef36681ea Mon Sep 17 00:00:00 2001 From: lambo4jos Date: Sun, 31 Oct 2021 00:35:52 -0700 Subject: [PATCH 1/5] ADDED test and code for ensuring only the owner could make changes that were covered in the webinar --- contracts/SimpleStorage.sol | 6 ++++++ migrations/2_deploy_simple_storage.js | 2 +- test/simple_storage.js | 15 +++++++++++++++ 3 files changed, 22 insertions(+), 1 deletion(-) diff --git a/contracts/SimpleStorage.sol b/contracts/SimpleStorage.sol index a70b166..dc25821 100644 --- a/contracts/SimpleStorage.sol +++ b/contracts/SimpleStorage.sol @@ -3,12 +3,18 @@ pragma solidity >=0.5.0 <0.8.0; contract SimpleStorage { uint256 public storedData; + address owner = msg.sender; + + constructor(uint256 _num) public { + storedData = _num; + } function getStoredData() public view returns (uint256) { return storedData; } function setStoredData(uint256 x) public { + require(msg.sender == owner, "Not the owner!"); storedData = x; } } diff --git a/migrations/2_deploy_simple_storage.js b/migrations/2_deploy_simple_storage.js index 13b9825..7ec6519 100644 --- a/migrations/2_deploy_simple_storage.js +++ b/migrations/2_deploy_simple_storage.js @@ -1,5 +1,5 @@ const SimpleStorage = artifacts.require("SimpleStorage"); module.exports = function(deployer) { - deployer.deploy(SimpleStorage); + deployer.deploy(SimpleStorage, 0); } diff --git a/test/simple_storage.js b/test/simple_storage.js index 7231fdc..91169a0 100644 --- a/test/simple_storage.js +++ b/test/simple_storage.js @@ -27,5 +27,20 @@ contract("SimpleStorage", function (accounts) { const storedData = await ssInstance.getStoredData.call(); assert.equal(storedData, 42, `${storedData} was not stored!`); }); + + it("should not let someone else change the variable", async () => { + const [ owner, badBob ] = accounts; + const ssInstance = await SimpleStorage.new(42, { from: owner }); + + const balance = await web3.eth.getBalance(owner); + console.log(balance); + + try { + await ssInstance.setStoredData(22, { from: badBob }); + } catch(err) { } + + const storedData = await ssInstance.getStoredData.call(); + assert.equal(storedData, 42, "storedData was not changed!"); + }); }); }); From ed77e0697017d9b6eec18acd02996024830c7451 Mon Sep 17 00:00:00 2001 From: lambo4jos Date: Sun, 31 Oct 2021 02:02:57 -0700 Subject: [PATCH 2/5] ADDED instructions from the webinar not included --- steps.md | 159 +++++++++++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 149 insertions(+), 10 deletions(-) diff --git a/steps.md b/steps.md index ce755df..4c1288e 100644 --- a/steps.md +++ b/steps.md @@ -1,17 +1,19 @@ # Truffle BDD/TDD walkthrough +- [Truffle BDD/TDD walkthrough](#truffle-bddtdd-walkthrough) - [Project initialization](#project-initialization) - [SimpleStorage Behavior](#simplestorage-behavior) - * [the contract test](#the-contract-test) - * [the Contract subject](#the-contract-subject) - * [the migration](#the-migration) - * [business logic](#business-logic) - + [define initial deployment value of storedData](#define-initial-deployment-value-of-storeddata) - - [test: watch it fail](#test--watch-it-fail) - + [implement getStoredData](#implement-getstoreddata) - - [test: getStoredData](#test--getstoreddata) - + [define setStoredData behavior](#define-setstoreddata-behavior) + - [the contract test](#the-contract-test) + - [the Contract subject](#the-contract-subject) + - [the migration](#the-migration) + - [business logic](#business-logic) + - [define initial deployment value of storedData](#define-initial-deployment-value-of-storeddata) + - [test: watch it fail](#test-watch-it-fail) + - [implement getStoredData](#implement-getstoreddata) + - [test: getStoredData](#test-getstoreddata) + - [define setStoredData behavior](#define-setstoreddata-behavior) - [implement setStoredData](#implement-setstoreddata) + - [implement constructor and a way to ensure only the owner is able to execute setStoredData](#implement-constructor-and-a-way-to-ensure-only-the-owner-is-able-to-execute-setstoreddata) - [Conclusion](#conclusion) Table of contents generated with markdown-toc @@ -277,6 +279,9 @@ Our SimpleStorage contract should have: - [ ] its `storedData` value at deployment be zero - [ ] a function `getStoredData`, to retrieve the current `storedData` value. - [ ] a function `setStoredData`, to set the `storedData` value. + - [ ] a constructor to set the initial value of `storedData`. + - [ ] a contract `owner`. + - [ ] a way to ensure only the contract `owner` is able to call the `setStoredData` function. ### define initial deployment value of storedData @@ -423,7 +428,7 @@ Initial deployment ### define setStoredData behavior -There's one bit of feature missing. Lets implement! +Lets implement the next part! - [ ] a function `setStoredData`, to set the `storedData` value. Add the following `describe` block to our test and run truffle test. @@ -597,6 +602,140 @@ Initial deployment ``` +#### implement constructor and a way to ensure only the owner is able to execute setStoredData + +Add the following `it` block to the Functionality `describe` block test and run truffle test. + +``` javascript +it("should not let someone else change the variable", async () => { + const [ owner, badBob ] = accounts; + const ssInstance = await SimpleStorage.new(42, { from: owner }); + + const balance = await web3.eth.getBalance(owner); + console.log(balance); + + try { + await ssInstance.setStoredData(22, { from: badBob }); + } catch(err) { } + + const storedData = await ssInstance.getStoredData.call(); + assert.equal(storedData, 42, "storedData was not changed!"); +}); + +``` + +> :question: Try to predict the test outcome. + +``` sh +$ truffle test +``` + +
see output + +``` sh +$ 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 +✓ should store the value 42 +1) should not let someone else change the variable +> No events were emitted + + +3 passing (82ms) + 1 failing + + 1) Contract: SimpleStorage + Functionality + should not let someone else change the variable: + Error: Invalid number of parameters for "undefined". Got 2 expected 0! + at Context. (test/simple_storage.js:33:49) +at processImmediate (internal/timers.js:461:21) +``` + +
+ + +> :information_desk_person: `Error: Invalid number of parameters for "undefined". +> Got 2 expected 0!` +> This error indicates that we haven't included a constructor parameter and set +> the contract owner. + +Add the following variable to the SimpleStorage contract + +``` solidity +address owner = msg.sender; +``` + +Add the following constructor to the SimpleStorage contract + +``` solidity +constructor(uint256 _num) public { + storedData = _num; +} +``` + +Add the following require statement to the top of the setStoredData function in SimpleStorage contract + +``` solidity +require(msg.sender == owner, "Not the owner!"); +``` + +Initialize the storedData value to 0 by adding a parameter in 2_deploy_simple_storage.js + +``` solidity +module.exports = function(deployer) { + deployer.deploy(SimpleStorage, 0); +} +``` + +Lets run truffle test once again + +``` sh +$ truffle test +``` + +
see results + +``` sh +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) +99986593140000000000 +✓ should not let someone else change the variable + + +4 passing (125ms) +``` +
+ + :tada: :sparkles: Congratulations! You did it! I hope this exercise was helpful and recommend you continue exploring. From af91acdf3b966e9bc13beae852c4ea5fa74e7604 Mon Sep 17 00:00:00 2001 From: lambo4jos Date: Wed, 10 Nov 2021 13:33:42 -0800 Subject: [PATCH 3/5] UPDATE steps.md instructions and contract initialization --- migrations/2_deploy_simple_storage.js | 3 ++- steps.md | 25 ++++++++++++------------- test/simple_storage.js | 18 +++++++++++++----- 3 files changed, 27 insertions(+), 19 deletions(-) diff --git a/migrations/2_deploy_simple_storage.js b/migrations/2_deploy_simple_storage.js index 7ec6519..4c30224 100644 --- a/migrations/2_deploy_simple_storage.js +++ b/migrations/2_deploy_simple_storage.js @@ -1,5 +1,6 @@ const SimpleStorage = artifacts.require("SimpleStorage"); module.exports = function(deployer) { - deployer.deploy(SimpleStorage, 0); + /* invoke SimpleStorage's constructor, and pass 12 as its first argument */ + deployer.deploy(SimpleStorage, 12); } diff --git a/steps.md b/steps.md index 4c1288e..bf81f0c 100644 --- a/steps.md +++ b/steps.md @@ -276,7 +276,7 @@ test, contract and migration pieces 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 + - [ ] its `storedData` value at deployment be 12 - [ ] a function `getStoredData`, to retrieve the current `storedData` value. - [ ] a function `setStoredData`, to set the `storedData` value. - [ ] a constructor to set the initial value of `storedData`. @@ -296,12 +296,12 @@ contract("SimpleStorage", (/* accounts */) => { assert.isTrue(true); }); - it("was deployed and it's intial value is 0", async () => { + it("was deployed and its intial value is 12", async () => { // get subject const ssInstance = await SimpleStorage.deployed(); - // verify it starts with zero + // verify it starts with 12 const storedData = await ssInstance.getStoredData.call(); - assert.equal(storedData, 0, `Initial state should be zero`); + assert.equal(storedData, 12, `Initial state should be 12`); }); }); }); @@ -419,12 +419,11 @@ Initial deployment :tada: SimpleStorage has: - [x] a state, `storedData`. This is the location to store an integer value - - [x] its `storedData` value at deployment be zero + - [x] its `storedData` value at deployment be 12 - [x] a function `getStoredData`, to retrieve the current `storedData` value. - [ ] a function `setStoredData`, to set the `storedData` value. -> :question: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](https://solidity.readthedocs.io/en/v0.7.0/control-structures.html#scoping-and-declarations) from the Solidity documentation. +> :information_desk_person: Please note, we have declared and initialized the storedData variable, but solidity has rules for default values for unintialized variables. See [section](https://solidity.readthedocs.io/en/v0.7.0/control-structures.html#scoping-and-declarations) from the Solidity documentation. ### define setStoredData behavior @@ -474,7 +473,7 @@ Compiling your contracts... Contract: SimpleStorage Initial deployment ✓ should assert true -✓ was deployed and it's intial value is 0 +✓ was deployed and its intial value is 12 Functionality 1) should store the value 42 > No events were emitted @@ -674,7 +673,7 @@ at processImmediate (internal/timers.js:461:21) > This error indicates that we haven't included a constructor parameter and set > the contract owner. -Add the following variable to the SimpleStorage contract +Add and initialize the following state variable to the SimpleStorage contract ``` solidity address owner = msg.sender; @@ -694,11 +693,12 @@ Add the following require statement to the top of the setStoredData function in require(msg.sender == owner, "Not the owner!"); ``` -Initialize the storedData value to 0 by adding a parameter in 2_deploy_simple_storage.js +Initialize the storedData value to 12 by adding a parameter in 2_deploy_simple_storage.js ``` solidity module.exports = function(deployer) { - deployer.deploy(SimpleStorage, 0); + /* invoke SimpleStorage's constructor, and pass 12 as its first argument */ + deployer.deploy(SimpleStorage, 12); } ``` @@ -724,10 +724,9 @@ Compiling your contracts... Contract: SimpleStorage Initial deployment ✓ should assert true -✓ was deployed and it's intial value is 0 +✓ was deployed and its intial value is 12 Functionality ✓ should store the value 42 (55ms) -99986593140000000000 ✓ should not let someone else change the variable diff --git a/test/simple_storage.js b/test/simple_storage.js index 91169a0..9909279 100644 --- a/test/simple_storage.js +++ b/test/simple_storage.js @@ -7,12 +7,12 @@ contract("SimpleStorage", function (accounts) { assert.isTrue(true); }); - it("was deployed and it's intial value is 0", async () => { + it("was deployed and its intial value is 12", async () => { // get subject const ssInstance = await SimpleStorage.deployed(); - // verify it starts with zero + // verify it starts with 12 const storedData = await ssInstance.getStoredData.call(); - assert.equal(storedData, 0, `Initial state should be zero`); + assert.equal(storedData, 12, `Initial state should be 12`); }); }); describe("Functionality", () => { @@ -32,8 +32,16 @@ contract("SimpleStorage", function (accounts) { const [ owner, badBob ] = accounts; const ssInstance = await SimpleStorage.new(42, { from: owner }); - const balance = await web3.eth.getBalance(owner); - console.log(balance); + /* + * + * @comment: if you wanted to interact with web3, here is an + * example of checking the balance of an account and outputting + * the result to the test console + * + * const balance = await web3.eth.getBalance(owner); + * console.log(balance); + * + */ try { await ssInstance.setStoredData(22, { from: badBob }); From f9aad680e54283ded2e04d79ae8d3799b182e6d2 Mon Sep 17 00:00:00 2001 From: lambo4jos Date: Wed, 10 Nov 2021 13:50:21 -0800 Subject: [PATCH 4/5] UPDATE steps.md instructions for missed changes --- steps.md | 14 +++++++++++--- 1 file changed, 11 insertions(+), 3 deletions(-) diff --git a/steps.md b/steps.md index bf81f0c..033d2e7 100644 --- a/steps.md +++ b/steps.md @@ -610,8 +610,16 @@ it("should not let someone else change the variable", async () => { const [ owner, badBob ] = accounts; const ssInstance = await SimpleStorage.new(42, { from: owner }); - const balance = await web3.eth.getBalance(owner); - console.log(balance); + /* + * + * @comment: if you wanted to interact with web3, here is an + * example of checking the balance of an account and outputting + * the result to the test console + * + * const balance = await web3.eth.getBalance(owner); + * console.log(balance); + * + */ try { await ssInstance.setStoredData(22, { from: badBob }); @@ -647,7 +655,7 @@ Compiling your contracts... Contract: SimpleStorage Initial deployment ✓ should assert true -✓ was deployed and it's intial value is 0 +✓ was deployed and its intial value is 12 Functionality ✓ should store the value 42 1) should not let someone else change the variable From 87af0da9b1af96891e093cc49b485f815bd72d26 Mon Sep 17 00:00:00 2001 From: lambo4jos Date: Wed, 10 Nov 2021 14:11:09 -0800 Subject: [PATCH 5/5] UPDATE steps.md instructions correcting grammar and zero's --- steps.md | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/steps.md b/steps.md index 033d2e7..5989e63 100644 --- a/steps.md +++ b/steps.md @@ -339,7 +339,7 @@ Compiling your contracts... Contract: SimpleStorage Initial deployment ✓ should assert true - 1) was deployed and it's intial value is 0 + 1) was deployed and its intial value is 12 > No events were emitted @@ -348,7 +348,7 @@ Compiling your contracts... 1) Contract: SimpleStorage Initial deployment - was deployed and it's intial value is 0: + was deployed and its intial value is 12: 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) @@ -409,7 +409,7 @@ Compiling your contracts... Contract: SimpleStorage Initial deployment ✓ should assert true -✓ was deployed and it's intial value is 0 +✓ was deployed and its intial value is 12 2 passing (63ms) @@ -532,7 +532,7 @@ Compiling your contracts... Contract: SimpleStorage Initial deployment ✓ should assert true -✓ was deployed and it's intial value is 0 +✓ was deployed and its intial value is 12 Functionality 1) should store the value 42 > No events were emitted @@ -592,7 +592,7 @@ Compiling your contracts... Contract: SimpleStorage Initial deployment ✓ should assert true -✓ was deployed and it's intial value is 0 +✓ was deployed and its intial value is 12 Functionality ✓ should store the value 42 (55ms)