编写你的第一个hyperledger fabric network程序

Writing Your First Application
编写你的第一个程序。
本文档的目标是使阅读本文档的人能基于HyperLedger Fabric Network编写自己的第一个程序。
在这个程序里,我们提供最基本的功能: 用户能在账本中查询一条数据或者用户更新账本中的一条数据。

我们的程序是使用JAVASCRIPT编写,运行在NODEJS平台上。 此文档将会指导你通过三步去编写你的第一个程序。
第一步:
  运行一个HyperLedger Fabric BlockChain Network。
  我们需要一些基础组件去查询和更新账本。 一个通道节点(peer node), 排序节点(Ordering node)和认证授权中心(Certificate Authority )--作为我们的网络支撑。
  我们还需要一个CLI容器来执行管理命令。 一个单独的脚本将会下载和运行这个测试网络。
第二步:
  学习我们的程序将要使用的智能合约的参数。
  我们的智能合约允许我们使用多种不同的方法去操作账本(Ledger)。 例如: 我们可以读取数据的整体或者是一部分(we can
                   read data holistically or on a more granular level.)。
第三步:
  部署程序去查询和更新数据。
  我们提供两种简单的程序 ---- 一个查询账本,另一个更新账本。 我们的程序将会使用超级账本的SDK API 去影响 HyperLedger.
  
  在完成本教程之后,你应该对如何使用HyperLedger Fabric Network提供的NODEJS的API去编写自己的程序有一个基本的理解。
  
首先,让我们运行我们的HyperLedger。
Getting a Test Network
获取一个测试网络。
  确保你已经安装了所有必须的依赖软件。
  首先决定你要克隆Fabric-samples到哪一个目录,然后运行克隆脚本,随后进入fabric的子目录。
  
  git clone https://github.com/hyperledger/fabric-samples.git
  cd fabric-samples/fabcar
  
  这个子目录(fabcar)包含脚本和程序代码可以运行简单的fabric程序。 运行ls命令你将会看到如下内容:
  chaincode invoke.js network package.json query.js startFabric.sh
  现在我们运行startFabric.sh脚本去运行网络。
  
  ./startFabric.sh
  
  注意: 这个脚本将会下载HyperLedger Fabric Docker 镜像,所以它将会运行一会儿。
  
  为了简洁起见, 我们不会深挖这个命令运行时发生了什么, 这里给出了一个大概内容。
    启动一个通道节点(peer node), 排序节点(Ordering node)和认证授权中心(Certificate Authority )和一个CLI 容器。
    创建一个通道并且将启动的节点加入通道。
    在启动的节点上安装智能合约并且安装启动chainCode。
    调用initLedger添加10条数据到账本中。
    
  注意: 这些操作通常有管理员来完成。 本脚本使用CLI执行脚本完成操作。同时SDK也能很好的支持这些操作。
  
  执行docker ps 命令去显示startFabric.sh脚本运行的进程。 你可以通过Building Your First Network来学习更多细节和关于这些机器的操作。
  在这里我们把注意力放在程序这里。下面的图片简单展示了程序是如何访问HyperLedger Fabric Netowrk。

  
  好吧, 现在你有了一个简单的网络环境和一些能与这个环境交互的代码,让我们将这两部分组合到一起吧。
How Applications Interact with the Network 
如何使用程序访问Fabric
  程序使用APIs去访问智能合约(chaincode)。这些智能合约在网络的主机上以名字和版本作为唯一标识。例如: 我们的智能合约的标题是: dev-peer0.org.example.com-fabcar-1.0 ----名称的部分是fabcar, 版本的部分是1.0,运行的peer的名称是dev-peer0.org.example.com.
  
  API可以通过软件开发工具包(SDK)来访问。这次练习中我们将使用Hyper-ledger Fabric Node SDK来开发。当然你也可以使用JAVA SDK或者CLI来进行练习。
  
Querying the Ledger
查询账本
  查询是从账本中读取数据。你可以查询数据的单个值或者多个值。如果账本写入的时候使用的是像JSON一样的富文本 ---- 执行复杂的搜索。
  
  
  
  
  正如我们之前说的,我们的网络中有一个运行Chaincode容器和一个账本,其中有10条不同的数据。 我们还需要一些简单的JAVASCRIPT脚本 --
  -- 在fabcar目录下可以去查询账本下的块的信息。
  
  在我们了解程序如何运行之前,我们需要为我们的程序安装依赖的SDK模块,在你的fabcar目录下,运行如下命令:npm install。
  
  注意: 你将从fabcar目录发出所有后续的命令。
  
  现在我们可以运行我们的JAVASCRIPT程序了。 首先,让我们运行query.js程序列出账本中的所有区块的信息。 一个函数将会查询出所有的区块,
  queryAllCars, 预加载到我们的程序中, 我们可以简单的运行这个脚本: node query.js
  
  它将会返回如下内容:
   Query result count = 1
   Response is [{"Key":"CAR0", "Record":{"colour":"blue","make":"Toyota","model":"Prius
   ˓→ ","owner":"Tomoko"}},
   {"Key":"CAR1", "Record":{"colour":"red","make":"Ford","model":"Mustang","owner":
   ˓→ "Brad"}},
   {"Key":"CAR2", "Record":{"colour":"green","make":"Hyundai","model":"Tucson","owner":
   ˓→ "Jin Soo"}},
   {"Key":"CAR3", "Record":{"colour":"yellow","make":"Volkswagen","model":"Passat","owner
   ˓→ ":"Max"}},
   {"Key":"CAR4", "Record":{"colour":"black","make":"Tesla","model":"S","owner":"Adriana
   ˓→ "}},
   {"Key":"CAR5", "Record":{"colour":"purple","make":"Peugeot","model":"205","owner":
   ˓→ "Michel"}}, 
   {"Key":"CAR6", "Record":{"colour":"white","make":"Chery","model":"S22L","owner":"Aarav"}},
   
   {"Key":"CAR7", "Record":{"colour":"violet","make":"Fiat","model":"Punto","owner":"Pari
   ˓→ "}},
   {"Key":"CAR8", "Record":{"colour":"indigo","make":"Tata","model":"Nano","owner":
   ˓→ "Valeria"}},
   {"Key":"CAR9", "Record":{"colour":"brown","make":"Holden","model":"Barina","owner":
   ˓→ "Shotaro"}}]
   
   
   这里有10条数据。 一辆属于adriana黑色(black)的特斯拉(Tesla)Mode S,一辆红色(red)的福特(ford)野马(mustang)是Brad 和其他的。
   账本是基于键值(key/value)来实现,而且我们实现键(key)的CAR0 到CAR9。 这一点将会变的非常重要。
   
   让我们看看程序是如何实现的吧。 使用编辑器打开程序query.js。
   
   程序的初始化章节定义了主要的变量例如: (智能合约)chaincode ID, 通道名称(channel name)和 访问端口。
   
   var options = {
      wallet_path : path.join(__dirname, './network/creds'),
      user_id: 'PeerAdmin',
      channel_id: 'mychannel',
      chaincode_id: 'fabcar',
      network_url: 'grpc://localhost:7051',
   
   
   这一部分使我们构造查询的语句:
   
   // queryCar - requires 1 argument, ex: args: ['CAR4'],
   // queryAllCars - requires no arguments , ex: args: [''],
   const request = {
   chaincodeId: options.chaincode_id,
   txId: transaction_id,
   fcn: 'queryAllCars',
   args: ['']
   
   我们将chaincode_id定义为fabcar,允许我们到达这个智能合约。 然后调用智能合约中定义的queryAllCars函数。
   当我们运行 node query.js命令的时候,这个函数被调用来查询账本。然而这并不是我们唯一能调用的函数。
   
   我们来看下其他的代码。进入chaincode子目录然后打开fabcar.go文件, 你将会看到被声明为 -- initLedger,
   queryAllCars, createCar 和 changeCarOwner等函数。 让我们在queryAllCars中去了解如何操作账本。
   func (s
    * SmartContract) queryAllCars(APIstub shim.ChaincodeStubInterface) sc.Response
    ˓→ {
    startKey := "CAR0"
    endKey := "CAR999"
    resultsIterator, err := APIstub.GetStateByRange(startKey, endKey)
这个函数使用了shim的接口函数GetStateByRange通过两个区间函数startKey和endKey来返回之间的账本数据。
   我们分别定义了CAR0到CAR999的键。因此, 如果键被正确使用的话,我们可以存储1000辆车,而且queryAllCars会列出每一辆车。
   
   下面的图片展示了程序如何智能合约(chaincode)的不同的函数。




在上图中我们可以看到我们的queryAllCars, 还有一个createCar。createCar函数允许我们更新账本。createCar最终将一个新的块添加到区块链中。
   但是首先我们先做另一个查询。
   
   回到我们的query.js程序,编辑query.js的构造请求去查询一个特殊的车辆。  我们将会修改queryAllCars为queryCar,并且指定一个Key作为参数(这里我们
   使用CAR4)。我们编辑query.js成下面这样。
   const request = {
       chaincodeId: options.chaincode_id,
       txId: transaction_id,
       fcn: 'queryCar',
       args: ['CAR4']
   保存程序并返回fabcar目录下。然后再次运行程序:
   node query.js
   你应该看到如下的数据:
   {"colour":"black","make":"Tesla","model":"S","owner":"Adriana"}
   
   现在我们已经从查询所有到查询指定的一条。 使用queryCar函数,我们可以查询任意Key获取其映射的数据。
   
   非常好, 现在你应该能使用智能合约(chaincode)的基础函数用来查询并且可以传递查询参数了。
   
Updating the Ledger
更新账本。
   
   现在我们已经做过了账本查询并且修改了一点点代码, 我们已经做好了更新账本的准本。
   我们可以对账本做很多更新,但是前提是我们得有数据来更新。所以首先我们来创建一辆车(一条数据)。
   
   账本更新从一个事务的生成开始。 就像查询一样, 一个请求需要一个唯一的channelID, 调用函数和智能合约才能被创建,并且需要将这些数据传递。
   然后程序将调用channel.SendTransactionProposal API 将事务发送给peer获得建议(endorsement)。
   
   
   
   Fabric 返回一个请求响应,应用程序使用它来创建和提交一个事务请求。这个请求通过调用channel.sendTransaction API 发送到 排序服务
   (The Ordering Service)。排序服务(The Ordering Service)将此事务绑定到一个块中, 然后将该块发送给所有对等的节点上去认证
   (我们的例子中仅有一个节点)。
    
    
    
   最终程序会调用eh.setPeerAddr去链接节点的事件监听端口并且调用eh.registerTxEvent去注册时间关联到事务的ID。这个API允许程序去获取
   交易的流程(成功(successfully), committed(已提交)或者未成功(unsuccessful))。把他想象成为一个通知机制。
   
   
   注意: 我们不会深入的了解事务的生命周期。你可以查阅事务流程(Transaction Flow)文档来了解如何提交事务给账本的底层细节。
   
   
   我们最初的调用是建立一项新的资产(也就是一辆车)。我们有一个单独的程序 - invoke.js -我们将使用这些事务。就像查询(query.js)一样,
   我们使用编辑工具打开invoke.js 并找到构造调用API的代码:
   // createCar - requires 5 args, ex: args: ['CAR11', 'Honda', 'Accord', 'Black', 'Tom
    ˓→ '],
    // changeCarOwner - requires 2 args , ex: args: ['CAR10', 'Barry'],
    // send proposal to endorser
    var request = {
    targets: targets,
    chaincodeId: options.chaincode_id,
    fcn: '',
    args: [''],
    chainId: options.channel_id,
    txId: tx_id
   
    
   你将会看到我们调用两个函数 createCar和changeCarOwner. 让我们创建一辆红色的Chevy Volt,并且将他的拥有者设置为 Nick. 我们的账本的唯一标识的最顶端
   是CAR9, 所以我们将用CAR10. 更改代码块像下面这样:
   var request = {
    targets: targets,
    chaincodeId: options.chaincode_id,
    fcn: 'createCar',
    args: ['CAR10', 'Chevy', 'Volt', 'Red', 'Nick'],
    chainId: options.channel_id,
    txId: tx_id
    
    
   保存并运行它:
    node invoke.js
   
   你的控制台终将会输出响应和事务ID。当我们看到这条信息的时候:
    The transaction has been committed on peer localhost:7053
    
   节点会发出事件通知,我们通过eh.registerTxEvent去接受通知。所以现在如果我们重复去运行query.js的话我们是可以查询到CAR10的。
   我们应该会看到如下结果:
    Response is {"colour":"Red","make":"Chevy","model":"Volt","owner":"Nick"}
   
   最后让我们调用我们最后一个函数: changeCarOwner。 Nick是非常慷慨大方的,它准备将他的车送个一个叫Barry的朋友。所以我们简单
   的修改一下我们的invoke.js
    var request = {
     targets: targets
     chaincodeId: options.chaincode_id,
     fcn: 'changeCarOwner',
     args: ['CAR10', 'Barry'],
     chainId: options.channel_id,
     txId: tx_id
   再次运行程序 - node invoke.js - 然后再运行query.js。 我们将看到CAR10, 我们看到的是修改后的:
    Response is {"colour":"Red","make":"Chevy","model":"Volt","owner":"Barry"}

Additional Resources  
附加资源
   Hyperledger Fabric Node SDK repo 中有更多的优秀的资源和实例代码。当然你也可以咨询Hyperledger FAbric的社区。




    
    
    
    
    
    
    
    
    
















阅读更多

更多精彩内容