本文参考:https://blog.csdn.net/pony_maggie/article/details/79685942
实现了最终的结果,并在其基础上做出了更加详细的说明与部分细节的完善,供读者参考。
ubuntu 14.0.4(16.0.4), 64位
还需要安装以太坊相关的环境:
具体安装步骤可以参考这篇文章:http://blog.csdn.net/pony_maggie/article/details/79531534
关于testrpc与truffle可参考这篇文章:https://blog.csdn.net/chenyufeng1991/article/details/53456270
首先在用户目录(home)下新建conference目录(任意目录都可以),进入目录执行truffle init,该命令会建立如下的子目录和文件:
修改truffle.js文件,改成如下:
module.exports = {
networks: {
development: {
host: "localhost",
port: 8545,
network_id: "*" // 匹配任何network id
}
}
};
这里是设置我们稍后要部署智能合约的位置, 否则会报网络错误。
开启一个终端,输入testrpc运行测试节点。testrpc是一个完整的在内存中的区块链测试环境,启动 testrpc 经后,会默认创建10个帐号,Available Accounts是帐号列表,Private Keys是相对应的帐号密钥。
进入contracts目录,这里是存放合约代码的地方。我们可以使用sublime等工具编写测试合约代码。我这里只贴出部分代码,文章最后会给出完整源码的地址。
pragma solidity ^0.4.19;
contract Conference { // can be killed, so the owner gets sent the money in the end
address public organizer;
mapping (address => uint) public registrantsPaid;
uint public numRegistrants;
uint public quota;
event Deposit(address _from, uint _amount); // so you can log the event
event Refund(address _to, uint _amount); // so you can log the event
function Conference() {
organizer = msg.sender;
quota = 100;
numRegistrants = 0;
}
}
...
合约内容很简单,是一个针对会议的智能合约,通过它参会者可以买票,组织者可以设置参会人数上限,以及退款策略。
修改migrations下的1_initial_migration.js文件,改成如下:
//var Migrations = artifacts.require("./Migrations.sol");
var Conference = artifacts.require("./Conference.sol");
module.exports = function(deployer) {
//deployer.deploy(Migrations);
deployer.deploy(Conference);
};
编译,
sudo truffle compile --compile-all
此处编译智能合约可能会有警告,提示构造函数形式的改变,原因是高版本的solidity使用了 constructor作为声明构造函数的关键字。
Truffle仅默认编译自上次编译后被修改过的文件,来减少不必要的编译。如果你想编译全部文件,可以使用–compile-all选项。
然后会多出一个build目录,该目录下的文件都不要做任何的修改。
部署,
sudo truffle migrate --reset
这个命令会执行所有migrations目录下的js文件。如果之前执行过truffle migrate命令,再次执行,只会部署新的js文件,如果没有新的js文件,不会起任何作用。如果使用–reset参数,则会重新的执行所有脚本的部署
测试下,在test目录新增一个conference.js测试文件,
var Conference = artifacts.require("./Conference.sol");
contract('Conference', function(accounts) {
console.log("start testing");
//console.log(accounts);
var owner_account = accounts[0];
var sender_account = accounts[1];
it("Initial conference settings should match", function(done) {
Conference.new({from: owner_account}).then(
function(conference) {
conference.quota.call().then(
function(quota) {
assert.equal(quota, 100, "Quota doesn't match!");
}).then(
function() {
return conference.numRegistrants.call();
}).then(
function(num) {
assert.equal(num, 0, "Registrants doesn't match!");
return conference.organizer.call();
}).then(
function(organizer) {
assert.equal(organizer, owner_account, "Owner doesn't match!");
done();
}).catch(done);
}).catch(done);
});
});
...
这里只贴出部分代码,一个测试case,运行truffle test查看测试结果。
在conference目录下执行npm init,然后一路回车,会生成一个名为package.json的文件,编辑这个文件,在scripts部分增加两个命令,最终如下:
{
"name": "conference",
"version": "1.0.0",
"description": "",
"main": "truffle-config.js",
"directories": {
"test": "test"
},
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"start": "webpack",
"server": "webpack-dev-server --open"
},
"author": "",
"license": "ISC"
}
package.json文件定义了这个项目所需要的各种模块,以及项目的配置信息(比如名称、版本、许可证等元数据)。npm 命令根据这个配置文件,自动下载所需的模块,也就是配置项目所需的运行和开发环境。
然后在conference目录下新建app目录,并创建index.html文件,如下:
<html>
<head>
<title>Conference DApp2title>
<link href='https://fonts.loli.net/css?family=Open+Sans:400,700,300' rel='stylesheet' type='text/css'>
<script type="text/javascript" src="http://code.jquery.com/jquery-1.9.1.min.js">script>
<script src="./app.js">script>
head>
<body>
<h1>Conference DApph1>
<div class="section">
Contract deployed at: <div id="confAddress">div>
div>
<div class="section">
Organizer: <input type="text" id="confOrganizer" />
div>
<div class="section">
Quota: <input type="text" id="confQuota" />
<button id="changeQuota">Changebutton>
<span id="changeQuotaResult">span>
div>
<div class="section">
Registrants: <span id="numRegistrants">0span>
div>
<hr/>
body>
html>
然后在app目录下新建javascripts目录和styleheets目录,分别存放js脚本文件和css样式文件。真正和合约交互的就是脚本文件。
脚本文件名为app.js,完整代码如下(原文中没有给出完整代码导致无法实现文中结果):
import "../stylesheets/app.css";
import { default as Web3 } from 'web3';
import { default as contract } from 'truffle-contract';
import conference_artifacts from '../../build/contracts/Conference.json'
var accounts, sim;
var Conference = contract(conference_artifacts);
window.addEventListener('load', function() {
//alert("aaaaa");
// Checking if Web3 has been injected by the browser (Mist/MetaMask)
if (typeof web3 !== 'undefined') {
console.warn("Using web3 detected from external source. If you find that your accounts don't appear or you have 0 MetaCoin, ensure you've configured that source properly. If using MetaMask, see the following link. Feel free to delete this warning. :) http://truffleframework.com/tutorials/truffle-and-metamask")
// Use Mist/MetaMask's provider
window.web3 = new Web3(web3.currentProvider);
} else {
console.warn("No web3 detected. Falling back to http://localhost:8545. You should remove this fallback when you deploy live, as it's inherently insecure. Consider switching to Metamask for development. More info here: http://truffleframework.com/tutorials/truffle-and-metamask");
// fallback - use your fallback strategy (local node / hosted node + in-dapp id mgmt / fail)
window.web3 = new Web3(new Web3.providers.HttpProvider("http://localhost:8545"));
}
Conference.setProvider(web3.currentProvider);
App.start();
$("#changeQuota").click(function() {
var newquota = $("#confQuota").val();
App.changeQuota(newquota);
});
// Wire up the UI elements
});
window.App = { //where to close
start: function() {
var self = this;
web3.eth.getAccounts(function(err, accs) {
if (err != null) {
alert("There was an error fetching your accounts.");
return;
}
if (accs.length == 0) {
alert("Couldn't get any accounts! Make sure your Ethereum client is configured correctly.");
return;
}
accounts = accs;
//$("#tentantAddress").html(getBalance(accounts[0])); //prints balance
//console.log(accounts);
self.initializeConference();
});
},
initializeConference: function() {
var self = this;
Conference.deployed().then(function(instance) {
sim = instance;
$("#confAddress").html(sim.address);
self.checkValues();
}).catch(function(e) {
console.log(e);
});
},
checkValues: function() {
Conference.deployed().then(function(instance) {
sim = instance;
console.log(sim);
sim.quota.call().then(
function(quota) {
console.log(quota);
$("input#confQuota").val(quota);
return sim.organizer.call();
}).then(
function(organizer){
$("input#confOrganizer").val(organizer);
return sim.numRegistrants.call();
}).then(
function(num){
$("#numRegistrants").html(num.toNumber());
});
});
},
changeQuota: function(newquota){
Conference.deployed().then(function(instance) {
sim = instance;
console.log(sim);
sim.changeQuota(newquota,{from:accounts[0],gas:3000000}).then(
function() {
return sim.quota.call();
}).then(
function(quota){
var msgResult;
if(quota == newquota){
msgResult = "change sucessful";
}else{
msgResult = "change failed";
}
$("#changeQuotaResult").html(msgResult);
});
});
}
};//loop for main
到这里为止,web部分基本已经准备好了,我们只需要用webpack打包部署即可。webpack打包还需要一个配置文件,名为webpack.config.js,这个文件是告诉webpack打包的规则,涉及webpack的用法,这里不做过多的解释。
打包部署需要安装webpack和相关的组件,安装的方式有全局安装和局部安装两种。所谓的局部安装,是指组件都是安装在项目的目录下(conference/node_modules)。我这里采用的就是局部安装。根据我们项目的实际情况,需要安装以下组件,
npm install --save-dev [email protected]
npm install babel-loader --save-dev
npm install babel-core --save-dev
npm install html-loader --save-dev
npm install --save-dev [email protected]
npm install html-webpack-plugin --save-dev
npm install truffle-contract --save-dev
npm install --save-dev style-loader css-loader
没报错的话,进入build目录可以看到bundle.js和index.html两个文件,这两个就是最终打包好的网页文件。
然后部署,
此时,浏览器会自动打开:http://localhost:8080,效果如下:
可以看到合约的发布地址和会议组织者地址(msg.sender)都已经成功的显示出来了,点击change按钮还可以改变quota的值。
最后附上github代码地址:https://github.com/pony-maggie/conference