这篇博客演示的基本操作系统环境是CentOS 7,参考书籍:以太坊开发实战——以太坊关键技术与案例分析 第六章(吴寿鹤、冯翔、刘涛、周广益 著)。要运行本博文相关程序,需要搭建以太坊私有链、安装testrpc
文章结构:
一.安装与启动web3.js
二.账户相关API
1.查看所有可用账户
2.查看账户余额
三、交易相关API
1.账户之间转账交易
2.发送一个已经签名的转账交易
3.调用合约
$ npm install --save [email protected]
首先在一个终端中运行testrpc以启动测试链,然后在另一个终端,编写相关js文件。这里编写test.js文件如下:
var Web3 = require("web3");
//创建web对象
var web3 = new Web3();
//连接到以太坊节点
web3.setProvider(new Web3.providers.HttpProvider('http://localhost:8545'));
//列出所有的可用用户
var accounts = web3.eth.accounts;
console.info(accounts);
//编写完成后执行如下命令运行js文件
[root@localhost geth]# node test.js
//以下为输出结果,即所有可用账户
[ '0x623053f6e32ea1740ab791fae0e2e402b367e501',
'0xc8f4672e8e988df4d5f6ee9c1322eba005583267',
'0x242e2ad79192f5f28c13d6d9ee57976ef346d63f',
'0x88304656b1031469af4412df4b2de8b2c7759385',
'0x55b0d4de3900b00c7ac1c5ede911a832c3f0ec91',
'0xaf767e3d3da4b8bcba5ca9978d5ca4a7bdf6532f',
'0x7ce1fd31c404682eba0d7fd33d8635c47ee2cc4b',
'0x40023a7f603fd4fc9675c1670b249d054ae33eef',
'0xa723cea532564ded55ad8e6256fb016d25cf18b5',
'0x49ab097da3c5089764e0d5c4117f950263274c0b' ]
//而在testrpc终端下,可以看到对测试链的操作:eth_accounts
…………
…………
HD Wallet
==================
Mnemonic: penalty silk correct trial good drama action arctic drip rebel weather level
Base HD Path: m/44'/60'/0'/0/{account_index}
Listening on localhost:8545
eth_accounts
1.查看所有可用账户
var accounts = web3.eth.accounts;
console.info(accounts);
2.查看账户余额
//查看accounts[0]账户的余额,返回的余额是以wei为单位表示的
var balance_1 = web3.eth.getBalance(web3.eth.accounts[0]);
//将余额用ether为单位表示
var balance_eth = web3.fromWei(balance_1,"ether");
console.log(balance_eth.toString());
在以太坊中两个账户之间的转账操作可以当作时一个交易,同样调用智能合约中的函数也可以当作时一个交易。
1.账户之间转账交易
在转账之前如果我们不知道需要给交易设置多少gas,可以通过下面的语句估算并本次交易大概需要消耗的gas:
var estimate_gas = web3.eth.estimateGas({
from: web3.eth.accounts[0],
to: web3.eth.accounts[1],
value: web3.toWei(1,"ether")
});
console.log(estimate_gas);//21000
一般简单的转账交易通常会花费21000gas,得到gas后便可以构建一个转账交易了。转账交易组成部分:
//开始转账操作
var txId = web3.eth.sendTransaction({
from: web3.eth.accounts[0],
to: web3.eth.accounts[1],
value: web3.toWei(1,"ether"),
gas: estimate_gas
});
//返回值为交易hash
console.log(txId);//0x24ab2da77fe1cc65e75a7b9e8d51b058becd4fd8a3a67b765974bf2f692bd898
testrpc终端显示内容:
Transaction: 0x24ab2da77fe1cc65e75a7b9e8d51b058becd4fd8a3a67b765974bf2f692bd898
Gas usage: 21000
Block Number: 1
Block Time: Sat Mar 30 2019 17:44:04 GMT+0800 (Hong Kong Standard Time)
//查询转账交易详情
var tx = web3.eth.getTransaction(txId);
console.log(tx);
//输出结果
{ hash:
'0x498c3835a3eeeb8d9a8f398aa036cd6c59a81bdd5af2390bd518bb9aa7622c70',
nonce: 1,
blockHash:
'0x51b5cb1aa7a1258070ba2586c57f224ead9fd10c745af802f018ea56a41e09a1',
blockNumber: 2,
transactionIndex: 0,
from: '0x623053f6e32ea1740ab791fae0e2e402b367e501',
to: '0xc8f4672e8e988df4d5f6ee9c1322eba005583267',
value: BigNumber { s: 1, e: 18, c: [ 10000 ] },
gas: 21000,
gasPrice: BigNumber { s: 1, e: 0, c: [ 1 ] },
input: '0x0' }
//查询交易收据详情
var tx_receipt = web3.eth.getTransactionReceipt(txId);
console.log(tx_receipt);
//输出结果
{ transactionHash:
'0x498c3835a3eeeb8d9a8f398aa036cd6c59a81bdd5af2390bd518bb9aa7622c70',
transactionIndex: 0,
blockHash:
'0x51b5cb1aa7a1258070ba2586c57f224ead9fd10c745af802f018ea56a41e09a1',
blockNumber: 2,
gasUsed: 21000,
cumulativeGasUsed: 21000,
contractAddress: null,
logs: [],
status: 1 }
2.发送一个已经签名的转账交易
上面发起交易的帐号都存储在我们连接的以太坊节点中,如果我们自己有一个帐号,并且这个帐号不存储在我们连接的以太坊节点中,那么我们就需要通过web3.eth.sendRawTransaction方法完成交易的签名与发送。
在后面操作中,要给刚刚创建的交易rawTx加上签名时,需要使用另外一个库:
$ npm install --save [email protected]
建立test.js,编辑如下内容,完成后在test.js同一目录下,执行node test.js来加载运行。
//引入web3
var Web3 = require("web3");
//创建web3对象
var web3 = new Web3();
//链接到以太坊节点
web3.setProvider(new Web3.providers.HttpProvider('http://localhost:8545'));
/** (1)创建待发送的交易 */
var address = "0xeaf9e84c229c9b1023dd9f8315c01045dd7b741d";
var nonce = web3.eth.getTransactionCount(address,'pending');
var amount = web3.toWei(1,"ether");
var rawTx = {
from:address,
to:"0xb4f58a932a1cadf638f8e0eafce9bda71144c56e",
value:web3.toHex(amount),
nonce:web3.toHex(nonce),
gasLimit:web3.toHex("49674")
}
/**(2).给交易添加签名 */
//引入'ethereumjs-tx'包
var Tx = require('ethereumjs-tx');
//创建raw transaction
var tx = new Tx(rawTx);
//from地址的私钥
var privateKey = new Buffer('9ff1ba4ce83afdfb326fb68b00997f93277d3fd4c606fd61b5c5484bed92d369','hex');
//使用私钥给rawTx签名
tx.sign(privateKey);
//签名后的交易
var serializedTx = tx.serialize();
//控制台输出签名后交易信息
console.log(serializedTx.toString('hex'));
/**(3)发送已签名交易*/
//发送成功后会返回一个交易的hash值
web3.eth.sendRawTransaction('0x' + serializedTx.toString('hex'), function(err,hash) {
if (!err) {
console.log(hash);
} else {
console.log(err);
}
});
运行成功后,控制台返回如下内容,即签名后的交易信息和交易发送成功后的交易hash值
f867808082c20a94b4f58a932a1cadf638f8e0eafce9bda71144c56e880de0b6b3a7640000801ba04538ca57dd5b47173e58a4fe6868408ddeb2e6832524517c423e9e421e4bb736a01b4cd69f9519549bcf53b87edb364958aed0846194ee820fbf34affa7bf20445
0x9b1ef283db7211968b76520466942ae417cc1ac18ec761202a01065e71c768a3
3.调用合约
在以太坊中可以通过交易调用智能合约中的函数,首先编写要使用的只能合约:
pragma solidity >=0.4.21 <0.6.0;
contract Storage {
uint256 public storedData;
function set(uint256 data) public {
storedData = data;
}
function get() public returns (uint256) {
return storedData;
}
}
然后把合约部署到以太坊网络上,合约部署成功后会得到三个比较重要的东西:
(1).合约地址:"0xb69ff7afae8e014f76a4ad1a693076f9197620f8"
(2).合约的接口描述文件:
abi: [{
constant: true,
inputs: [],
name: "storedData",
outputs: [{...}],
payable: false,
stateMutability: "view",
type: "function"
}, {
constant: false,
inputs: [{...}],
name: "set",
outputs: [],
payable: false,
stateMutability: "nonpayable",
type: "function"
}, {
constant: true,
inputs: [],
name: "get",
outputs: [{...}],
payable: false,
stateMutability: "view",
type: "function"
}]
(3).合约的签名,合约函数的获取方式如下:
//将合约的get()方法经过sha3计算后,取除了0x外的前面的8位
> var get_func_sign = web3.sha3('get()').substr(2,8);
//将合约的set(uint256)方法经过sha3计算后,取除了0x外的前面的8位
> var set_func_sign = web3.sha3('set(uint256)').substr(2,8);
//查看函数签名
> set_func_sign
"60fe47b1"
> get_func_sign
"6d4ce63c"
目前有两种方法可以调用合约:
这里介绍第二种方式,首先构建一个调用get方法的交易:
var tx_get = {
"from": "0x8c7ae59ab7e5d510ae3f09a9544978f50315b5f5",
"to": "0xb69ff7afae8e014f76a4ad1a693076f9197620f8",
"data": "0x" + "6d4ce63c" + "0000000000000000000000000000000000000000000000000000000000000000"
};
执行交易:
> var result = web3.eth.call(tx_get);
undefined
> console.info(result);
0x
执行完成后返回的是0,因为还没有给storedData设置值。所以需要调用set方法给storedData设置一个值,首先需要构建一个set方法的交易,并设置set函数传入值为1。
var tx_set = {
"from": "0x8c7ae59ab7e5d510ae3f09a9544978f50315b5f5",
"to": "0xb69ff7afae8e014f76a4ad1a693076f9197620f8",
"gaslimit": web3.toHex("50000"),
"data": "0x" + "60fe47b1" + "0000000000000000000000000000000000000000000000000000000000000001"
};
//执行交易,(需要解锁账户、挖矿)返回交易hash
> var txId = web3.eth.sendTransaction(tx_set);
> txId
"0xb20213f7e852a619a81abe0019da3c998ea84ac94953443b5afb6903044584a7"
> var result = web3.eth.call(tx_get);
console.info(result);
//正确返回结果应该为1