Introduction to DApp Development¶
Stage 1: Starting the Application¶
Note
Begin instructions from within the VM(via VirtualBox) that was configured and run in step 6 of the prerequisites.
1. Open a new terminal window¶
- Click on the terminal icon in the left dock
2. Start the app¶
- 2.1 Change directory into the
blg/wallet-template
foldercd Desktop/blg/wallet-template
- 2.2 Start the server
yarn start
- Example output:
$ yarn start
Starting the development server...
Compiled successfully!
You can now view my-app in the browser.
Local: http://localhost:3000/
On Your Network: http://172.17.0.2:3000/
Note that the development build is not optimized.
To create a production build, use yarn build.
- Chrome should automatically be opened and the application rendered!
3. Open the application’s code in the Sublime text editor¶
- Open the Sublime text editor by clicking on the Sublime icon in the left dock.
- From within Sublime open the wallet-template folder. Click on
File
in the top left corner and selectOpen Folder...
in the menu. SelectDesktop/blg/wallet-template
to open, and we can get to coding!
END Stage 1: Starting the Application!
Stage 2: Testing Your Token¶
Important
The following videos may make note of the use of something called “Docker” and “containers”, but do note that Docker has since been omitted. The same commands that are mentioned may be run directly on the machine without entering into the noted container.
1. Create a new tab in your terminal window or a new terminal window for our Ethereum node¶
Note
While within the terminal window select File -> Open Terminal to create a new window.
To create a new tab from within a terminal window:
ctrl+shft+t
- Example output: Result is a new empty terminal, in the same directory you were in.
adam@adam:~/Desktop/blg$
2. Start up your Ethereum node, ganache-cli¶
ganache-cli
- Example output:
# ganache-cli
Ganache CLI v6.0.3 (ganache-core: 2.0.2)
[...]
Listening on localhost:8545
3. Create a new window or tab for our Truffle commands¶
Note
While within the terminal window select File -> Open Terminal to create a new window.
To create a new tab from within a terminal window:
ctrl+shft+t
- Example output: Result is a new empty terminal, in the same directory you were in.
adam@adam:~/Desktop/blg$
4. Create the Test Case¶
Note
- contracts/Token.sol has been provided.
- Also one test file template has been provided in order to test the buy method was implemented correctly.
- Open the test file within Sublime,
src/test/test_buy.js
- Import the token’s build artifacts,
src/test/test_buy.js
line 2
const Token = artifacts.require("./Token.sol")
- Define the owner account, note
truffle test
exposes the accounts array for us, line 6
const owner = accounts[0]
- Create a new instance of the token contract, line 10
const token = await Token.new({ from: owner })
- Specify the wei value of tokens you wish to purchase, line 13
const value = 100
- Send the transaction to the token’s buy method, line 16
const txResponse = await token.buy({ from: owner, value })
- Pull the rate from the token, line 19
const rate = await token.rate()
- Compute the token amount to be minted to the buyer, line 22
const tokenAmount = value * rate
- Access the event object from the transaction receipt, line 25
const event = txResponse.logs[0]
- Assert the correct values were emitted, line 28-31
assert.equal(event.event, 'TokensMinted', 'TokensMinted event was not emitted.')
assert.equal(event.args.to, owner, 'Incorrect to was emitted.')
assert.equal(event.args.value, value, 'Incorrect value was emitted.')
assert.equal(event.args.totalSupply.toNumber(), tokenAmount, 'Incorrect totalSupply was emitted.')
Ensure the state of the contract is updated correctly
- Assert the buyer’s balance is correct, line 34-35
const balance = await token.balanceOf(owner)
assert.equal(balance.toNumber(), tokenAmount, 'Incorrect token balance.')
- Assert the total supply is correct, line 38-39
const supply = await token.totalSupply()
assert.equal(supply.toNumber(), tokenAmount, 'Incorrect total supply.')
5. Execute the Test Case¶
- Change into the
src
directorycd src
- Execute the test
truffle test
- Example output:
$ truffle test
Using network 'development'.
Contract: Token.buy()
✓ should buy new tokens. (133ms)
1 passing (148ms)
#
END Stage 2: Testing Your Token
Stage 3: Token Deployment¶
Note
- A default, and required, initial migration script(src/migrations/1_initial_migration.js), has been included. Do not remove this script.
1. Write the Deployment Script¶
- Create a new file in order to deploy the token,
src/migrations/2_deploy_contracts.js
- Simply right-click on the migrations directory and create the new file.
- Import the token’s artifacts, line 1
const Token = artifacts.require("./Token.sol");
- Define the owner account, note
truffle migrate
exposes the web3 object, line 2
const owner = web3.eth.accounts[0]
- Utilize truffle’s deployer object in order to deploy an instance of the token, line 4-6
module.exports = deployer => {
deployer.deploy(Token, { from: owner })
}
2. Deploy your Token¶
truffle migrate
- Example output:
# truffle migrate
Using network 'development'.
Running migration: 1_initial_migration.js
Deploying Migrations...
... 0x26ff3f480502a228f34363e938289c3164edf8bc49c75f5d6d9623a05da92dbf
Migrations: 0x3e47fad1423cbf6bd97fee18ae2de546b0e9188a
Saving successful migration to network...
... 0x19a7a819df452847f34815e2573765be8c26bac43b1c10d3b7528e6d952ac02c
Saving artifacts...
Running migration: 2_deploy_contracts.js
Deploying Token...
... 0x4a69e7840d0f96067964fb515ffea1a04a98fc5759849d3308584af4770c8f7b
Token: 0xd58c6b5e848d70fd94693a370045968c0bc762a7
Saving successful migration to network...
... 0xd1e9bef5f19bb37daa200d7e563f4fa438da60dbc349f408d1982f8626b3c202
Saving artifacts...
#
END Stage 3: Token Deployment
Stage 4: Token Interface¶
1. Import the web3 library, src/app.js
#line 5¶
- Open
src/app.js
in the Sublime text editor
import Web3 from 'web3'
2. Import the token build artifacts into the application, app.js#line 14¶
import tokenArtifacts from './build/contracts/Token.json'
3. Create a web3 connection to the local Ethereum node(ganache-cli), app.js#line 26¶
this.web3 = new Web3(new Web3.providers.HttpProvider("http://localhost:8545"));
4. Check if the connection was successful, app.js#line 28-30¶
if (this.web3.isConnected()) {
}
5. Detect the current network id that is connected, app.js#line 29-31¶
this.web3.version.getNetwork(async (err, netId) => {
})
6. Extract the recently deploy token address from the build artifacts, app.js#line 30-33¶
// Create a reference object to the deployed token contract
if (netId in tokenArtifacts.networks) {
const tokenAddress = tokenArtifacts.networks[netId].address
}
7. Create a client side reference to the contract and save it in the application’s state, app.js#line 33-35¶
const token = this.web3.eth.contract(tokenArtifacts.abi).at(tokenAddress)
this.setState({ token })
console.log(token)
8. Refresh your chrome browser and open up the developer console¶
This can be accomplished by right-clicking anywhere in the chrome browser and in the dropdown selecting inspect
or inspect element
or by utilizing the shortcut: ctrl+shift+i
.
View in the developer console the token instance is now present
- Example output:
Contract {_eth: Eth, transactionHash: null, address: "0xd58c6b5e848d70fd94693a370045968c0bc762a7", abi: Array[20]}
END Stage 4: Token Interface
Stage 5: Load Available On-chain Accounts¶
1. Get the available accounts from the web3 connection, this is to wrap the existing token interface code, line 29 & 39¶
this.web3.eth.getAccounts((err, accounts) => { // Line 29
}) // Line 39
2. Set the default account to use, line 30¶
const defaultAccount = this.web3.eth.accounts[0]
3. Load the available accounts into the user interface¶
- Import the Material UI MenuItem, line 8
import MenuItem from 'material-ui/MenuItem';
- Add an availableAccounts arrary into the app’s state, line 21
availableAccounts: [],
- Append all accounts into the UI dropdown menu, line 34-41
// Append all available accounts
for (let i = 0; i < accounts.length; i++) {
this.setState({
availableAccounts: this.state.availableAccounts.concat(
<MenuItem value={i} key={accounts[i]} primaryText={accounts[i]} />
)
})
}
4. Set the default account¶
- Add a defaultAccount variable to the state, line 22
defaultAccount: 0,
- Set the defaultAccount in the state when the drowdown value changes, line 86
this.setState({ defaultAccount })
END Stage 5: Load Available Accounts
Stage 6: Token Interaction - GET¶
1. Load the token metadata from the contract¶
- Add the token’s symbol to the state, line 23
tokenSymbol: 0,
- Load the token’s symbol, line 52-55
// Set token symbol below
token.symbol((err, tokenSymbol) => {
this.setState({ tokenSymbol })
})
- Add the token’s rate to the state, line 23
rate: 1,
- Load the token’s rate, line 58-61
// Set wei / token rate below
token.rate((err, rate) => {
this.setState({ rate: rate.toNumber() })
})
END Stage 6: Token Interaction - GET
Stage 7: Load Account Balances¶
1. Load the default account’s ETH and Token balances, completing the loadAccountBalances
method¶
- Confirm the token has been loaded, line 73-75
if (this.state.token) {
}
- Add tokenBalance to the state, line 24
tokenBalance: 0,
- Set the token balance, line 75-78
// Set token balance below
this.state.token.balanceOf(account, (err, balance) => {
this.setState({ tokenBalance: balance.toNumber() })
})
- Add ethBalance to the state, line 23
ethBalance: 0,
- Set the eth balance, line 81-84
// Set ETH balance below
this.web3.eth.getBalance(account, (err, ethBalance) => {
this.setState({ ethBalance })
})
- Call the
loadAccountBalances
method on load, line 67
this.loadAccountBalances(defaultAccount)
- Also load the balances whenever a new account is selected in the dropdown, line 111
this.loadAccountBalances(this.state.availableAccounts[index].key)
2. View the default account balances and token information in your browser!¶
END Stage 7: Load Available Account Balances
Stage 8: Purchasing Tokens¶
1. Add token amount to the state, line 21.¶
amount: 0,
2. Complete the method to buy tokens, sending a transaction to the token contract, line 99-104.¶
this.state.token.buy({
from: this.web3.eth.accounts[this.state.defaultAccount],
value: amount
}, (err, res) => {
err ? console.error(err) : console.log(res)
})
3. In the GUI buy tokens with several available accounts.¶
Note
Note transaction hash in the developer console
Example transaction hash: 0x4b396191e87c31a02e80160cb6a2661da6086c073f6e91e9bd1f796e29b0c983
4. Refresh the browser or select a different account and come back, and view the account’s balance of shiny new tokens!¶
END Stage 8: Purchasing Tokens
Stage 9: Events¶
1. Add an event to listen for when tokens are transferred and reload the account’s balances, line 94-99¶
// Watch tokens transfer event below
this.state.token.Transfer({ fromBlock: 'latest', toBlock: 'latest' })
.watch((err, res) => {
console.log(`Tokens Transferred! TxHash: ${res.transactionHash} \n ${JSON.stringify(res.args)}`)
this.loadAccountBalances(this.web3.eth.accounts[this.state.defaultAccount])
})
2. Load the contract events, line 66¶
this.loadEventListeners()
3. Buy tokens and view the log confirmation in the developer console and token and ETH balance updated dynamically!¶
END Stage 9: Events
Stage 10: Transfer Tokens¶
Try this portion on your own! [Solution noted at the bottom]
The required components included:
- Add the transferAmount and transferUser to the app’s state.
- Add the React transfer tokens form component.
- Complete the transfer method to send the transfer transaction.
Finally transfer tokens between accounts and review balances.
END Stage 10: Transfer Tokens
Solutions¶
Stage 10: Transfer Tokens¶
- Add the transferAmount and transferUser to the app’s state, line 28 & 29.
transferAmount: '',
transferUser: '',
- Add the React transfer tokens form component, line 150-161.
<div>
<h3>Transfer Tokens</h3>
<TextField floatingLabelText="User to transfer tokens to." style={{width: 400}} value={this.state.transferUser}
onChange={(e, transferUser) => { this.setState({ transferUser }) }}
/>
<TextField floatingLabelText="Amount." style={{width: 100}} value={this.state.transferAmount}
onChange={(e, transferAmount) => { this.setState({ transferAmount })}}
/>
<RaisedButton label="Transfer" labelPosition="before" primary={true}
onClick={() => this.transfer(this.state.transferUser, this.state.transferAmount)}
/>
</div>
- Complete the transfer method to send the transfer transaction, line 117-124.
if (amount > 0) {
// Execute token transfer below
this.state.token.transfer(user, amount, {
from: this.web3.eth.accounts[this.state.defaultAccount]
}, (err, res) => {
err ? console.error(err) : console.log(res)
})
}
Complete Wallet Solution¶
git clone https://github.com/Blockchain-Learning-Group/wallet-eod2.git
cd wallet-eod2
git checkout tags/2.0
Bonus: Extend Your Wallet¶
2. Metamask Integration¶
- Ensure Metamask is installed, unlocked and connected to the local client(localhost:8545). - Metamask may be insalled here
- Fund your Metamask account!
$ truffle console
truffle(development> web3.eth.sendTransaction({ from: web3.eth.accounts[0], to: 'METAMASK_ADDRESS', value: 1e18 })
- Transfer tokens to your metamask account(from within the application).
- Add a conditional to use the Metamask web3 provider if present, wallet-template/src/App.js#L35
if (window.web3)
this.web3 = new Web3(window.web3.currentProvider)
else
- Refresh the browser and connect to your Metamask account. View your Metamask account now available within the application.
3. Sync an Ethereum node of your own¶
Note
Look to setup a node locally or via Azure. Azure is a nice option to begin with as a node locally can be quite heavy and resource intensive.
- Getting Started With Azure
- Sync a Parity node to Kovan
- Instructions to deploy to Azure here
- Parity Homepage
- Sync a Geth node to Rinkeby
- Instructions here
- Geth Homepage