Hyperledger-Fabric源码分析(Gossip-Anti-Entropy)

Anti-Entropy

首先我们先看一个概念,Anti-Entropy。

Gossip算法又被称为反熵(Anti-Entropy),熵是物理学上的一个概念,代表杂乱无章,而反熵就是在杂乱无章中寻求一致,这充分说明了 Gossip的特点:在一个有界网络中,每个节点都随机地与其他节点通信,经过一番杂乱无章的通信,最终所有节点的状态都会达成一致。每个节点可能知道所有其他节点,也可能仅知道几个邻居节点,只要这些节可以通过网络连通,最终他们的状态都是一致的,当然这也是疫情传播的特点。

初看感觉不知所云,废话不多说,我们下面深入看看fabric的Anti-Entropy是怎么实现的。

GossipStateProvider

在开始之前,我们要知道反熵是在同步什么?当然是block。而GossipStateProvider就是为了这个目的而生,他随着peer启动而启动。启动之后会默默的为达成最终一致而努力。

func NewGossipStateProvider(chainID string, services *ServicesMediator, ledger ledgerResources) GossipStateProvider {

  ...

   // Listen for incoming communication
   go s.listen()
   // Deliver in order messages into the incoming channel
   go s.deliverPayloads()
   // Execute anti entropy to fill missing gaps
   go s.antiEntropy()
   // Taking care of state request messages
   go s.processStateRequests()

   return s
}

在启动的过程他主要做几件事,下面我们深入分析下

  • listen
    • 这里值得注意的是state里面的payloads,不管是从orderer收到的block还是从peer同步来的block,都会第一时间push到payloads里面,而deliverPayloads会伺机将block保存到本地账本。
  • deliverPayloads
  • antiEntropy
  • processStateRequests

listen

func (s *GossipStateProviderImpl) listen() {
   defer s.done.Done()

   for {
      select {
      case msg := <-s.gossipChan:
         logger.Debug("Received new message via gossip channel")
         go s.queueNewMessage(msg)
      case msg := <-s.commChan:
         logger.Debug("Dispatching a message", msg)
         go s.dispatch(msg)
      case <-s.stopCh:
         s.stopCh <- struct{}{}
         logger.Debug("Stop listening for new messages")
         return
      }
   }
}
  • gossipChan就是用来接收orderer发来的block消息
  • 而这里主要是commChan,来接收peer发来的GossipMessage_StateRequest或GossipMessage_StateResponse

antiEntropy

func (s *GossipStateProviderImpl) antiEntropy() {
   defer s.done.Done()
   defer logger.Debug("State Provider stopped, stopping anti entropy procedure.")

   for {
      select {
      case <-s.stopCh:
         s.stopCh <- struct{}{}
         return
      case <-time.After(defAntiEntropyInterval):
         ourHeight, err := s.ledger.LedgerHeight()
         if err != nil {
            // Unable to read from ledger continue to the next round
            logger.Errorf("Cannot obtain ledger height, due to %+v", errors.WithStack(err))
            continue
         }
         if ourHeight == 0 {
            logger.Error("Ledger reported block height of 0 but this should be impossible")
            continue
         }
         maxHeight := s.maxAvailableLedgerHeight()
         if ourHeight >= maxHeight {
            continue
         }

         s.requestBlocksInRange(uint64(ourHeight), uint64(maxHeight)-1)
      }
   }
}

这里便正式开始处理了。

  • 每隔一段时间去获取ledger的高度,这里的ledger当然是本地。
  • 接着会遍历本地的成员列表中ledger高度
  • 接下来,就是根据本地缺失的部分去拉取数据了。

maxAvailableLedgerHeight

func (s *GossipStateProviderImpl) maxAvailableLedgerHeight() uint64 {
   max := uint64(0)
   for _, p := range s.mediator.PeersOfChannel(common2.ChainID(s.chainID)) {
      if p.Properties == nil {
         logger.Debug("Peer", p.PreferredEndpoint(), "doesn't have properties, skipping it")
         continue
      }
      peerHeight := p.Properties.LedgerHeight
      if max < peerHeight {
         max = peerHeight
      }
   }
   return max
}
  • 这里就是遍历的现场啦,拿到最大的高度值
  • 还记得之前的GossipMessage_StateInfo篇么?那里的Properties的LedgerHeight在这里用上了。不管是主动拉取还是被动获得,总之,本地保存了其他成员的账本状态。

requestBlocksInRange

func (s *GossipStateProviderImpl) requestBlocksInRange(start uint64, end uint64) {
   atomic.StoreInt32(&s.stateTransferActive, 1)
   defer atomic.StoreInt32(&s.stateTransferActive, 0)

   for prev := start; prev <= end; {
      next := min(end, prev+defAntiEntropyBatchSize)

      gossipMsg := s.stateRequestMessage(prev, next)

      responseReceived := false
      tryCounts := 0

      for !responseReceived {
         if tryCounts > defAntiEntropyMaxRetries {
            logger.Warningf("Wasn't  able to get blocks in range [%d...%d), after %d retries",
               prev, next, tryCounts)
            return
         }
         // Select peers to ask for blocks
         peer, err := s.selectPeerToRequestFrom(next)
         if err != nil {
            logger.Warningf("Cannot send state request for blocks in range [%d...%d), due to %+v",
               prev, next, errors.WithStack(err))
            return
         }

         logger.Debugf("State transfer, with peer %s, requesting blocks in range [%d...%d), "+
            "for chainID %s", peer.Endpoint, prev, next, s.chainID)

         s.mediator.Send(gossipMsg, peer)
         tryCounts++

         // Wait until timeout or response arrival
         select {
         case msg := <-s.stateResponseCh:
            if msg.GetGossipMessage().Nonce != gossipMsg.Nonce {
               continue
            }
            // Got corresponding response for state request, can continue
            index, err := s.handleStateResponse(msg)
            if err != nil {
               logger.Warningf("Wasn't able to process state response for "+
                  "blocks [%d...%d], due to %+v", prev, next, errors.WithStack(err))
               continue
            }
            prev = index + 1
            responseReceived = true
         case <-time.After(defAntiEntropyStateResponseTimeout):
         case <-s.stopCh:
            s.stopCh <- struct{}{}
            return
         }
      }
   }
}
  • 当然了,不是一次性全部去拉取,也不是一个一个去拉取,做事还是要讲究个方式方法,得有计划分批请求。
  • 创建GossipMessage_StateRequest消息,也没什么,主要是start和end,表示我要拉取这个range的数据。在channel成员中随机选取一个height能到达next的peer节点进行发送,并等待GossipMessage_StateResponse
  • 前面的listen会将收到的response转发到这里
  • 如果Nonce不一致,说明不是前面请求所期待的response,直接跳过
  • 接着就是handleStateResponse

handleStateResponse

func (s *GossipStateProviderImpl) handleStateResponse(msg proto.ReceivedMessage) (uint64, error) {
   max := uint64(0)
   // Send signal that response for given nonce has been received
   response := msg.GetGossipMessage().GetStateResponse()
   // Extract payloads, verify and push into buffer
   if len(response.GetPayloads()) == 0 {
      return uint64(0), errors.New("Received state transfer response without payload")
   }
   for _, payload := range response.GetPayloads() {
      logger.Debugf("Received payload with sequence number %d.", payload.SeqNum)
      if err := s.mediator.VerifyBlock(common2.ChainID(s.chainID), payload.SeqNum, payload.Data); err != nil {
         err = errors.WithStack(err)
         logger.Warningf("Error verifying block with sequence number %d, due to %+v", payload.SeqNum, err)
         return uint64(0), err
      }
      if max < payload.SeqNum {
         max = payload.SeqNum
      }

      err := s.addPayload(payload, blocking)
      if err != nil {
         logger.Warningf("Block [%d] received from block transfer wasn't added to payload buffer: %v", payload.SeqNum, err)
      }
   }
   return max, nil
}
  • 这里主要是拿到StateResponse,push到stateprovider的payloads里面,而payloads后面会有针对性的处理

deliverPayloads

func (s *GossipStateProviderImpl) deliverPayloads() {
   defer s.done.Done()

   for {
      select {
      // Wait for notification that next seq has arrived
      case <-s.payloads.Ready():
         logger.Debugf("[%s] Ready to transfer payloads (blocks) to the ledger, next block number is = [%d]", s.chainID, s.payloads.Next())
         // Collect all subsequent payloads
         for payload := s.payloads.Pop(); payload != nil; payload = s.payloads.Pop() {
            rawBlock := &common.Block{}
            if err := pb.Unmarshal(payload.Data, rawBlock); err != nil {
               logger.Errorf("Error getting block with seqNum = %d due to (%+v)...dropping block", payload.SeqNum, errors.WithStack(err))
               continue
            }
            if rawBlock.Data == nil || rawBlock.Header == nil {
               logger.Errorf("Block with claimed sequence %d has no header (%v) or data (%v)",
                  payload.SeqNum, rawBlock.Header, rawBlock.Data)
               continue
            }
            logger.Debugf("[%s] Transferring block [%d] with %d transaction(s) to the ledger", s.chainID, payload.SeqNum, len(rawBlock.Data.Data))

            // Read all private data into slice
            var p util.PvtDataCollections
            if payload.PrivateData != nil {
               err := p.Unmarshal(payload.PrivateData)
               if err != nil {
                  logger.Errorf("Wasn't able to unmarshal private data for block seqNum = %d due to (%+v)...dropping block", payload.SeqNum, errors.WithStack(err))
                  continue
               }
            }
            if err := s.commitBlock(rawBlock, p); err != nil {
               if executionErr, isExecutionErr := err.(*vsccErrors.VSCCExecutionFailureError); isExecutionErr {
                  logger.Errorf("Failed executing VSCC due to %v. Aborting chain processing", executionErr)
                  return
               }
               logger.Panicf("Cannot commit block to the ledger due to %+v", errors.WithStack(err))
            }
         }
      case <-s.stopCh:
         s.stopCh <- struct{}{}
         logger.Debug("State provider has been stopped, finishing to push new blocks.")
         return
      }
   }
}
  • 前面收到peer发来的statereponse后,push到payloads里面,这里的Ready就会激活。
  • 将payload理解为block,那么这里就是遍历payloads里面累计的block,然后commit到本地ledger。
  • 当然,最后更新ledger的高度。不要忘了,这个变化会最终同步给其他成员,以stateinfo消息的方式。

processStateRequests

func (s *GossipStateProviderImpl) processStateRequests() {
   defer s.done.Done()

   for {
      select {
      case msg := <-s.stateRequestCh:
         s.handleStateRequest(msg)
      case <-s.stopCh:
         s.stopCh <- struct{}{}
         return
      }
   }
}
  • 前面的listen会将收到的request转发到这里

handleStateRequest

func (s *GossipStateProviderImpl) handleStateRequest(msg proto.ReceivedMessage) {
   if msg == nil {
      return
   }
   request := msg.GetGossipMessage().GetStateRequest()

   batchSize := request.EndSeqNum - request.StartSeqNum
   if batchSize > defAntiEntropyBatchSize {
      logger.Errorf("Requesting blocks batchSize size (%d) greater than configured allowed"+
         " (%d) batching for anti-entropy. Ignoring request...", batchSize, defAntiEntropyBatchSize)
      return
   }

   if request.StartSeqNum > request.EndSeqNum {
      logger.Errorf("Invalid sequence interval [%d...%d], ignoring request...", request.StartSeqNum, request.EndSeqNum)
      return
   }

   currentHeight, err := s.ledger.LedgerHeight()
   if err != nil {
      logger.Errorf("Cannot access to current ledger height, due to %+v", errors.WithStack(err))
      return
   }
   if currentHeight < request.EndSeqNum {
      logger.Warningf("Received state request to transfer blocks with sequence numbers higher  [%d...%d] "+
         "than available in ledger (%d)", request.StartSeqNum, request.StartSeqNum, currentHeight)
   }

   endSeqNum := min(currentHeight, request.EndSeqNum)

   response := &proto.RemoteStateResponse{Payloads: make([]*proto.Payload, 0)}
   for seqNum := request.StartSeqNum; seqNum <= endSeqNum; seqNum++ {
      logger.Debug("Reading block ", seqNum, " with private data from the coordinator service")
      connInfo := msg.GetConnectionInfo()
      peerAuthInfo := common.SignedData{
         Data:      connInfo.Auth.SignedData,
         Signature: connInfo.Auth.Signature,
         Identity:  connInfo.Identity,
      }
      block, pvtData, err := s.ledger.GetPvtDataAndBlockByNum(seqNum, peerAuthInfo)

      if err != nil {
         logger.Errorf("cannot read block number %d from ledger, because %+v, skipping...", seqNum, err)
         continue
      }

      if block == nil {
         logger.Errorf("Wasn't able to read block with sequence number %d from ledger, skipping....", seqNum)
         continue
      }

      blockBytes, err := pb.Marshal(block)

      if err != nil {
         logger.Errorf("Could not marshal block: %+v", errors.WithStack(err))
         continue
      }

      var pvtBytes [][]byte
      if pvtData != nil {
         // Marshal private data
         pvtBytes, err = pvtData.Marshal()
         if err != nil {
            logger.Errorf("Failed to marshal private rwset for block %d due to %+v", seqNum, errors.WithStack(err))
            continue
         }
      }

      // Appending result to the response
      response.Payloads = append(response.Payloads, &proto.Payload{
         SeqNum:      seqNum,
         Data:        blockBytes,
         PrivateData: pvtBytes,
      })
   }
   // Sending back response with missing blocks
   msg.Respond(&proto.GossipMessage{
      // Copy nonce field from the request, so it will be possible to match response
      Nonce:   msg.GetGossipMessage().Nonce,
      Tag:     proto.GossipMessage_CHAN_OR_ORG,
      Channel: []byte(s.chainID),
      Content: &proto.GossipMessage_StateResponse{StateResponse: response},
   })
}
  • 基本上你也能猜得到,就是根据请求,尽可能满足对方的要求,将自己能给的block打包成GossipMessage_StateResponse发给他。

总结

至此,可以看到Fabric的Anti-Entropy其实很简单,就是定时去跟成员比对账本状态,达到最终一致。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容