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-18 04:32:39 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/p2p/encoder"
|
2019-08-19 21:20:56 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
|
|
deprecatedp2p "github.com/prysmaticlabs/prysm/shared/deprecated-p2p"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/event"
|
2019-08-18 04:32:39 +00:00
|
|
|
)
|
2019-08-16 17:13:04 +00:00
|
|
|
|
|
|
|
// TestP2P represents a p2p implementation that can be used for testing.
|
|
|
|
type TestP2P struct {
|
2019-08-21 06:08:30 +00:00
|
|
|
t *testing.T
|
|
|
|
Host host.Host
|
|
|
|
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 {
|
2019-08-21 06:08:30 +00:00
|
|
|
pinfo := b.Peerstore().PeerInfo(b.ID())
|
|
|
|
return a.Connect(context.Background(), pinfo)
|
2019-08-16 17:13:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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-19 21:20:56 +00:00
|
|
|
func (p *TestP2P) Broadcast(ctx context.Context, 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.
|
|
|
|
}
|
2019-08-19 21:20:56 +00:00
|
|
|
|
2019-08-24 18:41:24 +00:00
|
|
|
// PeerID returns the Peer ID of the local peer.
|
|
|
|
func (p *TestP2P) PeerID() peer.ID {
|
|
|
|
return p.Host.ID()
|
|
|
|
}
|
|
|
|
|
2019-08-19 21:20:56 +00:00
|
|
|
// Send a message to a specific peer.
|
2019-08-22 23:02:46 +00:00
|
|
|
func (p *TestP2P) Send(ctx context.Context, msg proto.Message, pid peer.ID) (network.Stream, error) {
|
|
|
|
return nil, nil
|
2019-08-19 21:20:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Subscribe to some topic. Not implemented.
|
|
|
|
func (p *TestP2P) Subscribe(msg proto.Message, ch chan deprecatedp2p.Message) event.Subscription {
|
|
|
|
// TODO(3147): remove this.
|
|
|
|
return nil
|
|
|
|
}
|
2019-08-21 20:58:38 +00:00
|
|
|
|
|
|
|
// Started always returns true.
|
|
|
|
func (p *TestP2P) Started() bool {
|
|
|
|
return true
|
|
|
|
}
|