链码安装和实例化之后就可以调用chaincode执行交易,下面分析简单的账户转账操作是如何完成的。

源码分析

1.首先看app.js的路由函数

app.post('/channels/:channelName/chaincodes/:chaincodeName', async function(req, res) {
var peers = req.body.peers;
var chaincodeName = req.params.chaincodeName;
var channelName = req.params.channelName;
var fcn = req.body.fcn;
var args = req.body.args;
// 此处省略了参数校验
let message = await invoke.invokeChaincode(peers, channelName, chaincodeName, fcn, args, req.username, req.orgname);
res.send(message);
});

2.接下来找到

var invokeChaincode = async function(peerNames, channelName, chaincodeName, fcn, args, username, org_name) {
var error_message = null;
var tx_id_string = null;
try {
var client = await helper.getClientForOrg(org_name, username); // 创建client对象
var channel = client.getChannel(channelName); // 创建channel对象
// 获取交易id:基于client对象上分配的用户身份(_userContext)
var tx_id = client.newTransactionID();
tx_id_string = tx_id.getTransactionID();
// 请求结构
var request = {
targets: peerNames,
chaincodeId: chaincodeName,
fcn: fcn,
args: args,
chainId: channelName,
txId: tx_id
};
// SDK根据request生成proposal,并调用sendPeersProposal()
// 将提案发送给背书节点,然后将返回的提案响应连同交易提案一起打包返回给client
let results = await channel.sendTransactionProposal(request); // 返回的交易提案和提案响应
var proposalResponses = results[0];
var proposal = results[1]; // 检查提案响应是否正确
var all_good = true;
for (var i in proposalResponses) {
let one_good = false;
if (proposalResponses && proposalResponses[i].response &&
proposalResponses[i].response.status === 200) {
one_good = true;
logger.info('invoke chaincode proposal was good');
} else {
logger.error('invoke chaincode proposal was bad');
}
all_good = all_good & one_good;
} if (all_good) {
var promises = [];
// 基于channel的事件中心
let event_hubs = channel.getChannelEventHubsForOrg();
#event_hubs.forEach((eh) => {
#let invokeEventPromise = new Promise((resolve, reject) => {
#let event_timeout = setTimeout(() => {
let message = 'REQUEST_TIMEOUT:' + eh.getPeerAddr();
logger.error(message);
eh.disconnect();
}, 3000);
// 注册交易事件监听,当交易被peer提交到账本中时可以得到反馈
#eh.registerTxEvent(tx_id_string, (tx, code, block_num) => {
clearTimeout(event_timeout);
}, (err) => {
clearTimeout(event_timeout);
logger.error(err);
reject(err);
},
{unregister: true, disconnect: true}
);
eh.connect();
});
promises.push(invokeEventPromise);
});
// 将txID,交易提案和提案响应打包成交易请求
var orderer_request = {
txId: tx_id,
proposalResponses: proposalResponses,
proposal: proposal
};
// 将交易请求发送给orderer节点,内部调用sendBroadcast(envelope)
var sendPromise = channel.sendTransaction(orderer_request);
promises.push(sendPromise);
let results = await Promise.all(promises); let response = results.pop();
if (response.status === 'SUCCESS') {
logger.info('Successfully sent transaction to the orderer.');
} else {
error_message = util.format('Failed to order the transaction. Error code: %s',response.status);
logger.debug(error_message);
} } else {
error_message = util.format('Failed to send Proposal and receive all good ProposalResponse');
logger.debug(error_message);
}
} catch (error) {
logger.error('Failed to invoke due to error: ' + error.stack ? error.stack : error);
error_message = error.toString();
} if (!error_message) {
// 返回交易id
return tx_id_string;
} else {
let message = util.format('Failed to invoke chaincode. cause:%s',error_message);
logger.error(message);
throw new Error(message);
}
};

3.如果我们进行简单的转账交易 A->B,则需要调用链码中的move方法,代码如下:

func (t *SimpleChaincode) move(stub shim.ChaincodeStubInterface, args []string) pb.Response {
// 两个账户A,B
var A, B string
// 账户的余额数
var Aval, Bval int
// 转移的数值
var X int
var err error if len(args) != 3 {
return shim.Error("Incorrect number of arguments. Expecting 4, function followed by 2 names and 1 value")
} A = args[0]
B = args[1] // 获取账户A的值
Avalbytes, err := stub.GetState(A)
if err != nil {
return shim.Error("Failed to get state")
}
if Avalbytes == nil {
return shim.Error("Entity not found")
}
Aval, _ = strconv.Atoi(string(Avalbytes))
// 获取账户B的值
Bvalbytes, err := stub.GetState(B)
if err != nil {
return shim.Error("Failed to get state")
}
if Bvalbytes == nil {
return shim.Error("Entity not found")
}
Bval, _ = strconv.Atoi(string(Bvalbytes)) X, err = strconv.Atoi(args[2])
if err != nil {
return shim.Error("Invalid transaction amount, expecting a integer value")
}
// 余额转移计算
Aval = Aval - X
Bval = Bval + X
logger.Infof("Aval = %d, Bval = %d\n", Aval, Bval) // 将改变后的值写入状态数据库中
err = stub.PutState(A, []byte(strconv.Itoa(Aval)))
if err != nil {
return shim.Error(err.Error())
} err = stub.PutState(B, []byte(strconv.Itoa(Bval)))
if err != nil {
return shim.Error(err.Error())
} return shim.Success(nil);
}

测试

交易:

curl -s -X POST \
http://localhost:4000/channels/mychannel/chaincodes/mycc \
-H "authorization: Bearer <Token>" \
-H "content-type: application/json" \
-d '{
"peers": ["peer0.org1.example.com","peer1.org1.example.com"],
"fcn":"move",
"args":["a","b","10"]
}'

结果:

Transacton ID is 970928c6acbae452a43cbc59e1cb9a558a09c4f354cda0025bd51dafcca3ad96

最新文章

  1. 伪类before和after
  2. Chrome开发工具之Console
  3. [转]CIDR简介
  4. PHP替换,只替换匹配到的第一个
  5. jQuery无缝滚动插件
  6. c/c++常用代码 -- ini文件操作
  7. Qt之进程间通信(QProcess)
  8. 自定义滚动条CSS样式
  9. CentOS 启动提示unexpected inconsistency;RUN fsck MANUALLY
  10. Android学习之RecyclerView
  11. mysql实现full join
  12. Linux显示历史记录
  13. mutex.go
  14. 简单Java类 全网最详细讲解 !!!
  15. hdu-1052(贪心)
  16. MongoDB学习笔记(转)
  17. 基于stm32f427实现SVPWM控制永磁同步开环转动
  18. jq的遍历关系元素方法集合
  19. CentOS 6.5 x64下查找依赖包,或用YUM安装
  20. AttributeError: &#39;int&#39; object has no attribute &#39;isdigit&#39;(python下的isdigit函数)

热门文章

  1. C++-doctest-测试框架
  2. LaTex中文article模板(支持代码、数学、TikZ)
  3. bootstrap-分页-默认分页
  4. centos 7.0运行docker出现内核报错解决方法
  5. 图论--网络流--最大流 HDU 2883 kebab(离散化)
  6. POJ Building a Space Station 最小生成树
  7. Entity Framework使用EntityState和Attach来保存数据变化以及更新实体的个别字段
  8. java :技巧
  9. 实现MapReduce
  10. rsync 服务及部署