2023-07-05 18:02:21 -04:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"strings"
|
|
|
|
"syscall"
|
2023-07-06 09:32:41 -04:00
|
|
|
"time"
|
2023-07-05 18:02:21 -04:00
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
2023-10-18 19:57:38 -04:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/dnsdisc"
|
2023-07-05 18:02:21 -04:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
2023-10-18 11:34:19 -04:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/enr"
|
2023-07-05 18:02:21 -04:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/nat"
|
|
|
|
"github.com/waku-org/go-discover/discover"
|
2023-10-18 20:07:01 -04:00
|
|
|
wenr "github.com/waku-org/go-waku/waku/v2/protocol/enr"
|
2024-06-25 21:19:39 +05:30
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/filter"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/lightpush"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/relay"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/store"
|
2023-07-05 18:02:21 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
var portFlag = flag.Int("port", 6000, "port number")
|
|
|
|
var bootnodeFlag = flag.String("bootnodes", "", "comma separated bootnodes")
|
2023-10-18 19:57:38 -04:00
|
|
|
var dnsdiscFlag = flag.String("dns-disc-url", "", "DNS discovery URL")
|
|
|
|
|
2023-07-05 18:02:21 -04:00
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
db, err := enode.OpenDB("")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ip := GetLocalIP()
|
|
|
|
|
|
|
|
priv, _ := crypto.GenerateKey()
|
|
|
|
localnode := enode.NewLocalNode(db, priv)
|
|
|
|
localnode.SetFallbackIP(ip)
|
|
|
|
|
|
|
|
var protocolID = [6]byte{'d', '5', 'w', 'a', 'k', 'u'}
|
|
|
|
|
|
|
|
bootnodesStr := strings.Split(*bootnodeFlag, ",")
|
|
|
|
var bootnodes []*enode.Node
|
2023-10-18 19:57:38 -04:00
|
|
|
|
2023-07-05 18:02:21 -04:00
|
|
|
for _, addr := range bootnodesStr {
|
|
|
|
if addr == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
bootnode, err := enode.Parse(enode.ValidSchemes, addr)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
bootnodes = append(bootnodes, bootnode)
|
|
|
|
}
|
|
|
|
|
2023-10-18 19:57:38 -04:00
|
|
|
if *dnsdiscFlag != "" {
|
|
|
|
c := dnsdisc.NewClient(dnsdisc.Config{})
|
|
|
|
tree, err := c.SyncTree(*dnsdiscFlag)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
bootnodes = append(bootnodes, tree.Nodes()...)
|
|
|
|
}
|
|
|
|
|
2023-07-05 18:02:21 -04:00
|
|
|
config := discover.Config{
|
2023-10-18 20:07:01 -04:00
|
|
|
PrivateKey: priv,
|
|
|
|
Bootnodes: bootnodes,
|
|
|
|
V5Config: discover.V5Config{
|
|
|
|
ProtocolID: &protocolID,
|
|
|
|
},
|
2023-07-05 18:02:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
udpAddr := &net.UDPAddr{
|
|
|
|
IP: ip,
|
|
|
|
Port: *portFlag,
|
|
|
|
}
|
|
|
|
|
|
|
|
conn, err := net.ListenUDP("udp", udpAddr)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
udpAddr = conn.LocalAddr().(*net.UDPAddr)
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
if !udpAddr.IP.IsLoopback() {
|
|
|
|
go func() {
|
|
|
|
nat.Map(nat.Any(), ctx.Done(), "udp", udpAddr.Port, udpAddr.Port, "test discv5 discovery")
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
localnode.SetFallbackUDP(udpAddr.Port)
|
|
|
|
|
2024-06-17 16:38:05 -04:00
|
|
|
fmt.Println("Your node:")
|
2023-07-05 18:02:21 -04:00
|
|
|
fmt.Println(localnode.Node())
|
|
|
|
listener, err := discover.ListenV5(conn, localnode, config)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2024-06-17 16:38:05 -04:00
|
|
|
if len(bootnodes) > 0 {
|
|
|
|
fmt.Println("\nBootnodes:")
|
|
|
|
for i, b := range bootnodes {
|
|
|
|
fmt.Println(i+1, "-", b.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-06 09:32:41 -04:00
|
|
|
peerDelay := 50 * time.Millisecond
|
|
|
|
bucketSize := 15
|
|
|
|
|
|
|
|
fmt.Println("\nDiscovered peers:")
|
|
|
|
fmt.Println("===============================================================================")
|
|
|
|
|
2023-07-05 18:02:21 -04:00
|
|
|
go func() {
|
|
|
|
iterator := listener.RandomNodes()
|
2023-10-18 11:34:19 -04:00
|
|
|
seen := make(map[enode.ID]*enode.Node)
|
2023-07-06 09:32:41 -04:00
|
|
|
peerCnt := 0
|
2023-07-05 18:02:21 -04:00
|
|
|
for iterator.Next() {
|
2023-07-06 09:32:41 -04:00
|
|
|
|
|
|
|
start := time.Now()
|
|
|
|
hasNext := iterator.Next()
|
|
|
|
if !hasNext {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
elapsed := time.Since(start)
|
|
|
|
|
|
|
|
if elapsed < peerDelay {
|
|
|
|
t := time.NewTimer(peerDelay - elapsed)
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
case <-t.C:
|
|
|
|
t.Stop()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delay every 15 peers being returned
|
|
|
|
peerCnt++
|
|
|
|
if peerCnt == bucketSize {
|
|
|
|
peerCnt = 0
|
|
|
|
t := time.NewTimer(5 * time.Second)
|
|
|
|
fmt.Println("Waiting 5 secs...")
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
case <-t.C:
|
|
|
|
t.Stop()
|
|
|
|
fmt.Println("Attempting to discover more peers")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-05 18:02:21 -04:00
|
|
|
node := iterator.Node()
|
2023-07-06 09:32:41 -04:00
|
|
|
_ = node
|
2023-10-18 11:34:19 -04:00
|
|
|
n, ok := seen[node.ID()]
|
|
|
|
|
|
|
|
recType := "NEW"
|
|
|
|
|
2023-07-05 18:02:21 -04:00
|
|
|
if ok {
|
2023-10-18 11:34:19 -04:00
|
|
|
if node.Seq() > n.Seq() {
|
|
|
|
seen[node.ID()] = node
|
|
|
|
recType = "UPDATE"
|
|
|
|
} else {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
seen[node.ID()] = node
|
2023-07-05 18:02:21 -04:00
|
|
|
}
|
2023-10-18 11:34:19 -04:00
|
|
|
|
2023-10-18 19:57:38 -04:00
|
|
|
fmt.Println(len(seen), "-", recType, "-", node.String())
|
2023-10-18 20:07:01 -04:00
|
|
|
|
|
|
|
peerID, multiaddresses, err := wenr.Multiaddress(node)
|
|
|
|
if err == nil {
|
|
|
|
fmt.Println("peerID", peerID)
|
|
|
|
if len(multiaddresses) > 0 {
|
|
|
|
fmt.Println("multiaddr:", multiaddresses)
|
|
|
|
} else {
|
|
|
|
fmt.Println("multiaddr: field contains no value")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fmt.Println("multiaddr:", err.Error())
|
|
|
|
}
|
|
|
|
|
2023-10-18 19:57:38 -04:00
|
|
|
fmt.Println(fmt.Sprintf("ip %s:%d", node.IP(), node.TCP()))
|
2024-06-25 21:19:39 +05:30
|
|
|
shards, err := wenr.RelaySharding(node.Record())
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("rs/rsv:", "field contains no value")
|
2023-10-18 11:34:19 -04:00
|
|
|
}
|
2024-06-25 21:19:39 +05:30
|
|
|
if shards != nil {
|
|
|
|
fmt.Println("cluster-id: ", shards.ClusterID)
|
|
|
|
fmt.Println("shards: ", shards.ShardIDs)
|
2023-10-18 20:07:01 -04:00
|
|
|
} else {
|
2024-06-25 21:19:39 +05:30
|
|
|
fmt.Println("cluster-id:", "not available")
|
|
|
|
fmt.Println("shards:", "not available")
|
2023-10-18 11:34:19 -04:00
|
|
|
}
|
2024-06-25 21:19:39 +05:30
|
|
|
DecodeWaku2ENRField(node.Record())
|
2023-10-18 11:34:19 -04:00
|
|
|
fmt.Println()
|
2023-07-05 18:02:21 -04:00
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
iterator.Close()
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Wait for a SIGINT or SIGTERM signal
|
|
|
|
ch := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(ch, syscall.SIGINT, syscall.SIGTERM)
|
|
|
|
<-ch
|
|
|
|
fmt.Println("\n\n\nshutting down...")
|
|
|
|
|
|
|
|
cancel()
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-10-18 11:34:19 -04:00
|
|
|
func ReadValue(record *enr.Record, name string) ([]byte, error) {
|
|
|
|
var field []byte
|
|
|
|
if err := record.Load(enr.WithEntry(name, &field)); err != nil {
|
|
|
|
if enr.IsNotFound(err) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return field, nil
|
|
|
|
}
|
|
|
|
|
2023-07-05 18:02:21 -04:00
|
|
|
// GetLocalIP returns the non loopback local IP of the host
|
|
|
|
func GetLocalIP() net.IP {
|
|
|
|
addrs, err := net.InterfaceAddrs()
|
|
|
|
if err != nil {
|
|
|
|
return net.IPv4zero
|
|
|
|
}
|
|
|
|
for _, address := range addrs {
|
|
|
|
// check the address type and if it is not a loopback the display it
|
|
|
|
if ipnet, ok := address.(*net.IPNet); ok && !ipnet.IP.IsLoopback() {
|
|
|
|
if ipnet.IP.To4() != nil {
|
|
|
|
return ipnet.IP
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return net.IPv4zero
|
|
|
|
}
|
2024-06-25 21:19:39 +05:30
|
|
|
|
|
|
|
func DecodeWaku2ENRField(record *enr.Record) {
|
|
|
|
//Decoding Waku2 field
|
|
|
|
var enrField wenr.WakuEnrBitfield
|
|
|
|
var protosSupported []string
|
|
|
|
if err := record.Load(enr.WithEntry("waku2", &enrField)); err != nil {
|
|
|
|
if enr.IsNotFound(err) {
|
|
|
|
fmt.Println("waku2:", "field contains no value")
|
|
|
|
} else {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if enrField&relay.WakuRelayENRField != 0 {
|
|
|
|
protosSupported = append(protosSupported, string(relay.WakuRelayID_v200))
|
|
|
|
}
|
|
|
|
if enrField&filter.FilterSubscribeENRField != 0 {
|
|
|
|
protosSupported = append(protosSupported, string(filter.FilterSubscribeID_v20beta1))
|
|
|
|
}
|
|
|
|
if enrField&lightpush.LightPushENRField != 0 {
|
|
|
|
protosSupported = append(protosSupported, string(lightpush.LightPushID_v20beta1))
|
|
|
|
}
|
|
|
|
if enrField&store.StoreENRField != 0 {
|
|
|
|
protosSupported = append(protosSupported, string(store.StoreID_v20beta4))
|
|
|
|
}
|
|
|
|
fmt.Println("Wakuv2 Protocols Supported:")
|
|
|
|
for _, proto := range protosSupported {
|
|
|
|
fmt.Println(proto)
|
|
|
|
}
|
|
|
|
}
|