Hyperledger Fabric——balance transfer(三)创建和加入Channel
详细解析blance transfer示例的创建通道(Channel)和加入节点到通道的过程。
创建Channel
1.首先看app.js的路由函数
var createChannel = require('./app/create-channel.js');
app.post('/channels', async function(req, res) {
// 接收参数channel名称和配置文件的路径
// 通道配置交易路径: ./artifacts/channel/mychannel.tx
var channelName = req.body.channelName;
var channelConfigPath = req.body.channelConfigPath;
if (!channelName) {
res.json(getErrorMessage('\'channelName\''));
return;
}
if (!channelConfigPath) {
res.json(getErrorMessage('\'channelConfigPath\''));
return;
}
// 创建通道(username和orgname参数从token中获取)
let message = await createChannel.createChannel(channelName, channelConfichannelNamegPath, req.username, req.orgname);
res.send(message);
});
2.再来看create-channel.js
中的createChannel()
方法
// 通过发送交易的方法向orderer节点发送创建channel的请求
var createChannel = async function(channelName, channelConfigPath, username, orgName) {
try {
// 生成组织的客户端实例
var client = await helper.getClientForOrg(orgName);
// 从channel配置交易(mychannel.tx)读取源字节
var envelope = fs.readFileSync(path.join(__dirname, channelConfigPath));
// 提取channel配置信息
var channelConfig = client.extractChannelConfig(envelope);
// 客户端使用 组织管理员 的私钥 签名,作为背书(endorsement)
// "../artifacts/channel/crypto-config/peerOrganizations/org1.example.com/users/Admin@org1.example.com/msp/signcerts"
// 在内部通过_getSigningIdentity(true)获取org admin的私钥 进行签名
let signature = client.signChannelConfig(channelConfig);
// 请求结构:
let request = {
config: channelConfig,
signatures: [signature],
name: channelName,
txId: client.newTransactionID(true)
};
// 发送给orderer节点
var response = await client.createChannel(request)
if (response && response.status === 'SUCCESS') {
logger.debug('Successfully created the channel.');
let response = {
success: true,
message: 'Channel \'' + channelName + '\' created Successfully'
};
return response;
} else {
throw new Error('Failed to create the channel \'' + channelName + '\'');
}
} catch (err) {
logger.error('Failed to initialize the channel: ' + err.stack ? err.stack : err);
throw new Error('Failed to initialize the channel: ' + err.toString());
}
};
加入节点到Channel
1.首先看app.js里的路由函数
// 首先看app.js中的路由函数
app.post('/channels/:channelName/peers', async function(req, res) {
// 接收参数 channel名称(URL中)和peers(body中)
var channelName = req.params.channelName;
var peers = req.body.peers;
if (!channelName) {
res.json(getErrorMessage('\'channelName\''));
return;
}
if (!peers || peers.length == 0) {
res.json(getErrorMessage('\'peers\''));
return;
}
// 将peers加入channel
let message = await join.joinChannel(channelName, peers, req.username, req.orgname);
res.send(message);
});
2.然后看join-channel.js中的joinChannel()函数
// 然后看join-channel.js中的joinChannel()函数
var joinChannel = async function(channel_name, peers, username, org_name) {
var error_message = null;
var all_eventhubs = [];
try {
// 首先创建client对象
var client = await helper.getClientForOrg(org_name, username);
// 生成channel对象
var channel = client.getChannel(channel_name);
if(!channel) {
let message = util.format('Channel %s was not defined in the connection profile', channel_name);
logger.error(message);
throw new Error(message);
}
// 生成基于组织管理员的 TransactionID 对象
// 该对象包含一个随机数(nonce)以及t(nonce + signingIdentity)的hash值
let request = {
txId : client.newTransactionID(true)
};
// 从orderer节点获取创世区块(genesis block)
let genesis_block = await channel.getGenesisBlock(request);
// 告诉节点加入channel,等待每个节点的event hub通知我们节点加入channel成功
var promises = [];
var block_registration_numbers = [];
// 给组织中的每个节点分配一个EventHub对象
let event_hubs = client.getEventHubsForOrg(org_name);
#event_hubs.forEach((eh) => {
#let configBlockPromise = new Promise((resolve, reject) => {
// 设置超时时间
#let event_timeout = setTimeout(() => {
let message = 'REQUEST_TIMEOUT:' + eh._ep._endpoint.addr;
logger.error(message);
eh.disconnect();
reject(new Error(message));
}, 60000);
// 注册区块监听
#let block_registration_number = eh.registerBlockEvent((block) => {
clearTimeout(event_timeout);
// 配置区块中只能有一个交易
if (block.data.data.length === 1) {
// 一个节点可能属于多个channel
// 所以需要检验该区块是否来自目标channel
var channel_header = block.data.data[0].payload.header.channel_header;
if (channel_header.channel_id === channel_name) {
let message = util.format('EventHub %s has reported a block update for channel %s',eh._ep._endpoint.addr,channel_name);
logger.info(message)
resolve(message);
} else {
let message = util.format('Unknown channel block event received from %s',eh._ep._endpoint.addr);
logger.error(message);
reject(new Error(message));
}
}
}, (err) => {
clearTimeout(event_timeout);
let message = 'Problem setting up the event hub :'+ err.toString();
logger.error(message);
reject(new Error(message));
});
// 每一个client实例对应一个注册number,后面将作为参数对监听进行注销
block_registration_numbers.push(block_registration_number);
// 保存EventHub对象,方便后面对事件流断开连接
all_eventhubs.push(eh);
});
promises.push(configBlockPromise);
// 开启事件流
eh.connect();
});
// 封装加入channel的请求
let join_request = {
targets: peers, // 要加入channel的节点
txId: client.newTransactionID(true), // 基于组织管理员的TransactionID
block: genesis_block // 创世区块
};
// 调用SDK中的joinChannel()方法,主要是通过sendProposal()将
// 加入channel的交易提案发送给背书节点进行背书
let join_promise = channel.joinChannel(join_request);
// 保存返回结果:提案响应(ProposalResponse)的Promise
promises.push(join_promise);
let results = await Promise.all(promises);
logger.debug(util.format('Join Channel RESPONSE : %j', results));
// 检查所有Promise返回(包括监听事件和发送join请求)
// 只要有一个结果异常则宣布join channel失败
let peers_results = results.pop();
for(let i in peers_results) {
let peer_result = peers_results[i];
if(peer_result.response && peer_result.response.status == 200) {
logger.info('Successfully joined peer to the channel %s',channel_name);
} else {
let message = util.format('Failed to joined peer to the channel %s',channel_name);
error_message = message;
logger.error(message);
}
}
// 查看事件中心的消息报告
for(let i in results) {
let event_hub_result = results[i];
let event_hub = event_hubs[i];
let block_registration_number = block_registration_numbers[i];
logger.debug('Event results for event hub :%s',event_hub._ep._endpoint.addr);
if(typeof event_hub_result === 'string') {
logger.debug(event_hub_result);
} else {
if(!error_message) error_message = event_hub_result.toString();
logger.debug(event_hub_result.toString());
}
// 注销事件监听
event_hub.unregisterBlockEvent(block_registration_number);
}
} catch(error) {
logger.error('Failed to join channel due to error: ' + error.stack ? error.stack : error);
error_message = error.toString();
}
// 断开所有event hub事件流
all_eventhubs.forEach((eh) => {
eh.disconnect();
});
};
测试
创建channel
curl -s -X POST \
http://localhost:4000/channels \
-H "authorization: Bearer $ORG1_TOKEN" \
-H "content-type: application/json" \
-d '{
"channelName":"mychannel",
"channelConfigPath":"../artifacts/channel/mychannel.tx"
}'返回结果
{"success":true,"message":"Channel 'mychannel' created Successfully"}
将Org1中的两个peer加入channel
curl -s -X POST \
http://localhost:4000/channels/mychannel/peers \
-H "authorization: Bearer $ORG1_TOKEN" \
-H "content-type: application/json" \
-d '{
"peers": ["peer0.org1.example.com","peer1.org1.example.com"]
}'返回结果
{"success":true,"message":"Successfully joined peers in organization Org1 to the channel:mychannel"}
总结
创建channel:
- 生成
client
对象,从channel配置交易mychannel.tx
中提取配置信息 - 利用组织
admin
的私钥进行签名,这个过程相当于背书 - 封装请求结构(包括channel配置信息,对配置信息的签名,channelName,基于admin的txId), 并调用
clinet.createChannel()
将请求发送到orderer节点 - 实际上在内部调用
orderer.sendBroadcast()
,向order service
广播交易,orderer
将配置交易打包成区块后广播给各记账节点,记入账本中。(这个配置区块就是创建的channel所对应的chain的创世区块)
加入channel:
- 生成
client
对象,生成channel
对象 - 从
orderer
节点获取genesis block
(调用orderer.sendDeliver()
接口) - 对要加入
channel
的每个peer设置事件区块监听,根据指定channel是否生成了新的配置区块来判断join channel
过程是否成功 - 封装请求结构(包括 目标peers,基于组织admin的txId,genesis block),调用
channel.joinChannel()
方法,内部是通过peer.sendProposal()
将该交易提案发送给背书节点进行背书,返回背书响应Promise
客户端将背书过的交易发送到orderer服务,排序后广播给记账节点,将该配置区块记录到ledger中,此时EventHub
监听到该事件。 - 加入channel成功后,注销区块事件监听,断开所有
event hub
事件流。
Hyperledger Fabric——balance transfer(三)创建和加入Channel的更多相关文章
- Hyperledger Fabric——balance transfer(一)启动示例
Blacne transfer是Hyperledger fabric Node SDK的一个示例应用,主要使用了SDK中fabric-client 和 fabric-ca-client 模块中的API ...
- Hyperledger Fabric——balance transfer(六)查询
balance transfer 提供了很多查询接口,包括链码查询,根据区块号查询区块数据,根据交易ID查询交易信息,查询链上的区块数,查询已安装或已实例化的链码,查询通道. 源码解析 1.调用链码查 ...
- Hyperledger Fabric——balance transfer(四)安装和实例化chaincode
详细解析blance transfer示例的安装(install)和实例化(Instantiate)链码(chaincode)的过程.安装chaincode会根据本地的链码文件生成chaincode镜 ...
- Hyperledger Fabric——balance transfer(二)注册用户
详细分析blance transfer示例的用户注册(register)与登录(enroll)功能. 源码分析 1.首先分析项目根目录的app.js文件中关于用户注册和登录的路由函数.注意这里的tok ...
- Hyperledger Fabric——balance transfer(五)执行交易
链码安装和实例化之后就可以调用chaincode执行交易,下面分析简单的账户转账操作是如何完成的. 源码分析 1.首先看app.js的路由函数 app.post('/channels/:channel ...
- Hyperledger Fabric(v1.2.0)代码分析1——channel创建
Hyperledger Fabric(v1.2.0)代码分析1--channel创建 0. e2e_cli Hyperledger Fabric提供了一个e2e的例子,该例中创建了一个基础的区块链网络 ...
- HyperLedger Fabric Introduction——区块链超级账本介绍
介绍 HyperLedger Fabric是一个基于模块化架构的分布式账本解决方案平台,它拥有深度加密.便捷扩展.部署灵活及可插拔等特性.它设计之初的目的是支持不同组件的可插拔实现,并适应整个经济生态 ...
- Hyperledger Fabric介绍
转载地址 https://blog.csdn.net/xiaonu123/article/details/81006936 简介 Hyperledger介绍 超级账本(Hyperledger)项目是首 ...
- 搭建基于hyperledger fabric的联盟社区(三) --生成公私钥证书及配置文件
一.生成公私钥和证书 Fabric中有两种类型的公私钥和证书,一种是给节点之前通讯安全而准备的TLS证书,另一种是用户登录和权限控制的用户证书.这些证书本来应该是由CA来颁发,但是目前只有两个社区,所 ...
随机推荐
- 【Linux常见命令】tee命令
tee - read from standard input and write to standard output and files tee命令用于读取标准输入的数据,并将其内容输出成文件. t ...
- Flask中可以利用Flask-SQLAlchemy
为什么80%的码农都做不了架构师?>>> 官方文档:http://flask-sqlalchemy.pocoo.org/2.3/ 1.安装(进入虚拟环境)--利用镜像安装PyMy ...
- 前端程序员难翻身,没有好的学习方法,你永远无法成功,vue.js专题
学习vue正确思路,是先学vue-cli,再学vue.js单文件引用的用法,这样会在极短时间内撤底撑握vue, 如果先学vue.js单文件用法,再去学vue-cli4,可以说是重新学vue,,,,难处 ...
- INTERVIEW #2
吐槽下ZZ的面试安排:面试时间12:30不说了,周围没有饭店,中午就没吃饭...不像其他公司给每个人安排不同的面试时间,这样可以节约大家的时间,SPDB是把一大批人都安排在了12:30,而且面试是5个 ...
- Jmeter 数据库测试参数化
1.JDBC Request 参数化 方法一.Jmeter 参数化,在 sql query 中使用变量 Jmeter 参数化,使用 csv 参数化 sql query 中使用 ${变量名} 引用 方法 ...
- muduo网络库源码学习————线程特定数据
muduo库线程特定数据源码文件为ThreadLocal.h //线程本地存储 // Use of this source code is governed by a BSD-style licens ...
- Python监控文件夹 && 发送邮件
直接上代码: # python3 # -*- coding: utf-8 -*- # 2017/06/16 by luohan from email.mime.text import MIMEText ...
- 软件——IDEA主题美化
前言 IntelliJ IDEA主要用于支持 Java.Scala.Groovy 等语言的开发工具,同时具备支持目前主流的技术和框架,擅长于企业应用.移动应用和 Web 应用的开发. IntelliJ ...
- django3开发完整博客带评价
纯django开发最完美博客 2020年5月打造最时尚博客系统教程 为了学习速度,集中精力学习django和博客开发, 没有使用其它框架,也没有使用css预处理等 这样学起来最方便, 博客前后端都完成 ...
- 怎么成为一名WEB前端开发工程师
对于刚开始学的人来说,web 就是HTML+CSS+JavaScript其实我们是可以这样理解的.web工程师负责或参与Web产品的页面开发,包含PC端.移动APP内嵌移动端.微信小程序.web ...