fabric gossip 源码解析 DeliverService

接口描述

fabric/core/deliverservice/deliveryclient.go

// DeliverService used to communicate with orderers to obtain
// new block and send the to the committer service
type DeliverService interface {
    // StartDeliverForChannel dynamically starts delivery of new blocks from ordering service
    // to channel peers.
    StartDeliverForChannel(chainID string, ledgerInfo blocksprovider.LedgerInfo) error

    // StopDeliverForChannel dynamically stops delivery of new blocks from ordering service
    // to channel peers.
    StopDeliverForChannel(chainID string) error
    // Stop terminates delivery service and closes the connection
    Stop()
}

根据描述可以很明显的看出来是与order服务进行收发block的接口,接下来阅读以下主要代码,三个函数只有 StartDeliverForChannel是重点,其余两个都不是重点,所以我们主要就搞懂这个就可以了,既然是收发代码,就是找到收发逻辑就可以了。

接口实现

// StartDeliverForChannel starts blocks delivery for channel
// initializes the grpc stream for given chainID, creates blocks provider instance
// that spawns in go routine to read new blocks starting from the position provided by ledger
// info instance.
func (d *deliverServiceImpl) StartDeliverForChannel(chainID string, ledgerInfo blocksprovider.LedgerInfo) error {
    d.lock.Lock()
    defer d.lock.Unlock()
    if d.stopping {
        errMsg := fmt.Sprintf("Delivery service is stopping cannot join a new channel %s", chainID)
        logger.Errorf(errMsg)
        return errors.New(errMsg)
    }
    if _, exist := d.clients[chainID]; exist {
        errMsg := fmt.Sprintf("Delivery service - block provider already exists for %s found, can't start delivery", chainID)
        logger.Errorf(errMsg)
        return errors.New(errMsg)
    } else {
        abc, err := d.clientsFactory.Create()
        if err != nil {
            logger.Errorf("Unable to initialize atomic broadcast, due to %s", err)
            return err
        }
        logger.Debug("This peer will pass blocks from orderer service to other peers")
        d.clients[chainID] = blocksprovider.NewBlocksProvider(chainID, abc, d.gossip)

        if err := d.clients[chainID].RequestBlocks(ledgerInfo); err == nil {
            // Start reading blocks from ordering service in case this peer is a leader for specified chain
            go d.clients[chainID].DeliverBlocks()
        }
    }
    return nil
}

通过代码可以看出来 blocksprovider.NewBlocksProvider 产生了真正的连接,在这个连接上会先 RequestBlocks,然后 DeliverBlocks ,我们只要搞清楚 与 谁建立的连接,RequestBlocks,DeliverBlocks 的逻辑后就弄清楚了 这个service的主要功能了

  • 连接的建立
func NewDeliverService(gossip blocksprovider.GossipServiceAdapter, endpoints []string) (DeliverService, error) {
    indices := rand.Perm(len(endpoints))
    for _, idx := range indices {
        logger.Infof("Creating delivery service to get blocks from the ordering service, %s", endpoints[idx])

        dialOpts := []grpc.DialOption{grpc.WithInsecure(), grpc.WithTimeout(3 * time.Second), grpc.WithBlock()}

        if comm.TLSEnabled() {
            dialOpts = append(dialOpts, grpc.WithTransportCredentials(comm.InitTLSForPeer()))
        } else {
            dialOpts = append(dialOpts, grpc.WithInsecure())
        }

        conn, err := grpc.Dial(endpoints[idx], dialOpts...)
        if err != nil {
            logger.Errorf("Cannot dial to %s, because of %s", endpoints[idx], err)
            continue
        }
        return NewFactoryDeliverService(gossip, &blocksDelivererFactoryImpl{conn}, conn), nil
    }
    return nil, fmt.Errorf("Wasn't able to connect to any of ordering service endpoints %s", endpoints)
}

可以看到连接是建立到了,endpoints,通过追踪代码可以发现就是 order 节点的地址,这里需要注意一点, 传过来的 endpoints 是一个数组,但是只随机的与某一个节点建立连接。
现在我们知道了是与order通信,由于fabric是基于grpc的,所以我们还要找出来与order的哪个服务进行通信,blocksDelivererFactoryImpl 这个struct定义了与哪个服务通信,我们一起看下

fabric/core/deliverservice/deliveryclient.go

type blocksDelivererFactoryImpl struct {
    conn *grpc.ClientConn
}

// Create a factory method which is capable to instantiate new BlocksDeliverer
func (factory *blocksDelivererFactoryImpl) Create() (blocksprovider.BlocksDeliverer, error) {
    var abc orderer.AtomicBroadcast_DeliverClient
    var err error
    abc, err = orderer.NewAtomicBroadcastClient(factory.conn).Deliver(context.TODO())
    if err != nil {
        return nil, err
    }

    return abc, nil
}

明显是AtomicBroadcast 服务的 Deliver 接口通信

  • RequestBlocks
fabric/core/deliverservice/blocksprovider/blocksprovider.go

func (b *blocksProviderImpl) RequestBlocks(ledgerInfoProvider LedgerInfo) error {
    height, err := ledgerInfoProvider.LedgerHeight()
    if err != nil {
        logger.Errorf("Can't get legder height from committer [%s]", err)
        return err
    }

    if height > 0 {
        logger.Debugf("Starting deliver with block [%d]", height)
        if err := b.seekLatestFromCommitter(height); err != nil {
            return err
        }
    } else {
        logger.Debug("Starting deliver with olders block")
        if err := b.seekOldest(); err != nil {
            return err
        }
    }

    return nil
}

func (b *blocksProviderImpl) seekLatestFromCommitter(height uint64) error {
    seekInfo := &orderer.SeekInfo{
        Start:    &orderer.SeekPosition{Type: &orderer.SeekPosition_Specified{Specified: &orderer.SeekSpecified{Number: height}}},
        Stop:     &orderer.SeekPosition{Type: &orderer.SeekPosition_Specified{Specified: &orderer.SeekSpecified{Number: math.MaxUint64}}},
        Behavior: orderer.SeekInfo_BLOCK_UNTIL_READY,
    }

    //TODO- epoch and msgVersion may need to be obtained for nowfollowing usage in orderer/configupdate/configupdate.go
    msgVersion := int32(0)
    epoch := uint64(0)
    env, err := utils.CreateSignedEnvelope(common.HeaderType_CONFIG_UPDATE, b.chainID, localmsp.NewSigner(), seekInfo, msgVersion, epoch)
    if err != nil {
        return err
    }
    return b.client.Send(env)
}

明显 可以看出来只是在请求 start - stop 之间的block

  • DeliverBlocks
func (b *blocksProviderImpl) DeliverBlocks() {
    for !b.isDone() {
        msg, err := b.client.Recv()   // 接收Request请求的代码
        if err != nil {
            logger.Warningf("Receive error: %s", err.Error())
            return
        }
        switch t := msg.Type.(type) {
        case *orderer.DeliverResponse_Status:
            if t.Status == common.Status_SUCCESS {
                logger.Warning("ERROR! Received success for a seek that should never complete")
                return
            }
            logger.Warning("Got error ", t)
        case *orderer.DeliverResponse_Block:
            seqNum := t.Block.Header.Number

            numberOfPeers := len(b.gossip.PeersOfChannel(gossipcommon.ChainID(b.chainID)))
            // Create payload with a block received
            payload := createPayload(seqNum, t.Block)
            // Use payload to create gossip message
            gossipMsg := createGossipMsg(b.chainID, payload)

            logger.Debugf("Adding payload locally, buffer seqNum = [%d], peers number [%d]", seqNum, numberOfPeers)
            // Add payload to local state payloads buffer
            b.gossip.AddPayload(b.chainID, payload)

            // Gossip messages with other nodes
            logger.Debugf("Gossiping block [%d], peers number [%d]", seqNum, numberOfPeers)
            b.gossip.Gossip(gossipMsg)  // gossip 发送
        default:
            logger.Warning("Received unknown: ", t)
            return
        }
    }
}

可以看出来,request 来的 block,在这边都通过 gossip 广播出去了

总结

这个这个代码很清晰了,DeliverService 就是从 order 请求 block,然后广播出去。 当然在整个fabric中的业务作用目前看不出来,还需要深入分析

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

推荐阅读更多精彩内容

  • 接口描述 根据描述可以很明显的看出来是与order服务进行收发block的接口,接下来阅读以下主要代码,三个函数只...
    上海大坤哥阅读 1,794评论 0 3
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,650评论 18 139
  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,369评论 8 265
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,096评论 1 32
  • 仿佛看透了爱情,一天三种心思,吃饭睡觉看书,一天三种想法,想人想事想死。洞察时间,已经不多,浪费的总是青春,总是徘...
    物非人非阅读 191评论 0 1