首页 > 代码库 > 区块链入门(5)Truffle 项目实战,Solidity IDE, 智能合约部署

区块链入门(5)Truffle 项目实战,Solidity IDE, 智能合约部署

在上一张我们学习了Truffle项目的创建,部署等相关内容,今天我们就来实战一下.

今天我们要做3件事: 
        1) 学习搭建一个Solidity IDE(Remix).
        2) 使用这个Solidity Ide编写一份智能合约.
        3) 在我们前面第1,2,3章中部署的私有网络节点集群中部署这个合约,并能够在不同的节点中调用这个合约.

Remix,是一个Solidity开发语言的Ide, 它是一款运行在浏览器端的solidity ide,也是官方推荐使用的ide.

另外还有其它的solidiy ide,比如:Ethereum Studio, IntelliJ IDEA Plugin, Vistual studio extension...等这些ide的详细情况请参考官方文档:
https://solidity.readthedocs.io/en/latest/index.html

我们主要讲解Remix.

官方推荐的Solidity基于浏览器端的Ide(Remix)

运行Remix有两种方式,一种是直接用浏览器打开网址: https://remix.ethereum.org/
直接打开官方部署的编辑器,进行开发.

另外一种方式就是在本地部署Remix服务, 顺序运行下面的命令:
> git clone https://github.com/ethereum/remix
> cd remix
> npm install
> npm start
Remix服务就部署好了,然后在浏览器中打开:http://127.0.0.1:8080
就可以直接使用本地部署Remix进行开发了.

本地搭建Rmix 编辑器的好处,是可以随意修改编辑器中,源代码的字体.
vim ./node_modules/brace/index.js

找到下面的内容:
var editorCss = ".ace_editor {    position: relative;    overflow: hidden;    font: 10px/normal ‘Ubuntu Mono‘, ‘Monaco‘, ‘Menlo‘, ‘Consolas‘, ‘source-code-pro‘, monospace;    direction: ltr;}.ace_scroller {    position: absolute;    overflow: hidden;    top: 0;    bottom: 0;    background-color: inherit;    -ms-user-select: none;    -moz-user-select: none;    -webkit-user-select: none;    user-select: none;    cursor: text;}
在font这这一行,把字体修改成自己喜欢的大小和样式.

然后重新执行命令:
npm install
npm start

重新打开网页,即可看到效果:

技术分享

在Remix上编写第一份智能合约

打开浏览器,访问http://127.0.0.1:8080, 新建文件‘SimpleStorage.sol‘
pragma solidity ^0.4.0;
contract SimpleStorage {
    uint storeddata;
    function set(uint x) {
        storeddata = http://www.mamicode.com/x; >

技术分享

 
这个SimpleStorage功能很简单,把一个数乘以7并保存起来,提供获取和设置它的接口.

大家最好手动输入以上内容体验一下Remix的效果.也可以故意输错,看看Remix是如何提醒错误的.

Remix调试模式

第一种是默认模式:javascript vm
第二种,是通过本地私有网络的rpc端口,链接到本地私有网络进行调试
还记得我们第2章中创建的以太坊私有网络节点集群吗?  进入之前的私有网络目录,并使用我们讲过的方法启动节点:
cd ~/Documents/private-chain/
geth --datadir ./data/00 --networkid 314590 --ipcdisable --port 61910 --rpcport 8200 console
回到Remix浏览器Ide中,选中右侧边栏的‘Contact‘选项卡.在‘Execution environment‘选项中,选择:‘web3 provider‘, ide会弹出提示,让你输入之前我们启动的私有网络的rpc地址和端口, 输入: 
http://localhost:8200

确定, 可以看到ide弹出提示:
Invalid JSON RPC response: "" 
我们按照官方提供的方法,增加几个选项,启动命令修改之后如下:
geth --datadir ./data/00 --networkid 314590 --ipcdisable --port 61910 --rpc --rpcapi ‘web3,eth,debug‘ --rpccorsdomain ‘*‘ --rpcport 8200 console
####我们增加了以下几个命令选项:####
--rpc --rpcapi ‘web3,eth,debug‘ --rpccorsdomain ‘*‘
 
然后Remix重新链接私有网络的RPC地址, Idea没有了错误提示. 回到刚才启动的网络节点,在命令行输入命令查看账号:
> eth.accounts ["0x5fba50fce50baf0b8a7314200ba46336958ac97e"]
观察Remix中, 有侧边栏‘Contract‘ tab页中的‘Transaction origin‘,这里显示的就是这个账号地址.  
点击下方的‘create‘按钮.错误又来了:

    callback contain no result Error: authentication needed: password or unlock

提示账号被锁定,需要‘Transaction origin‘中指定的账号的密码,或者解锁账户.
回到geth命令行,执行下面的命令解锁账户:
> personal.unlockAccount("0x5fba50fce50baf0b8a7314200ba46336958ac97e", ‘password of account‘)
true
再次点击‘Create‘, 出现提示:
    

技术分享

并且在geth命令行中,看到了一个提示: 
    > INFO [06-19|00:21:11] Submitted contract creation              fullhash=0x5aaf0021c94f52e37eda6c17cfbea2995c1ad1935816e7cac84c73448bd6ab2d contract=0xa7fdc326fa65a15665e3637ff33a120ff3758cdb
    > 

就像之前我们在第三章中,从一个账户发送以太币到另一个账户,需要调用miner.start(),挖矿来确认我们的交易一样.
这里我们调用一下:
    > miner.start(3); admin.sleepBlocks(1); miner.stop();
    INFO [06-19|00:22:31] Updated mining threads                   threads=3
    INFO [06-19|00:27:08] Successfully sealed new block            number=38 hash=b7fdac…eb3c3f
    INFO [06-19|00:27:08] ?? mined potential block                  number=38 hash=b7fdac…eb3c3f
    INFO [06-19|00:27:08] Commit new mining work                   number=39 txs=0 uncles=0 elapsed=1.836ms
    INFO [06-19|00:27:08] Successfully sealed new block            number=39 hash=210027…1218e0
    INFO [06-19|00:27:08] ?? mined potential block                  number=39 hash=210027…1218e0
    INFO [06-19|00:27:08] Commit new mining work                   number=40 txs=0 uncles=0 elapsed=183.453μs
    true
    > 
    这里这一串语句表示启动3个线程来挖矿,挖到第一个区块就停止.

####查看未确认的交易数量####
    >txpool.status

回到Remix中,我们可以看到,合约已经创建成功了.  

技术分享

其中,有get, multiply, 和set 3个按钮分别对应了SimpleStorage合约中的3个函数,用于测试函数的功能.
下面在set按钮后面的的输入框中,输入参数‘106‘,点击‘set‘按钮.

合约就被调用了.但是任然需要挖矿来确认.继续运行这个命令:
    miner.start(3); admin.sleepBlocks(1); miner.stop();

等到挖矿完成,我们可以在浏览器的Remix中看到合约的set()函数的运行结果.  
点击‘get‘按钮,获取SimpleStorage合约中‘storeddata‘的值,可以看到我们获取到的结果是‘106‘,说明我们的合约没有问题.
另外‘multiply‘,大家可以自行测试.

技术分享

在私有链中部署智能合约

首先创建truffle项目,依次输入以下命令:
    mkdir SimpleStorage
    cd SimpleStorage
    truffle init
以上命令创建并初始化truffle项目,在第4章中有说明, 这里不在重复.

把Remix中的SimpleStorage.sol保存到SimpleStorage/contracts目录中.

增加SimpleStorage的部署
在migrations中增加部署SimpleStorage的功能.在第四章中曾经详细介绍Migration以及合约的部署.这里直接提供代码.  打开migration/2_deploy_contracts.js, 修改为:
var ConvertLib = artifacts.require("./ConvertLib.sol");
var MetaCoin = artifacts.require("./MetaCoin.sol");
var SimpleStorage = artifacts.require("./SimpleStorage.sol")

module.exports = function(deployer) {
  deployer.deploy(ConvertLib);
  deployer.link(ConvertLib, MetaCoin);
  deployer.deploy(MetaCoin);
  deployer.deploy(SimpleStorage);
};
编译
    cd SimpleStorage
    truflle compile

编译之后,我们开始把合约SipleStorage部署到我们的私有链中:
用我们第二章中的方法启动节点:
    > geth --datadir ./data/00 --networkid 314590 --ipcdisable --port 61910 --rpcport 8200 console
节点启动成功后,我们执行部署合约的命令.

    cd SimpleStorage
    truffle migrate

我靠,什么鬼意思...他妈的执行失败,显示下面的信息.  
    Error: Invalid JSON RPC response: ""
        at Object.InvalidResponse (/usr/local/node-v6.10.2-linux-x64/lib/node_modules/truffle/node_modules/web3/lib/web3/errors.js:35:16)

[image023]

查看大量资料后,发现在部署合约到私有链时,需要要注意以下几点:

    1) geth启动节点的时候,必须带有--rpcapi选项,并且选项的值必须包含有:"web3", "net", "eth"这3个设置,另外为了保证rpc为启用,还应该带有"--rpc"等一系列相关的参数.

    2) 注意SimpleStorage项目目录中:SimpleStorage/truffle.js(项目的配置文件),关于"rpcport"的配置必须要与启动节点时,设置的端口号一致.
        module.exports = {
          networks: {
            development: {
              host: "localhost",
              port: 8545,
              network_id: "*" // Match any network id
            }
          }
        };
        我们需要把prot: 8545,修改为geth启动节点时设置的:8200

    3) 启动节点后,节点中默认的账户是被锁定的,无法执行部署合约的操作,需要调用下面的命令解锁账户:

        > personal.unlockAccount(eth.accounts[0],"password", 1000*60*20)
    
        第一个参数是账户地址.
        第二个参数:账户密码.
        第三个参数:账户解锁持续的时间, 以毫秒为单位,这里这样只是为了更方便的设置以分钟为单位(20分钟)
    我们把账户解锁持续时间稍微设置久一点,以免在部署的过程中,账户突然被锁定,导致意外的出现.

有了以上3个需要注意的地方,我们重新执行部署:

调整启动命令后,用下面的命令启动:
    > geth --datadir ./data/00 --networkid 314590 --ipcdisable --port 61910 --rpc --rpcapi ‘web3,eth,net‘ --rpccorsdomain ‘*‘ --rpcport 8200 console
    INFO [06-24|19:15:09] Starting peer-to-peer node               instance=Geth/v1.6.1-stable-021c3c28/linux-amd64/go1.8.1
#####....这里是很多书输出的日志信息....#####
    Welcome to the Geth JavaScript console!

    instance: Geth/v1.6.1-stable-021c3c28/linux-amd64/go1.8.1
    coinbase: 0x5fba50fce50baf0b8a7314200ba46336958ac97e
    at block: 41 (Sat, 24 Jun 2017 00:34:18 CST)
     datadir: /home/zl/Documents/private-chain/data/00
     modules: admin:1.0 debug:1.0 eth:1.0 miner:1.0 net:1.0 personal:1.0 rpc:1.0 txpool:1.0 web3:1.0
#####....启动完成,我们执行解锁账户unlock.....#####
    > personal.unlockAccount(eth.accounts[0], "ko2005", 20*(60*1000))
    true


私有网络节点的操作完成后,回到刚才SimpleStorage所在的命令行, 执行部署命令:
    > truffle migrate    
    Using network ‘development‘.

    Running migration: 1_initial_migration.js
      Deploying Migrations...

可以看到,执行的输出结果与上一次不太一样了,好像离成功更近了.我们继续等一会儿.
[过了一分钟,还是没反应,先来放一首歌....]
[....等啊等....
[草它大爷的,等了快10分钟还是没有反应....]
[.....好像不按我们的套路来啊.....]
[........]

突然想起来一件事,我们在之前的私有网络中,用户之间转账都是要通过挖矿来确认的...
这尼玛..
回到刚才启动的私有网络命令行,执行挖矿:

    > miner.start(3); admin.sleepBlocks(1); miner.stop();
    INFO [06-24|19:16:43] Updated mining threads                   threads=3
    INFO [06-24|19:16:43] Starting mining operation 
    INFO [06-24|19:16:43] Commit new mining work                   number=42 txs=1 uncles=0 elapsed=3.079ms

这行命令的意思: 启动挖矿程序,在挖到一个区块之后,停止挖矿.

[挖矿又是一个耗时间的操作...]

一会儿,私有网络所在的命令行挖矿成功,并输出内容:
一会儿, 部署合约的命令行输出内容:
      Migrations: 0xbc14f4725810403de4b31cdaf00a84e2ea7252dc
    Saving successful migration to network...

同时,查看私有网络的命令行,输出内容: 
    > miner.start(3); admin.sleepBlocks(1); miner.stop();
    INFO [06-24|19:16:43] Updated mining threads                   threads=3
    INFO [06-24|19:16:43] Starting mining operation 
    INFO [06-24|19:16:43] Commit new mining work                   number=42 txs=1 uncles=0 elapsed=3.079ms
    INFO [06-24|19:32:05] Successfully sealed new block            number=42 hash=de460c…aefbfb
    INFO [06-24|19:32:05] ?? mined potential block                  number=42 hash=de460c…aefbfb
    INFO [06-24|19:32:05] Commit new mining work                   number=43 txs=0 uncles=0 elapsed=2.730ms
    true
    > INFO [06-24|19:32:06] Submitted transaction                    fullhash=0xdf799ee7e5981011125dca4e5edcdbbf344dfbc81f466be7e69ba266572e2724 recipient=0xbc14f4725810403de4b31cdaf00a84e2ea7252dc

看这命令行输出"Submitted transaction",意思又的让我们挖矿确认吧, ,老套路,我们懂, 再挖,命令行我就不打出来了.跟之前一样的.
又等一会儿..

SimpleStorage命令行输出: 
    Saving artifacts...

区块链网络命令行输出: 
    > INFO [06-24|19:49:31] Submitted contract creation              fullhash=0xb399d73bb5ed563e4acf6ad0b8ffc5e28ef114a1643df90b6218e568473c0b66 contract=0x6853da42bccdee332f180be1dd1e118e825ac57e

这尼玛套路我们算是明白了,但这尼玛套路也太长了吧. 用周星驰的话来说就是:
我走过最长的路,就是这尼玛以太坊的套路....

反正我就这样一直走下去,最后总算是部署成功了,这就是我的执行结果:

技术分享

合约部署成功后,我们尝试执行以下合约..今天就到这里吧...
这一章我们了解了以下几点:
    1) 基于浏览器端的solidity ide "Remix",以及如何搭建本地版本的"Remix"
    2) Remix 的大概使用方法
    3) 使用Remix部署智能合约到Debug网络中
    4) 如何创建一个简单的truffle 项目, 进行项目创建,编写部署代码, 编译..到执行部署的详细流程.
    5) 在部署智能合约到以太坊私有链时,需要注意的许多地方.
    6) 我们成功部署了一个智能合约在以太坊私有链中....

下一次我们讲一下如何在私有链中调用已经部署好的智能合约....
工作忙的时候,更新会稍慢,大家敬请期待..  

区块链入门(5)Truffle 项目实战,Solidity IDE, 智能合约部署