2018-06-06 02:38:16 +00:00
|
|
|
package p2p
|
|
|
|
|
2018-07-17 18:39:04 +00:00
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io/ioutil"
|
|
|
|
"reflect"
|
2018-07-25 16:11:15 +00:00
|
|
|
"sync"
|
2018-07-17 18:39:04 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/event"
|
|
|
|
"github.com/golang/protobuf/proto"
|
2018-07-24 15:21:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared"
|
2018-07-17 18:39:04 +00:00
|
|
|
|
|
|
|
floodsub "github.com/libp2p/go-floodsub"
|
|
|
|
swarmt "github.com/libp2p/go-libp2p-swarm/testing"
|
|
|
|
bhost "github.com/libp2p/go-libp2p/p2p/host/basic"
|
2018-07-20 21:31:26 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/sharding/v1"
|
2018-07-21 17:51:18 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2018-07-17 18:39:04 +00:00
|
|
|
logTest "github.com/sirupsen/logrus/hooks/test"
|
|
|
|
)
|
2018-06-06 02:38:16 +00:00
|
|
|
|
2018-06-13 12:44:33 +00:00
|
|
|
// Ensure that server implements service.
|
2018-07-24 15:21:58 +00:00
|
|
|
var _ = shared.Service(&Server{})
|
2018-07-17 18:39:04 +00:00
|
|
|
|
|
|
|
func init() {
|
2018-07-21 17:51:18 +00:00
|
|
|
logrus.SetLevel(logrus.DebugLevel)
|
|
|
|
logrus.SetOutput(ioutil.Discard)
|
2018-07-17 18:39:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestLifecycle(t *testing.T) {
|
|
|
|
hook := logTest.NewGlobal()
|
|
|
|
|
|
|
|
s, err := NewServer()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not start a new server: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Start()
|
2018-07-21 17:20:00 +00:00
|
|
|
msg := hook.Entries[0].Message
|
2018-07-17 18:39:04 +00:00
|
|
|
want := "Starting shardp2p server"
|
2018-07-21 17:20:00 +00:00
|
|
|
if msg != want {
|
2018-07-17 18:39:04 +00:00
|
|
|
t.Errorf("incorrect log. wanted: %s. got: %v", want, msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Stop()
|
2018-07-21 17:20:00 +00:00
|
|
|
msg = hook.LastEntry().Message
|
2018-07-17 18:39:04 +00:00
|
|
|
want = "Stopping shardp2p server"
|
2018-07-21 17:20:00 +00:00
|
|
|
if msg != want {
|
2018-07-17 18:39:04 +00:00
|
|
|
t.Errorf("incorrect log. wanted: %s. got: %v", want, msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
// The context should have been cancelled.
|
|
|
|
if s.ctx.Err() == nil {
|
|
|
|
t.Error("Context was not cancelled")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBroadcast(t *testing.T) {
|
|
|
|
s, err := NewServer()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not start a new server: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := &pb.CollationBodyRequest{}
|
|
|
|
s.Broadcast(msg)
|
|
|
|
|
|
|
|
// TODO: test that topic was published
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSubscribeToTopic(t *testing.T) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.TODO(), 1*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
h := bhost.New(swarmt.GenSwarm(t, ctx))
|
|
|
|
|
|
|
|
gsub, err := floodsub.NewFloodSub(ctx, h)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Failed to create floodsub: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
s := Server{
|
|
|
|
ctx: ctx,
|
|
|
|
gsub: gsub,
|
|
|
|
host: h,
|
|
|
|
feeds: make(map[reflect.Type]*event.Feed),
|
2018-07-25 16:11:15 +00:00
|
|
|
mutex: &sync.Mutex{},
|
2018-07-17 18:39:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
feed := s.Feed(pb.CollationBodyRequest{})
|
|
|
|
ch := make(chan Message)
|
|
|
|
sub := feed.Subscribe(ch)
|
|
|
|
defer sub.Unsubscribe()
|
|
|
|
|
2018-07-26 14:16:31 +00:00
|
|
|
testSubscribe(t, s, gsub, ch, ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSubscribe(t *testing.T) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.TODO(), 1*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
h := bhost.New(swarmt.GenSwarm(t, ctx))
|
|
|
|
|
|
|
|
gsub, err := floodsub.NewFloodSub(ctx, h)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Failed to create floodsub: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
s := Server{
|
|
|
|
ctx: ctx,
|
|
|
|
gsub: gsub,
|
|
|
|
host: h,
|
|
|
|
feeds: make(map[reflect.Type]*event.Feed),
|
|
|
|
mutex: &sync.Mutex{},
|
|
|
|
}
|
|
|
|
|
|
|
|
ch := make(chan Message)
|
|
|
|
sub := s.Subscribe(pb.CollationBodyRequest{}, ch)
|
|
|
|
defer sub.Unsubscribe()
|
|
|
|
|
|
|
|
testSubscribe(t, s, gsub, ch, ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func testSubscribe(t *testing.T, s Server, gsub *floodsub.PubSub, ch chan Message, ctx context.Context) {
|
2018-07-17 18:39:04 +00:00
|
|
|
topic := pb.Topic_COLLATION_BODY_REQUEST
|
|
|
|
msgType := topicTypeMapping[topic]
|
|
|
|
go s.subscribeToTopic(topic, msgType)
|
|
|
|
|
|
|
|
// Short delay to let goroutine add subscription.
|
|
|
|
time.Sleep(time.Millisecond * 10)
|
|
|
|
|
|
|
|
// The topic should be subscribed with gsub.
|
|
|
|
topics := gsub.GetTopics()
|
|
|
|
if len(topics) < 1 || topics[0] != topic.String() {
|
|
|
|
t.Errorf("Unexpected subscribed topics: %v. Wanted %s", topics, topic)
|
|
|
|
}
|
|
|
|
|
|
|
|
pbMsg := &pb.CollationBodyRequest{ShardId: 5}
|
|
|
|
|
|
|
|
done := make(chan bool)
|
|
|
|
go func() {
|
|
|
|
// The message should be received from the feed.
|
|
|
|
msg := <-ch
|
|
|
|
if !proto.Equal(msg.Data.(proto.Message), pbMsg) {
|
|
|
|
t.Errorf("Unexpected msg: %+v. Wanted %+v.", msg.Data, pbMsg)
|
|
|
|
}
|
|
|
|
|
|
|
|
done <- true
|
|
|
|
}()
|
|
|
|
|
|
|
|
b, err := proto.Marshal(pbMsg)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Failed to marshal pbMsg: %v", err)
|
|
|
|
}
|
|
|
|
if err = gsub.Publish(topic.String(), b); err != nil {
|
|
|
|
t.Errorf("Failed to publish message: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for our message assertion to complete.
|
|
|
|
select {
|
|
|
|
case <-done:
|
|
|
|
case <-ctx.Done():
|
|
|
|
t.Error("Context timed out before a message was received!")
|
|
|
|
}
|
|
|
|
}
|