|
|
- package p2p
-
- import (
- "fmt"
- "net"
- "strconv"
- "strings"
-
- crypto "github.com/tendermint/go-crypto"
- )
-
- const maxNodeInfoSize = 10240 // 10Kb
-
- // NodeInfo is the basic node information exchanged
- // between two peers during the Tendermint P2P handshake.
- type NodeInfo struct {
- // Authenticate
- PubKey crypto.PubKey `json:"pub_key"` // authenticated pubkey
- ListenAddr string `json:"listen_addr"` // accepting incoming
-
- // Check compatibility
- Network string `json:"network"` // network/chain ID
- Version string `json:"version"` // major.minor.revision
-
- // Sanitize
- Moniker string `json:"moniker"` // arbitrary moniker
- Other []string `json:"other"` // other application specific data
- }
-
- // Validate checks the self-reported NodeInfo is safe.
- // It returns an error if the info.PubKey doesn't match the given pubKey.
- // TODO: constraints for Moniker/Other? Or is that for the UI ?
- func (info NodeInfo) Validate(pubKey crypto.PubKey) error {
- if !info.PubKey.Equals(pubKey) {
- return fmt.Errorf("info.PubKey (%v) doesn't match peer.PubKey (%v)",
- info.PubKey, pubKey)
- }
- return nil
- }
-
- // CONTRACT: two nodes are compatible if the major/minor versions match and network match
- func (info NodeInfo) CompatibleWith(other NodeInfo) error {
- iMajor, iMinor, _, iErr := splitVersion(info.Version)
- oMajor, oMinor, _, oErr := splitVersion(other.Version)
-
- // if our own version number is not formatted right, we messed up
- if iErr != nil {
- return iErr
- }
-
- // version number must be formatted correctly ("x.x.x")
- if oErr != nil {
- return oErr
- }
-
- // major version must match
- if iMajor != oMajor {
- return fmt.Errorf("Peer is on a different major version. Got %v, expected %v", oMajor, iMajor)
- }
-
- // minor version must match
- if iMinor != oMinor {
- return fmt.Errorf("Peer is on a different minor version. Got %v, expected %v", oMinor, iMinor)
- }
-
- // nodes must be on the same network
- if info.Network != other.Network {
- return fmt.Errorf("Peer is on a different network. Got %v, expected %v", other.Network, info.Network)
- }
-
- return nil
- }
-
- func (info NodeInfo) ID() ID {
- return PubKeyToID(info.PubKey)
- }
-
- func (info NodeInfo) NetAddress() *NetAddress {
- id := PubKeyToID(info.PubKey)
- addr := info.ListenAddr
- netAddr, err := NewNetAddressString(IDAddressString(id, addr))
- if err != nil {
- panic(err) // everything should be well formed by now
- }
- return netAddr
- }
-
- func (info NodeInfo) ListenHost() string {
- host, _, _ := net.SplitHostPort(info.ListenAddr) // nolint: errcheck, gas
- return host
- }
-
- func (info NodeInfo) ListenPort() int {
- _, port, _ := net.SplitHostPort(info.ListenAddr) // nolint: errcheck, gas
- port_i, err := strconv.Atoi(port)
- if err != nil {
- return -1
- }
- return port_i
- }
-
- func (info NodeInfo) String() string {
- return fmt.Sprintf("NodeInfo{pk: %v, moniker: %v, network: %v [listen %v], version: %v (%v)}", info.PubKey, info.Moniker, info.Network, info.ListenAddr, info.Version, info.Other)
- }
-
- func splitVersion(version string) (string, string, string, error) {
- spl := strings.Split(version, ".")
- if len(spl) != 3 {
- return "", "", "", fmt.Errorf("Invalid version format %v", version)
- }
- return spl[0], spl[1], spl[2], nil
- }
|