mirror of
https://github.com/status-im/go-waku.git
synced 2025-01-11 22:34:43 +00:00
552 lines
13 KiB
Go
552 lines
13 KiB
Go
package discv5
|
|
|
|
import (
|
|
"context"
|
|
"crypto/ecdsa"
|
|
"errors"
|
|
"fmt"
|
|
"math"
|
|
"math/rand"
|
|
"net"
|
|
"strconv"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
|
"github.com/ethereum/go-ethereum/p2p/enr"
|
|
"github.com/ethereum/go-ethereum/p2p/nat"
|
|
"github.com/libp2p/go-libp2p-core/discovery"
|
|
"github.com/libp2p/go-libp2p-core/host"
|
|
"github.com/libp2p/go-libp2p-core/peer"
|
|
ma "github.com/multiformats/go-multiaddr"
|
|
"github.com/status-im/go-discover/discover"
|
|
"github.com/status-im/go-waku/waku/v2/utils"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
type DiscoveryV5 struct {
|
|
sync.Mutex
|
|
|
|
discovery.Discovery
|
|
|
|
params *discV5Parameters
|
|
host host.Host
|
|
config discover.Config
|
|
udpAddr *net.UDPAddr
|
|
listener *discover.UDPv5
|
|
localnode *enode.LocalNode
|
|
NAT nat.Interface
|
|
quit chan struct{}
|
|
|
|
log *zap.SugaredLogger
|
|
|
|
wg *sync.WaitGroup
|
|
|
|
peerCache peerCache
|
|
|
|
// Used for those weird cases where updateAddress
|
|
// receives the same external address twice both with the original port
|
|
// and the nat port. Ideally this attribute should be removed by doing
|
|
// hole punching before starting waku
|
|
ogTCPPort int
|
|
}
|
|
|
|
type peerCache struct {
|
|
sync.RWMutex
|
|
recs map[peer.ID]peerRecord
|
|
rng *rand.Rand
|
|
}
|
|
|
|
type peerRecord struct {
|
|
expire int64
|
|
peer peer.AddrInfo
|
|
}
|
|
|
|
type discV5Parameters struct {
|
|
autoUpdate bool
|
|
bootnodes []*enode.Node
|
|
udpPort int
|
|
advertiseAddr *net.IP
|
|
}
|
|
|
|
type DiscoveryV5Option func(*discV5Parameters)
|
|
|
|
func WithAutoUpdate(autoUpdate bool) DiscoveryV5Option {
|
|
return func(params *discV5Parameters) {
|
|
params.autoUpdate = autoUpdate
|
|
}
|
|
}
|
|
|
|
func WithBootnodes(bootnodes []*enode.Node) DiscoveryV5Option {
|
|
return func(params *discV5Parameters) {
|
|
params.bootnodes = bootnodes
|
|
}
|
|
}
|
|
|
|
func WithAdvertiseAddr(addr net.IP) DiscoveryV5Option {
|
|
return func(params *discV5Parameters) {
|
|
params.advertiseAddr = &addr
|
|
}
|
|
}
|
|
|
|
func WithUDPPort(port int) DiscoveryV5Option {
|
|
return func(params *discV5Parameters) {
|
|
params.udpPort = port
|
|
}
|
|
}
|
|
|
|
func DefaultOptions() []DiscoveryV5Option {
|
|
return []DiscoveryV5Option{
|
|
WithUDPPort(9000),
|
|
}
|
|
}
|
|
|
|
func NewDiscoveryV5(host host.Host, addresses []ma.Multiaddr, priv *ecdsa.PrivateKey, wakuFlags utils.WakuEnrBitfield, log *zap.SugaredLogger, opts ...DiscoveryV5Option) (*DiscoveryV5, error) {
|
|
params := new(discV5Parameters)
|
|
optList := DefaultOptions()
|
|
optList = append(optList, opts...)
|
|
for _, opt := range optList {
|
|
opt(params)
|
|
}
|
|
|
|
logger := log.Named("discv5")
|
|
|
|
ipAddr, err := selectIPAddr(addresses)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
localnode, err := newLocalnode(priv, ipAddr, params.udpPort, wakuFlags, params.advertiseAddr, logger)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var NAT nat.Interface = nil
|
|
if params.advertiseAddr == nil {
|
|
NAT = nat.Any()
|
|
}
|
|
|
|
return &DiscoveryV5{
|
|
host: host,
|
|
params: params,
|
|
NAT: NAT,
|
|
wg: &sync.WaitGroup{},
|
|
peerCache: peerCache{
|
|
rng: rand.New(rand.NewSource(rand.Int63())),
|
|
recs: make(map[peer.ID]peerRecord),
|
|
},
|
|
localnode: localnode,
|
|
config: discover.Config{
|
|
PrivateKey: priv,
|
|
Bootnodes: params.bootnodes,
|
|
ValidNodeFn: func(n enode.Node) bool {
|
|
// TODO: track https://github.com/status-im/nim-waku/issues/770 for improvements over validation func
|
|
return evaluateNode(&n)
|
|
},
|
|
V5Config: discover.V5Config{
|
|
ProtocolID: [6]byte{'d', '5', 'w', 'a', 'k', 'u'},
|
|
},
|
|
},
|
|
udpAddr: &net.UDPAddr{
|
|
IP: net.IPv4zero,
|
|
Port: params.udpPort,
|
|
},
|
|
log: logger,
|
|
ogTCPPort: ipAddr.Port,
|
|
}, nil
|
|
}
|
|
|
|
func newLocalnode(priv *ecdsa.PrivateKey, ipAddr *net.TCPAddr, udpPort int, wakuFlags utils.WakuEnrBitfield, advertiseAddr *net.IP, log *zap.SugaredLogger) (*enode.LocalNode, error) {
|
|
db, err := enode.OpenDB("")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
localnode := enode.NewLocalNode(db, priv)
|
|
localnode.SetFallbackUDP(udpPort)
|
|
localnode.Set(enr.WithEntry(utils.WakuENRField, wakuFlags))
|
|
localnode.SetFallbackIP(net.IP{127, 0, 0, 1})
|
|
localnode.SetStaticIP(ipAddr.IP)
|
|
|
|
if udpPort > 0 && udpPort <= math.MaxUint16 {
|
|
localnode.Set(enr.UDP(uint16(udpPort))) // lgtm [go/incorrect-integer-conversion]
|
|
} else {
|
|
log.Error("could not set udpPort ", zap.Int("port", udpPort))
|
|
}
|
|
|
|
if ipAddr.Port > 0 && ipAddr.Port <= math.MaxUint16 {
|
|
localnode.Set(enr.TCP(uint16(ipAddr.Port))) // lgtm [go/incorrect-integer-conversion]
|
|
} else {
|
|
log.Error("could not set tcpPort ", zap.Int("port", ipAddr.Port))
|
|
}
|
|
|
|
if advertiseAddr != nil {
|
|
localnode.SetStaticIP(*advertiseAddr)
|
|
}
|
|
|
|
return localnode, nil
|
|
}
|
|
|
|
func (d *DiscoveryV5) listen() error {
|
|
conn, err := net.ListenUDP("udp", d.udpAddr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
d.udpAddr = conn.LocalAddr().(*net.UDPAddr)
|
|
|
|
if d.NAT != nil && !d.udpAddr.IP.IsLoopback() {
|
|
d.wg.Add(1)
|
|
go func() {
|
|
defer d.wg.Done()
|
|
nat.Map(d.NAT, d.quit, "udp", d.udpAddr.Port, d.udpAddr.Port, "go-waku discv5 discovery")
|
|
}()
|
|
|
|
}
|
|
|
|
d.localnode.SetFallbackUDP(d.udpAddr.Port)
|
|
|
|
listener, err := discover.ListenV5(conn, d.localnode, d.config)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
d.listener = listener
|
|
|
|
d.log.Info(fmt.Sprintf("Started Discovery V5 at %s:%d, advertising IP: %s:%d", d.udpAddr.IP, d.udpAddr.Port, d.localnode.Node().IP(), d.localnode.Node().TCP()))
|
|
d.log.Info("Discv5: discoverable ENR ", d.localnode.Node())
|
|
|
|
return nil
|
|
}
|
|
|
|
func (d *DiscoveryV5) Start() error {
|
|
d.Lock()
|
|
defer d.Unlock()
|
|
|
|
d.wg.Wait() // Waiting for other go routines to stop
|
|
|
|
d.quit = make(chan struct{}, 1)
|
|
|
|
err := d.listen()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (d *DiscoveryV5) Stop() {
|
|
d.Lock()
|
|
defer d.Unlock()
|
|
|
|
close(d.quit)
|
|
|
|
d.listener.Close()
|
|
d.listener = nil
|
|
|
|
d.log.Info("Stopped Discovery V5")
|
|
|
|
d.wg.Wait()
|
|
}
|
|
|
|
func (d *DiscoveryV5) UpdateAddr(addr *net.TCPAddr) error {
|
|
if !d.params.autoUpdate {
|
|
return nil
|
|
}
|
|
|
|
d.Lock()
|
|
defer d.Unlock()
|
|
|
|
// TODO: This code is not elegant and should be improved
|
|
|
|
if !isExternal(addr) && !isExternal(&net.TCPAddr{IP: d.localnode.Node().IP()}) {
|
|
if !((d.localnode.Node().IP().IsLoopback() && isPrivate(addr)) || (isPrivate(&net.TCPAddr{IP: d.localnode.Node().IP()}) && isExternal(addr))) {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
if addr.IP.IsUnspecified() || (d.localnode.Node().IP().Equal(addr.IP) && addr.Port == d.ogTCPPort) {
|
|
return nil
|
|
}
|
|
|
|
d.localnode.SetStaticIP(addr.IP)
|
|
d.localnode.Set(enr.TCP(uint16(addr.Port))) // lgtm [go/incorrect-integer-conversion]
|
|
d.log.Info(fmt.Sprintf("Updated Discovery V5 node: %s:%d", d.localnode.Node().IP(), d.localnode.Node().TCP()))
|
|
d.log.Info("Discovery V5 ", d.localnode.Node())
|
|
|
|
return nil
|
|
}
|
|
|
|
/*
|
|
func isWakuNode(node *enode.Node) bool {
|
|
enrField := new(utils.WakuEnrBitfield)
|
|
if err := node.Record().Load(enr.WithEntry(utils.WakuENRField, &enrField)); err != nil {
|
|
if !enr.IsNotFound(err) {
|
|
utils.Logger().Named("discv5").Error("could not retrieve port for enr ", zap.Any("node", node))
|
|
}
|
|
return false
|
|
}
|
|
|
|
if enrField != nil {
|
|
return *enrField != uint8(0)
|
|
}
|
|
|
|
return false
|
|
}
|
|
*/
|
|
|
|
func hasTCPPort(node *enode.Node) bool {
|
|
enrTCP := new(enr.TCP)
|
|
if err := node.Record().Load(enr.WithEntry(enrTCP.ENRKey(), enrTCP)); err != nil {
|
|
if !enr.IsNotFound(err) {
|
|
utils.Logger().Named("discv5").Error("could not retrieve port for enr ", zap.Any("node", node))
|
|
}
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func evaluateNode(node *enode.Node) bool {
|
|
if node == nil || node.IP() == nil {
|
|
return false
|
|
}
|
|
|
|
// TODO: consider node filtering based on ENR; we do not filter based on ENR in the first waku discv5 beta stage
|
|
if /*!isWakuNode(node) ||*/ !hasTCPPort(node) {
|
|
return false
|
|
}
|
|
|
|
_, err := utils.EnodeToPeerInfo(node)
|
|
|
|
if err != nil {
|
|
utils.Logger().Named("discv5").Error("could not obtain peer info from enode:", zap.Error(err))
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func (d *DiscoveryV5) Advertise(ctx context.Context, ns string, opts ...discovery.Option) (time.Duration, error) {
|
|
// Get options
|
|
var options discovery.Options
|
|
err := options.Apply(opts...)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// TODO: once discv5 spec introduces capability and topic discovery, implement this function
|
|
|
|
return 20 * time.Minute, nil
|
|
}
|
|
|
|
func (d *DiscoveryV5) iterate(ctx context.Context, iterator enode.Iterator, limit int, doneCh chan struct{}) {
|
|
defer d.wg.Done()
|
|
|
|
for {
|
|
if len(d.peerCache.recs) >= limit {
|
|
break
|
|
}
|
|
|
|
if ctx.Err() != nil {
|
|
break
|
|
}
|
|
|
|
exists := iterator.Next()
|
|
if !exists {
|
|
break
|
|
}
|
|
|
|
addresses, err := utils.Multiaddress(iterator.Node())
|
|
if err != nil {
|
|
d.log.Error(err)
|
|
continue
|
|
}
|
|
|
|
peerAddrs, err := peer.AddrInfosFromP2pAddrs(addresses...)
|
|
if err != nil {
|
|
d.log.Error(err)
|
|
continue
|
|
}
|
|
|
|
for _, p := range peerAddrs {
|
|
d.peerCache.recs[p.ID] = peerRecord{
|
|
expire: time.Now().Unix() + 3600, // Expires in 1hr
|
|
peer: p,
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
close(doneCh)
|
|
}
|
|
|
|
func (d *DiscoveryV5) removeExpiredPeers() int {
|
|
// Remove all expired entries from cache
|
|
currentTime := time.Now().Unix()
|
|
newCacheSize := len(d.peerCache.recs)
|
|
|
|
for p := range d.peerCache.recs {
|
|
rec := d.peerCache.recs[p]
|
|
if rec.expire < currentTime {
|
|
newCacheSize--
|
|
delete(d.peerCache.recs, p)
|
|
}
|
|
}
|
|
|
|
return newCacheSize
|
|
}
|
|
|
|
func (d *DiscoveryV5) FindPeers(ctx context.Context, topic string, opts ...discovery.Option) (<-chan peer.AddrInfo, error) {
|
|
// Get options
|
|
var options discovery.Options
|
|
err := options.Apply(opts...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
const maxLimit = 100
|
|
limit := options.Limit
|
|
if limit == 0 || limit > maxLimit {
|
|
limit = maxLimit
|
|
}
|
|
|
|
// We are ignoring the topic. Future versions might use a map[string]*peerCache instead where the string represents the pubsub topic
|
|
|
|
d.peerCache.Lock()
|
|
defer d.peerCache.Unlock()
|
|
|
|
cacheSize := d.removeExpiredPeers()
|
|
|
|
// Discover new records if we don't have enough
|
|
if cacheSize < limit && d.listener != nil {
|
|
d.Lock()
|
|
|
|
iterator := d.listener.RandomNodes()
|
|
iterator = enode.Filter(iterator, evaluateNode)
|
|
defer iterator.Close()
|
|
|
|
doneCh := make(chan struct{})
|
|
|
|
d.wg.Add(1)
|
|
go d.iterate(ctx, iterator, limit, doneCh)
|
|
|
|
select {
|
|
case <-ctx.Done():
|
|
case <-doneCh:
|
|
}
|
|
|
|
d.Unlock()
|
|
}
|
|
|
|
// Randomize and fill channel with available records
|
|
count := len(d.peerCache.recs)
|
|
if limit < count {
|
|
count = limit
|
|
}
|
|
|
|
chPeer := make(chan peer.AddrInfo, count)
|
|
|
|
perm := d.peerCache.rng.Perm(len(d.peerCache.recs))[0:count]
|
|
permSet := make(map[int]int)
|
|
for i, v := range perm {
|
|
permSet[v] = i
|
|
}
|
|
|
|
sendLst := make([]*peer.AddrInfo, count)
|
|
iter := 0
|
|
for k := range d.peerCache.recs {
|
|
if sendIndex, ok := permSet[iter]; ok {
|
|
peerInfo := d.peerCache.recs[k].peer
|
|
sendLst[sendIndex] = &peerInfo
|
|
}
|
|
iter++
|
|
}
|
|
|
|
for _, send := range sendLst {
|
|
chPeer <- *send
|
|
}
|
|
|
|
close(chPeer)
|
|
|
|
return chPeer, err
|
|
}
|
|
|
|
// IsPrivate reports whether ip is a private address, according to
|
|
// RFC 1918 (IPv4 addresses) and RFC 4193 (IPv6 addresses).
|
|
// Copied/Adapted from https://go-review.googlesource.com/c/go/+/272668/11/src/net/ip.go
|
|
// Copyright (c) The Go Authors. All rights reserved.
|
|
// @TODO: once Go 1.17 is released in Q42021, remove this function as it will become part of the language
|
|
func isPrivate(ip *net.TCPAddr) bool {
|
|
if ip4 := ip.IP.To4(); ip4 != nil {
|
|
// Following RFC 4193, Section 3. Local IPv6 Unicast Addresses which says:
|
|
// The Internet Assigned Numbers Authority (IANA) has reserved the
|
|
// following three blocks of the IPv4 address space for private internets:
|
|
// 10.0.0.0 - 10.255.255.255 (10/8 prefix)
|
|
// 172.16.0.0 - 172.31.255.255 (172.16/12 prefix)
|
|
// 192.168.0.0 - 192.168.255.255 (192.168/16 prefix)
|
|
return ip4[0] == 10 ||
|
|
(ip4[0] == 172 && ip4[1]&0xf0 == 16) ||
|
|
(ip4[0] == 192 && ip4[1] == 168)
|
|
}
|
|
// Following RFC 4193, Section 3. Private Address Space which says:
|
|
// The Internet Assigned Numbers Authority (IANA) has reserved the
|
|
// following block of the IPv6 address space for local internets:
|
|
// FC00:: - FDFF:FFFF:FFFF:FFFF:FFFF:FFFF:FFFF:FFFF (FC00::/7 prefix)
|
|
return len(ip.IP) == net.IPv6len && ip.IP[0]&0xfe == 0xfc
|
|
}
|
|
|
|
func isExternal(ip *net.TCPAddr) bool {
|
|
return !isPrivate(ip) && !ip.IP.IsLoopback() && !ip.IP.IsUnspecified()
|
|
}
|
|
|
|
func isLoopback(ip *net.TCPAddr) bool {
|
|
return ip.IP.IsLoopback()
|
|
}
|
|
func filter(ss []*net.TCPAddr, fn func(*net.TCPAddr) bool) (ret []*net.TCPAddr) {
|
|
for _, s := range ss {
|
|
if fn(s) {
|
|
ret = append(ret, s)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func selectIPAddr(addresses []ma.Multiaddr) (*net.TCPAddr, error) {
|
|
var ipAddrs []*net.TCPAddr
|
|
for _, addr := range addresses {
|
|
ipStr, err := addr.ValueForProtocol(ma.P_IP4)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
portStr, err := addr.ValueForProtocol(ma.P_TCP)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
port, err := strconv.Atoi(portStr)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
ipAddrs = append(ipAddrs, &net.TCPAddr{
|
|
IP: net.ParseIP(ipStr),
|
|
Port: port,
|
|
})
|
|
}
|
|
|
|
externalIPs := filter(ipAddrs, isExternal)
|
|
if len(externalIPs) > 0 {
|
|
return externalIPs[0], nil
|
|
}
|
|
|
|
privateIPs := filter(ipAddrs, isPrivate)
|
|
if len(privateIPs) > 0 {
|
|
return privateIPs[0], nil
|
|
}
|
|
|
|
loopback := filter(ipAddrs, isLoopback)
|
|
if len(loopback) > 0 {
|
|
return loopback[0], nil
|
|
}
|
|
|
|
return nil, errors.New("could not obtain ip address")
|
|
}
|