mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-07 18:21:20 +00:00
95 lines
2.3 KiB
Go
95 lines
2.3 KiB
Go
|
// Bootstrap / DHT query tool
|
||
|
//
|
||
|
// Usage: bazel run //tools/boostrap-query -- $BOOTNODE_ADDRESS
|
||
|
//
|
||
|
// This tool queries the bootstrap / DHT node for peers then attempts to dial
|
||
|
// and ping each of them.
|
||
|
package main
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"log"
|
||
|
"os"
|
||
|
|
||
|
ggio "github.com/gogo/protobuf/io"
|
||
|
"github.com/libp2p/go-libp2p"
|
||
|
host "github.com/libp2p/go-libp2p-host"
|
||
|
dhtpb "github.com/libp2p/go-libp2p-kad-dht/pb"
|
||
|
net "github.com/libp2p/go-libp2p-net"
|
||
|
p2p "github.com/prysmaticlabs/prysm/beacon-chain/p2p"
|
||
|
)
|
||
|
|
||
|
const dhtProtocol = "/prysm/0.0.0/dht"
|
||
|
|
||
|
func main() {
|
||
|
if len(os.Args) == 1 {
|
||
|
log.Fatal("Error: Bootnode address not provided.")
|
||
|
}
|
||
|
|
||
|
ctx := context.Background()
|
||
|
h, err := libp2p.New(ctx)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
addr := os.Args[1]
|
||
|
pi, err := p2p.MakePeer(addr)
|
||
|
if err != nil {
|
||
|
log.Fatalf("Error: Failed to make peer from string: %v", err)
|
||
|
}
|
||
|
if err := h.Connect(ctx, *pi); err != nil {
|
||
|
log.Fatalf("Error: Failed to create peer from string: %v", err)
|
||
|
}
|
||
|
|
||
|
s, err := h.NewStream(ctx, pi.ID, dhtProtocol)
|
||
|
if err != nil {
|
||
|
log.Printf("proto = %s", dhtProtocol)
|
||
|
log.Fatalf("Error: Failed to create ProtocolDHT stream: %v", err)
|
||
|
}
|
||
|
|
||
|
resp := sendMessageAndWait(s, dhtpb.NewMessage(dhtpb.Message_FIND_NODE, []byte{}, 0))
|
||
|
|
||
|
log.Printf("Bootstrap DHT node has %d peers\n", len(resp.GetCloserPeers()))
|
||
|
for i, p := range resp.GetCloserPeers() {
|
||
|
log.Printf("Dialing peer %d: %+v\n", i, p.Addresses())
|
||
|
|
||
|
if err := pingPeer(ctx, h, p); err != nil {
|
||
|
log.Printf("Error: unable to ping peer %v\n", err)
|
||
|
} else {
|
||
|
log.Println("OK")
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func pingPeer(ctx context.Context, h host.Host, p *dhtpb.Message_Peer) error {
|
||
|
pi := dhtpb.PBPeerToPeerInfo(p)
|
||
|
if err := h.Connect(ctx, *pi); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
s, err := h.NewStream(ctx, pi.ID, dhtProtocol)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// Any response is OK
|
||
|
_ = sendMessageAndWait(s, dhtpb.NewMessage(dhtpb.Message_PING, []byte{}, 0))
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func sendMessageAndWait(s net.Stream, msg *dhtpb.Message) dhtpb.Message {
|
||
|
r := ggio.NewDelimitedReader(s, 2000000)
|
||
|
w := ggio.NewDelimitedWriter(s)
|
||
|
|
||
|
if err := w.WriteMsg(msg); err != nil {
|
||
|
log.Fatalf("Error: failed to write message: %v", err)
|
||
|
}
|
||
|
|
||
|
var resp dhtpb.Message
|
||
|
if err := r.ReadMsg(&resp); err != nil {
|
||
|
log.Fatalf("Error: failed to read message: %v", err)
|
||
|
}
|
||
|
|
||
|
return resp
|
||
|
}
|