message.go 4.88 KB
Newer Older
1
package dht_pb
2

Jeromy's avatar
Jeromy committed
3
import (
4
	logging "github.com/ipfs/go-log"
5 6 7
	inet "github.com/libp2p/go-libp2p-net"
	peer "github.com/libp2p/go-libp2p-peer"
	pstore "github.com/libp2p/go-libp2p-peerstore"
8
	b58 "github.com/mr-tron/base58/base58"
9
	ma "github.com/multiformats/go-multiaddr"
Jeromy's avatar
Jeromy committed
10 11
)

Jeromy's avatar
Jeromy committed
12
var log = logging.Logger("dht.pb")
13

14
type PeerRoutingInfo struct {
Jeromy's avatar
Jeromy committed
15
	pstore.PeerInfo
16 17 18
	inet.Connectedness
}

19
// NewMessage constructs a new dht message with given type, key, and level
20
func NewMessage(typ Message_MessageType, key string, level int) *Message {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
21 22 23 24 25 26 27 28
	m := &Message{
		Type: &typ,
		Key:  &key,
	}
	m.SetClusterLevel(level)
	return m
}

29 30 31 32 33 34 35 36 37 38 39 40 41 42
func peerRoutingInfoToPBPeer(p PeerRoutingInfo) *Message_Peer {
	pbp := new(Message_Peer)

	pbp.Addrs = make([][]byte, len(p.Addrs))
	for i, maddr := range p.Addrs {
		pbp.Addrs[i] = maddr.Bytes() // Bytes, not String. Compressed.
	}
	s := string(p.ID)
	pbp.Id = &s
	c := ConnectionType(p.Connectedness)
	pbp.Connection = &c
	return pbp
}

Jeromy's avatar
Jeromy committed
43
func peerInfoToPBPeer(p pstore.PeerInfo) *Message_Peer {
44
	pbp := new(Message_Peer)
45

46 47 48
	pbp.Addrs = make([][]byte, len(p.Addrs))
	for i, maddr := range p.Addrs {
		pbp.Addrs[i] = maddr.Bytes() // Bytes, not String. Compressed.
Jeromy's avatar
Jeromy committed
49
	}
50 51
	s := string(p.ID)
	pbp.Id = &s
Jeromy's avatar
Jeromy committed
52
	return pbp
53 54
}

Jeromy's avatar
Jeromy committed
55
// PBPeerToPeer turns a *Message_Peer into its pstore.PeerInfo counterpart
Jeromy's avatar
Jeromy committed
56 57
func PBPeerToPeerInfo(pbp *Message_Peer) *pstore.PeerInfo {
	return &pstore.PeerInfo{
58 59
		ID:    peer.ID(pbp.GetId()),
		Addrs: pbp.Addresses(),
60 61 62
	}
}

63
// RawPeerInfosToPBPeers converts a slice of Peers into a slice of *Message_Peers,
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
64
// ready to go out on the wire.
Jeromy's avatar
Jeromy committed
65
func RawPeerInfosToPBPeers(peers []pstore.PeerInfo) []*Message_Peer {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
66
	pbpeers := make([]*Message_Peer, len(peers))
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
67
	for i, p := range peers {
68
		pbpeers[i] = peerInfoToPBPeer(p)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
69 70 71 72
	}
	return pbpeers
}

73 74 75
// PeersToPBPeers converts given []peer.Peer into a set of []*Message_Peer,
// which can be written to a message and sent out. the key thing this function
// does (in addition to PeersToPBPeers) is set the ConnectionType with
76
// information from the given inet.Network.
Jeromy's avatar
Jeromy committed
77
func PeerInfosToPBPeers(n inet.Network, peers []pstore.PeerInfo) []*Message_Peer {
78
	pbps := RawPeerInfosToPBPeers(peers)
79
	for i, pbp := range pbps {
80
		c := ConnectionType(n.Connectedness(peers[i].ID))
81 82 83 84 85
		pbp.Connection = &c
	}
	return pbps
}

86 87 88 89 90 91 92 93
func PeerRoutingInfosToPBPeers(peers []PeerRoutingInfo) []*Message_Peer {
	pbpeers := make([]*Message_Peer, len(peers))
	for i, p := range peers {
		pbpeers[i] = peerRoutingInfoToPBPeer(p)
	}
	return pbpeers
}

Jeromy's avatar
Jeromy committed
94
// PBPeersToPeerInfos converts given []*Message_Peer into []pstore.PeerInfo
95
// Invalid addresses will be silently omitted.
Jeromy's avatar
Jeromy committed
96 97
func PBPeersToPeerInfos(pbps []*Message_Peer) []*pstore.PeerInfo {
	peers := make([]*pstore.PeerInfo, 0, len(pbps))
98 99
	for _, pbp := range pbps {
		peers = append(peers, PBPeerToPeerInfo(pbp))
100
	}
101
	return peers
102 103
}

104
// Addresses returns a multiaddr associated with the Message_Peer entry
105
func (m *Message_Peer) Addresses() []ma.Multiaddr {
106
	if m == nil {
107
		return nil
108
	}
109

110 111 112
	maddrs := make([]ma.Multiaddr, 0, len(m.Addrs))
	for _, addr := range m.Addrs {
		maddr, err := ma.NewMultiaddrBytes(addr)
113
		if err != nil {
114
			log.Warningf("error decoding Multiaddr for peer: %s", m.GetId())
115
			continue
116
		}
117 118

		maddrs = append(maddrs, maddr)
119
	}
120
	return maddrs
121 122
}

123 124 125
// GetClusterLevel gets and adjusts the cluster level on the message.
// a +/- 1 adjustment is needed to distinguish a valid first level (1) and
// default "no value" protobuf behavior (0)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
126
func (m *Message) GetClusterLevel() int {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
127 128
	level := m.GetClusterLevelRaw() - 1
	if level < 0 {
129
		return 0
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
130
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
131
	return int(level)
132
}
133

134 135 136
// SetClusterLevel adjusts and sets the cluster level on the message.
// a +/- 1 adjustment is needed to distinguish a valid first level (1) and
// default "no value" protobuf behavior (0)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
137 138 139
func (m *Message) SetClusterLevel(level int) {
	lvl := int32(level)
	m.ClusterLevelRaw = &lvl
140
}
141

142
// Loggable turns a Message into machine-readable log output
143 144 145 146
func (m *Message) Loggable() map[string]interface{} {
	return map[string]interface{}{
		"message": map[string]string{
			"type": m.Type.String(),
Jeromy's avatar
Jeromy committed
147
			"key":  b58.Encode([]byte(m.GetKey())),
148 149 150
		},
	}
}
151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184

// ConnectionType returns a Message_ConnectionType associated with the
// inet.Connectedness.
func ConnectionType(c inet.Connectedness) Message_ConnectionType {
	switch c {
	default:
		return Message_NOT_CONNECTED
	case inet.NotConnected:
		return Message_NOT_CONNECTED
	case inet.Connected:
		return Message_CONNECTED
	case inet.CanConnect:
		return Message_CAN_CONNECT
	case inet.CannotConnect:
		return Message_CANNOT_CONNECT
	}
}

// Connectedness returns an inet.Connectedness associated with the
// Message_ConnectionType.
func Connectedness(c Message_ConnectionType) inet.Connectedness {
	switch c {
	default:
		return inet.NotConnected
	case Message_NOT_CONNECTED:
		return inet.NotConnected
	case Message_CONNECTED:
		return inet.Connected
	case Message_CAN_CONNECT:
		return inet.CanConnect
	case Message_CANNOT_CONNECT:
		return inet.CannotConnect
	}
}