用Docker容器开发区块链应用
很 多刚接触区块链的开发者面临的问题,是如何上手实践基础的区块链应用开发技术,如何了解它的关键技术实践点。本文介绍如何采用Docker容器技术,快速 构建私有节点的比特币测试网络(bitcoin-testnet),并结合Node.js程序语言例子,说明如何调用比特币钱包节点提供的RPC接口服 务,实现涉及比特币区块链的具体应用功能。
本 文采用Ubuntu14.04 Desktop 操作系统作为基础环境。对于常用的Microsoft Windows7 64bit桌面操作系统,可以安装Oracle VM VirtualBox虚拟机软件来进一步安装Ubuntu14.04 Desktop版操作系统。系统安装成功后,缺省带有Node.js软件,如果要查询是否已安装Node.js及相应版本的命令参考如下:
$ apt-cache policy nodejs
$ node -v
$ npm -v
如果尚未安装Node.js,可以参考下述命令进行安装:
$sudo add-apt-repository ‘deb https://deb.nodesource.com/node trusty main’
$sudo apt-get update
$sudo apt-get install nodejs
确认已安装Node.js后,可以安装对应的RPC支持库。常用的Node.js的RPC支持库有多个,我们这里选用开源项目kapitalize。
在Ubuntu终端命令行界面输入以下命令进行安装:
$npm install kapitalize
关于kapitalize开源项目的更多说明可以参考下述网址:
https://github.com/shamoons/Kapitalize
Docker是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的 Linux 机器上。
在Ubuntu14.04操作系统上快速安装Docker运行环境的方法如下:
1)在命令行下,输入下述命令安装Docker容器支持软件。
$sudo apt-get install docker.io
2)创建软连接。
$sudo ln -sf /usr/bin/docker.io /usr/local/bin/docker
3)查看Docker版本。
$docker –version
如看到提示信息类似“Docker version 1.6.2, build 7c8fca2”则说明你已经安装Docker成功了。
1)下载比特币测试网络(bitcoin-testnet)的Docker镜像。
$sudo docker pull freewil/bitcoin-testnet-box
2)运行Docker镜像。
$sudo docker run -t -i -p 19001:19001 -p 19011:19011 freewil/bitcoin-testnet-box
注意:上述命令中的19001和19011是配置给两个节点提供RPC服务的端口。
3)进入Docker运行环境后,输入下面的命令来启动比特币测试网络:
$ make start
启动成功后,将在本机模拟运行两个比特币测试钱包节点,组成一个私有范围的比特币测试网络。
输入下面的命令可以查看测试网络节点状态信息,从中可以了解到比特币测试网络的配置和运行状态,比如协议版本、区块链长度和挖矿计算难度等内容,具体可以通过网络或技术书籍进一步查询了解:
$ make getinfo
4)初始化和测试区块链数据。
在Docker运行窗口里依次输入下面的命令来初始化创建基本的区块链数据,供进一步的程序示例来使用。
make generate //模拟新产生1个区块记录
make generate BLOCKS=200 //模拟新产生200个区块记录
5)查看最新的钱包状态包括 balance 余额信息,这时可以留意看到第一个钱包节点的账户余额发生了变动,新的余额即通过模拟区块挖矿产生的测试比特币。
make getinfo
6)给作为示例的测试钱包地址转账10个BTC。
make sendfrom1 ADDRESS=mkiytxYA6kxUC8iTnzLPgMfCphnz91zRfZ AMOUNT=10
注意:这里的示例地址mkiytxYA6kxUC8iTnzLPgMfCphnz91zRfZ是比特币测试网络下的钱包地址(以字符m起始),与比特币正式网络下的钱包地址(一般以数字1或3起始)是有区别的。
7)模拟新产生10个区块记录,让上面的转账交易得到足够有效的确认。
make generate BLOCKS=10
8)查看最新的钱包状态包括余额信息, 这时可以留意看到第一个钱包节点的balance 账户余额发生了改变 ,差额部分即已经交易转账支出的10个BTC加上少许的矿工费用。
make getinfo
下文以Node.js开发语言为例,演示如何调用RPC接口来执行一些最基本的操作,比如导入比特币私钥,和发送一个最简单的转账交易。开发者通过该示例可以了解到最基本的区块链开发方法。
示例源码(含注释)可以从下述网址下载:
http://ppkpub.org/sample/RpcTestnet.js
然后在命令行下输入以下命令即可运行并看到运行结果:
node RpcTestnet.js
注意:每运行一次测试代码后,都需要到Docker运行环境的命令行下输入”make generate BLOCKS=10“,模拟产生新的区块记录,让测试代码产生的交易记录得到有效的确认。
在 理解此简单示例程序的基础上,经过对Bitcoin协议的进一步了解,我们可以调用RPC接口进一步开发出更复杂功能,如自行构建特定交易数据包(比如备 注信息、多重签名输出等)来满足特定业务需求。欲进一步深入学习研究区块链应用开发的朋友,可以从网上搜索相关学习资料,也可以参看文末推荐的《区块链技术指南》一书。
PPk 开放小组(The PPk Public Group)是一个开放的网络技术极客小组,集合了一群对比特币等加密货币感兴趣的P2P技术爱好者,小组成员多具有10多年以上通信和互联网行业技术研 发从业背景,对于互联网业态的发展趋势有着独立判断和独特理念,关注其以区块链为代表的底层技术的潜在价值,并尝试融合区块链、IPFS等创新P2P技术 来定义个实现一些中性、开放、开源的基础协议和工具集。
电子邮箱: ppkpub@gmail.com
互联网站: ppkpub.org
更多区块链的技术细节,包括比特币、以太坊、超级账本、共识算法、侧链、闪电网络、比特币开发技术、以太坊智能合约开发等等,请参考邹均博士等作者合著的新书《区块链技术指南》,机械工业出版社:
RpcTestnet.js
//************************************************//
// Bitcoin-Testnet RPC sample of node.js //
// PPk Public Group ? 2016. //
// http://ppkpub.org //
// Released under the MIT License. //
//************************************************//
//对应比特币测试网络(Bitcoin testnet)的RPC服务接口访问参数
var RPC_USERNAME='admin1';
var RPC_PASSWORD='123';
var RPC_HOST="127.0.0.1";
var RPC_PORT=19001;
//测试使用的钱包地址
TEST_ADDRESS='mkiytxYA6kxUC8iTnzLPgMfCphnz91zRfZ'; //测试用的钱包地址,注意与比特币正式地址的区别
TEST_PRIVATE_KEY='cTAUfueRoL1HUXasWdnETANA7uRq33BUp3Sw88vKZpo9Hs8xWP82'; //测试用的钱包私钥
TEST_WALLET_NAME='TestWallet1'; //测试的钱包名称
MIN_DUST_AMOUNT=10000; //最小有效交易金额,单位satoshi,即0.00000001 BTC
MIN_TRANSACTION_FEE=10000; //矿工费用的最小金额,单位satoshi
console.log('Hello, Bitcoin-Testnet RPC sample.');
console.log(' PPk Public Group ? 2016 ');
//初始化访问RPC服务接口的对象
var client = require('kapitalize')()
client
.auth(RPC_USERNAME, RPC_PASSWORD)
.set('host', RPC_HOST)
.set({
port:RPC_PORT
});
//显示当前连接的比特币测试网络信息
client.getInfo(function(err, info) {
if (err) return console.log(err);
console.log('Info:', info);
});
//查看当前钱包下属地址账户余额变动情况
client.listaccounts(function(err, account_list) {
if (err) return console.log(err);
console.log("Accounts list:\n", account_list);
});
//检查测试帐号是否已存在于测试节点
client.getaccount(TEST_ADDRESS,function(err, result) {
if (err || result!=TEST_WALLET_NAME ) { //如不存在,则新导入测试帐号私钥
console.log('Import the test account[',TEST_WALLET_NAME,']:',TEST_ADDRESS);
client.importprivkey(TEST_PRIVATE_KEY,TEST_WALLET_NAME,function(err, imported_result) {
if (err) return console.log(err);
console.log('Imported OK:', imported_result);
doSample();
});
}else{ //如已存在,则直接执行示例
console.log('The test account[',TEST_WALLET_NAME,'] existed. Address:',TEST_ADDRESS);
doSample();
}
});
// 示例实现功能
function doSample(){
//获取未使用的交易(UTXO)用于构建新交易的输入数据块
client.listunspent(6,9999999,[TEST_ADDRESS],function(err, array_unspent) {
if (err) return console.log('ERROR[listunspent]:',err);
console.log('Unspent:', array_unspent);
var array_transaction_in=[];
var sum_amount=0;
for(var uu=0;uu<array_unspent.length;uu++){
var unspent_record=array_unspent[uu];
if(unspent_record.amount>0){
sum_amount+=unspent_record.amount*100000000; //注意:因为JS语言缺省不支持64位整数,此处示例程序简单采用32位整数,只能处理交易涉及金额数值不大于0xFFFFFFF即4294967295 satoshi = 42.94967295 BTC。 实际应用程序需留意完善能处理64位整数
array_transaction_in[array_transaction_in.length]={"txid":unspent_record.txid,"vout":unspent_record.vout};
if( sum_amount > (MIN_DUST_AMOUNT+MIN_TRANSACTION_FEE) )
break;
}
}
//确保新交易的输入金额满足最小交易条件
if (sum_amount<MIN_DUST_AMOUNT+MIN_TRANSACTION_FEE) return console.log('Invalid unspent amount');
console.log('Transaction_in:', array_transaction_in);
//生成测试新交易的输出数据块,此处示例是给指定目标测试钱包地址转账一小笔测试比特币
//注意:输入总金额与给目标转账加找零金额间的差额即MIN_TRANSACTION_FEE,就是支付给比特币矿工的交易成本费用
var obj_transaction_out={
"mieC38pnPwMqbMAN6sGWwHRQ3msp7nRnNz":MIN_DUST_AMOUNT/100000000, //目标转账地址和金额
"mkiytxYA6kxUC8iTnzLPgMfCphnz91zRfZ":(sum_amount-MIN_DUST_AMOUNT-MIN_TRANSACTION_FEE)/100000000 //找零地址和金额,默认用发送者地址
};
console.log('Transaction_out:', obj_transaction_out);
//生成交易原始数据包
client.createrawtransaction(array_transaction_in,obj_transaction_out,function(err2, rawtransaction) {
if (err2) return console.log('ERROR[createrawtransaction]:',err2);
console.log('Rawtransaction:', rawtransaction);
//签名交易原始数据包
client.signrawtransaction(rawtransaction,function(err3, signedtransaction) {
if (err3) return console.log('ERROR[signrawtransaction]:',err3);
console.log('Signedtransaction:', signedtransaction);
var signedtransaction_hex_str=signedtransaction.hex;
console.log('signedtransaction_hex_str:', signedtransaction_hex_str);
//广播已签名的交易数据包
client.sendrawtransaction(signedtransaction_hex_str,false,function(err4, sended) { //注意第二个参数缺省为false,如果设为true则指Allow high fees to force it to spend,会在in与out金额差额大于正常交易成本费用时强制发送作为矿工费用(谨慎!)
if (err4) return console.log('ERROR[sendrawtransaction]:',err4);
console.log('Sended TX:', sended);
client.listaccounts(function(err, account_list) {
if (err) return console.log(err);
console.log("Accounts list:\n", account_list); //发送新交易成功后,可以核对下账户余额变动情况
});
});
});
});
});
}