91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

bytom啟動后怎么連接別的節點

發布時間:2021-12-20 17:43:16 來源:億速云 閱讀:84 作者:iii 欄目:互聯網科技

本篇內容介紹了“bytom啟動后怎么連接別的節點”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

比原啟動后去哪里連接別的節點

最開始我對于這個問題一直有個疑惑:區塊鏈是一個分布式的網絡,那么一個節點啟動后,它怎么知道去哪里找別的節點從而加入網絡呢?

看到代碼之后,我才明白,原來在代碼中硬編碼了一些種子地址,這樣在啟動的時候,可以先通過種子地址加入網絡。雖然整個網絡是分布式的,但是最開始還是需要一定的中心化。

預編碼內容

對于配置文件config.toml,比原的代碼中硬編碼了配置文件內容:

config/toml.go#L22-L45

var defaultConfigTmpl = `# This is a TOML config file.
# For more information, see https://github.com/toml-lang/toml
fast_sync = true
db_backend = "leveldb"
api_addr = "0.0.0.0:9888"
`

var mainNetConfigTmpl = `chain_id = "mainnet"
[p2p]
laddr = "tcp://0.0.0.0:46657"
seeds = "45.79.213.28:46657,198.74.61.131:46657,212.111.41.245:46657,47.100.214.154:46657,47.100.109.199:46657,47.100.105.165:46657"
`

var testNetConfigTmpl = `chain_id = "testnet"
[p2p]
laddr = "tcp://0.0.0.0:46656"
seeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656"
`

var soloNetConfigTmpl = `chain_id = "solonet"
[p2p]
laddr = "tcp://0.0.0.0:46658"
seeds = ""
`

可以看出,對于不同的chain_id,預設的種子是不同的。

當然,如果我們自己知道某些節點的地址,也可以在初始化生成config.toml后,手動修改該文件添加進去。

啟動syncManager

那么,比原在代碼中是使用這些種子地址并連接它們的呢?關鍵在于,連接的代碼位于SyncManager中,所以我們要找到啟動syncManager的地方。

首先,當我們使用bytomd node啟動后,下面的函數將被調用:

cmd/bytomd/commands/run_node.go#L41

func runNode(cmd *cobra.Command, args []string) error {
    // Create & start node
    n := node.NewNode(config)
    if _, err := n.Start(); err != nil {
        // ...
    }
    // ...
}

這里調用了n.Start,其中的Start方法,來自于Node所嵌入的cmn.BaseService

node/node.go#L39

type Node struct {
    cmn.BaseService
    // ...
}

所以n.Start對應的是下面這個方法:

vendor/github.com/tendermint/tmlibs/common/service.go#L97

func (bs *BaseService) Start() (bool, error) {
    // ...
    err := bs.impl.OnStart()
    // ...
}

在這里,由于bs.impl對應于Node,所以將繼續調用Node.OnStart():

node/node.go#L169

func (n *Node) OnStart() error {
    // ...
    n.syncManager.Start()
    // ...
}

可以看到,我們終于走到了調用了syncManager.Start()的地方。

syncManager中的處理

然后就是在syncManager內部的一些處理了。

它主要是除了從config.toml中取得種子節點外,還需要把以前連接過并保存在本地的AddressBook.json中的節點也拿出來連接,這樣就算預設的種子節點失敗了,也還是有可能連接上網絡(部分解決了前面提到的中心化的擔憂)。

syncManager.Start()對應于:

netsync/handle.go#L141

func (sm *SyncManager) Start() {
    go sm.netStart()
    // ...
}

其中sm.netStart(),對應于:

netsync/handle.go#L121

func (sm *SyncManager) netStart() error {
    // ...
    // If seeds exist, add them to the address book and dial out
    if sm.config.P2P.Seeds != "" {
        // dial out
        seeds := strings.Split(sm.config.P2P.Seeds, ",")
        if err := sm.DialSeeds(seeds); err != nil {
            return err
        }
    }
    // ...
}

其中的sm.config.P2P.Seeds就對應于config.toml中的seeds。關于這兩者是怎么對應起來的,會在后面文章中詳解。

緊接著,再通過sm.DialSeeds(seeds)去連接這些seed,這個方法對應的代碼位于:

netsync/handle.go#L229

func (sm *SyncManager) DialSeeds(seeds []string) error {
    return sm.sw.DialSeeds(sm.addrBook, seeds)
}

其實是是調用了sm.sw.DialSeeds,而sm.sw是指Switch。這時可以看到,有一個叫addrBook的東西參與了進來,它保存了該結點之前成功連接過的節點地址,我們這里暫不多做討論。

Switch.DialSeeds對應于:

p2p/switch.go#L311

func (sw *Switch) DialSeeds(addrBook *AddrBook, seeds []string) error {
    // ...
    perm := rand.Perm(len(netAddrs))
    for i := 0; i < len(perm)/2; i++ {
        j := perm[i]
        sw.dialSeed(netAddrs[j])
    }
   // ...
}

這里引入了隨機數,是為了將發起連接的順序打亂,這樣可以讓每個種子都獲得公平的連接機會。

sw.dialSeed(netAddrs[j])對應于:

p2p/switch.go#L342

func (sw *Switch) dialSeed(addr *NetAddress) {
    peer, err := sw.DialPeerWithAddress(addr, false)
    // ...
}

sw.DialPeerWithAddress(addr, false)又對應于:

p2p/switch.go#L351

func (sw *Switch) DialPeerWithAddress(addr *NetAddress, persistent bool) (*Peer, error) {
    // ...
    log.WithField("address", addr).Info("Dialing peer")
    peer, err := newOutboundPeerWithConfig(addr, sw.reactorsByCh, sw.chDescs, sw.StopPeerForError, sw.nodePrivKey, sw.peerConfig)
    // ...
}

其中的persistent參數如果是true的話,表明這個peer比較重要,在某些情況下如果斷開連接后,還會嘗試重連。如果persistentfalse的,就沒有這個待遇。

newOutboundPeerWithConfig對應于:

p2p/peer.go#L69

func newOutboundPeerWithConfig(addr *NetAddress, reactorsByCh map[byte]Reactor, chDescs []*ChannelDescriptor, onPeerError func(*Peer, interface{}), ourNodePrivKey crypto.PrivKeyEd25519, config *PeerConfig) (*Peer, error) {
    conn, err := dial(addr, config)
    // ...
}

繼續dial,加入了超時:

p2p/peer.go#L284

func dial(addr *NetAddress, config *PeerConfig) (net.Conn, error) {
    conn, err := addr.DialTimeout(config.DialTimeout * time.Second)
    if err != nil {
        return nil, err
    }
    return conn, nil
}

addr.DialTimeout對應于:

p2p/netaddress.go#L141

func (na *NetAddress) DialTimeout(timeout time.Duration) (net.Conn, error) {
    conn, err := net.DialTimeout("tcp", na.String(), timeout)
    if err != nil {
        return nil, err
    }
    return conn, nil
}

“bytom啟動后怎么連接別的節點”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

平阳县| 龙州县| 东港市| 东安县| 汝南县| 荣成市| 宣化县| 会理县| 安福县| 图木舒克市| 漳平市| 拉萨市| 宁城县| 岳阳县| 岳阳市| 临沧市| 万全县| 洪洞县| 林西县| 绍兴县| 双鸭山市| 金堂县| 息烽县| 那曲县| 弥渡县| 边坝县| 铜梁县| 营山县| 莒南县| 益阳市| 兰西县| 金塔县| 湄潭县| 郧西县| 台南县| 开封县| 邢台县| 康定县| 花莲县| 亳州市| 潮州市|