2019-08-16 17:13:04 +00:00
|
|
|
package testing
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gogo/protobuf/proto"
|
|
|
|
bhost "github.com/libp2p/go-libp2p-blankhost"
|
|
|
|
"github.com/libp2p/go-libp2p-core/host"
|
|
|
|
"github.com/libp2p/go-libp2p-core/network"
|
2019-08-16 20:03:11 +00:00
|
|
|
"github.com/libp2p/go-libp2p-core/peer"
|
2019-08-16 17:13:04 +00:00
|
|
|
"github.com/libp2p/go-libp2p-core/protocol"
|
|
|
|
pubsub "github.com/libp2p/go-libp2p-pubsub"
|
|
|
|
swarmt "github.com/libp2p/go-libp2p-swarm/testing"
|
2019-08-16 20:03:11 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-08-16 17:13:04 +00:00
|
|
|
|
2019-08-18 04:32:39 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/p2p/encoder"
|
|
|
|
)
|
2019-08-16 17:13:04 +00:00
|
|
|
|
|
|
|
// TestP2P represents a p2p implementation that can be used for testing.
|
|
|
|
type TestP2P struct {
|
|
|
|
t *testing.T
|
2019-08-16 20:03:11 +00:00
|
|
|
Host host.Host
|
2019-08-16 17:13:04 +00:00
|
|
|
pubsub *pubsub.PubSub
|
2019-08-18 15:33:58 +00:00
|
|
|
|
|
|
|
BroadcastCalled bool
|
2019-08-16 17:13:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewTestP2P initializes a new p2p test service.
|
|
|
|
func NewTestP2P(t *testing.T) *TestP2P {
|
|
|
|
ctx := context.Background()
|
|
|
|
h := bhost.NewBlankHost(swarmt.GenSwarm(t, ctx))
|
|
|
|
ps, err := pubsub.NewFloodSub(ctx, h,
|
|
|
|
pubsub.WithMessageSigning(false),
|
|
|
|
pubsub.WithStrictSignatureVerification(false),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &TestP2P{
|
|
|
|
t: t,
|
2019-08-16 20:03:11 +00:00
|
|
|
Host: h,
|
2019-08-16 17:13:04 +00:00
|
|
|
pubsub: ps,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Connect two test peers together.
|
|
|
|
func (p *TestP2P) Connect(b *TestP2P) {
|
2019-08-16 20:03:11 +00:00
|
|
|
if err := connect(p.Host, b.Host); err != nil {
|
2019-08-16 17:13:04 +00:00
|
|
|
p.t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func connect(a, b host.Host) error {
|
|
|
|
pinfo := a.Peerstore().PeerInfo(a.ID())
|
|
|
|
return b.Connect(context.Background(), pinfo)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReceiveRPC simulates an incoming RPC.
|
|
|
|
func (p *TestP2P) ReceiveRPC(topic string, msg proto.Message) {
|
|
|
|
h := bhost.NewBlankHost(swarmt.GenSwarm(p.t, context.Background()))
|
2019-08-16 20:03:11 +00:00
|
|
|
if err := connect(h, p.Host); err != nil {
|
2019-08-16 17:13:04 +00:00
|
|
|
p.t.Fatalf("Failed to connect two peers for RPC: %v", err)
|
|
|
|
}
|
2019-08-16 20:03:11 +00:00
|
|
|
s, err := h.NewStream(context.Background(), p.Host.ID(), protocol.ID(topic+p.Encoding().ProtocolSuffix()))
|
2019-08-16 17:13:04 +00:00
|
|
|
if err != nil {
|
|
|
|
p.t.Fatalf("Failed to open stream %v", err)
|
|
|
|
}
|
|
|
|
defer s.Close()
|
|
|
|
|
|
|
|
n, err := p.Encoding().Encode(s, msg)
|
|
|
|
if err != nil {
|
|
|
|
p.t.Fatalf("Failed to encode message: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
p.t.Logf("Wrote %d bytes", n)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReceivePubSub simulates an incoming message over pubsub on a given topic.
|
|
|
|
func (p *TestP2P) ReceivePubSub(topic string, msg proto.Message) {
|
|
|
|
h := bhost.NewBlankHost(swarmt.GenSwarm(p.t, context.Background()))
|
|
|
|
ps, err := pubsub.NewFloodSub(context.Background(), h,
|
|
|
|
pubsub.WithMessageSigning(false),
|
|
|
|
pubsub.WithStrictSignatureVerification(false),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
p.t.Fatalf("Failed to create flood sub: %v", err)
|
|
|
|
}
|
2019-08-16 20:03:11 +00:00
|
|
|
if err := connect(h, p.Host); err != nil {
|
2019-08-16 17:13:04 +00:00
|
|
|
p.t.Fatalf("Failed to connect two peers for RPC: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// PubSub requires some delay after connecting for the (*PubSub).processLoop method to
|
|
|
|
// pick up the newly connected peer.
|
|
|
|
time.Sleep(time.Millisecond * 100)
|
|
|
|
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
if _, err := p.Encoding().Encode(buf, msg); err != nil {
|
|
|
|
p.t.Fatalf("Failed to encode message: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := ps.Publish(topic+p.Encoding().ProtocolSuffix(), buf.Bytes()); err != nil {
|
|
|
|
p.t.Fatalf("Failed to publish message; %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Broadcast a message.
|
2019-08-18 04:32:39 +00:00
|
|
|
func (p *TestP2P) Broadcast(msg proto.Message) error {
|
2019-08-18 15:33:58 +00:00
|
|
|
p.BroadcastCalled = true
|
|
|
|
return nil
|
2019-08-16 17:13:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// SetStreamHandler for RPC.
|
|
|
|
func (p *TestP2P) SetStreamHandler(topic string, handler network.StreamHandler) {
|
2019-08-16 20:03:11 +00:00
|
|
|
p.Host.SetStreamHandler(protocol.ID(topic), handler)
|
2019-08-16 17:13:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Encoding returns ssz encoding.
|
|
|
|
func (p *TestP2P) Encoding() encoder.NetworkEncoding {
|
|
|
|
return &encoder.SszNetworkEncoder{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// PubSub returns reference underlying floodsub. This test library uses floodsub
|
|
|
|
// to ensure all connected peers receive the message.
|
|
|
|
func (p *TestP2P) PubSub() *pubsub.PubSub {
|
|
|
|
return p.pubsub
|
|
|
|
}
|
2019-08-16 20:03:11 +00:00
|
|
|
|
|
|
|
// Disconnect from a peer.
|
|
|
|
func (p *TestP2P) Disconnect(pid peer.ID) error {
|
|
|
|
return p.Host.Network().ClosePeer(pid)
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddHandshake to the peer handshake records.
|
|
|
|
func (p *TestP2P) AddHandshake(pid peer.ID, hello *pb.Hello) {
|
|
|
|
// TODO(3147): add this.
|
|
|
|
}
|