go-libp2p-rendezvous/svc_test.go

264 lines
8.1 KiB
Go
Raw Permalink Normal View History

2018-04-24 17:05:05 +03:00
package rendezvous
import (
"context"
2018-04-24 18:16:44 +03:00
"fmt"
"math/rand"
2018-04-24 17:05:05 +03:00
"testing"
"time"
2018-04-24 17:05:05 +03:00
"github.com/libp2p/go-libp2p/core/host"
inet "github.com/libp2p/go-libp2p/core/network"
"github.com/libp2p/go-libp2p/core/peer"
mocknet "github.com/libp2p/go-libp2p/p2p/net/mock"
"github.com/libp2p/go-msgio/pbio"
"github.com/stretchr/testify/require"
db "github.com/waku-org/go-libp2p-rendezvous/db/sqlite"
pb "github.com/waku-org/go-libp2p-rendezvous/pb"
"github.com/waku-org/go-libp2p-rendezvous/test_utils"
2018-04-24 17:05:05 +03:00
)
func getRendezvousHosts(t *testing.T, ctx context.Context, m mocknet.Mocknet, n int) []host.Host {
return test_utils.GetRendezvousHosts(t, ctx, m, n)
2018-04-24 17:05:05 +03:00
}
func makeRendezvousService(ctx context.Context, host host.Host, path string) (*RendezvousService, error) {
dbi, err := db.OpenDB(ctx, path)
2018-04-24 17:05:05 +03:00
if err != nil {
return nil, err
2018-04-24 17:05:05 +03:00
}
return NewRendezvousService(host, dbi), nil
2018-04-24 17:05:05 +03:00
}
func getRendezvousPointsTest(t *testing.T, hosts []host.Host) []RendezvousPoint {
clients := make([]RendezvousPoint, len(hosts)-1)
2018-04-24 17:05:05 +03:00
for i, host := range hosts[1:] {
clients[i] = NewRendezvousPoint(host, hosts[0].ID())
2018-04-24 17:05:05 +03:00
}
return clients
}
func TestSVCRegistrationAndDiscovery(t *testing.T) {
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
m := mocknet.New()
defer m.Close()
hosts := getRendezvousHosts(t, ctx, m, 5)
2018-04-24 17:05:05 +03:00
svc, err := makeRendezvousService(ctx, hosts[0], ":memory:")
require.NoError(t, err)
2018-04-24 17:05:05 +03:00
defer svc.DB.Close()
clients := getRendezvousPointsTest(t, hosts)
2018-04-24 17:05:05 +03:00
const registerTTL = 60
recordTTL, err := clients[0].Register(ctx, "foo1", registerTTL)
require.NoError(t, err)
require.Equalf(t, registerTTL*time.Second, recordTTL, "expected record TTL to be %d seconds", DefaultTTL)
2018-04-24 17:05:05 +03:00
rrs, cookie, err := clients[0].Discover(ctx, "foo1", 10, nil)
require.NoError(t, err)
require.Len(t, rrs, 1)
2018-04-24 17:05:05 +03:00
checkHostRegistration(t, rrs[0], hosts[1])
for i, client := range clients[1:] {
recordTTL, err = client.Register(ctx, "foo1", registerTTL)
require.NoError(t, err)
require.Equalf(t, registerTTL*time.Second, recordTTL, "expected record TTL to be %d seconds", DefaultTTL)
2018-04-24 17:05:05 +03:00
rrs, cookie, err = clients[0].Discover(ctx, "foo1", 10, cookie)
require.NoError(t, err)
require.Len(t, rrs, 1)
2018-04-24 17:05:05 +03:00
checkHostRegistration(t, rrs[0], hosts[2+i])
}
for _, client := range clients[1:] {
rrs, _, err = client.Discover(ctx, "foo1", 10, nil)
require.NoError(t, err)
require.Len(t, rrs, 4)
2018-04-24 17:05:05 +03:00
for j, rr := range rrs {
checkHostRegistration(t, rr, hosts[1+j])
}
}
err = clients[0].Unregister(ctx, "foo1")
require.NoError(t, err)
2018-04-24 17:05:05 +03:00
for _, client := range clients[0:] {
rrs, _, err = client.Discover(ctx, "foo1", 10, nil)
require.NoError(t, err)
require.Lenf(t, rrs, 3, "Expected 3 registrations, got %d", len(rrs))
2018-04-24 17:05:05 +03:00
for j, rr := range rrs {
checkHostRegistration(t, rr, hosts[2+j])
}
}
err = clients[1].Unregister(ctx, "")
require.NoError(t, err)
2018-04-24 17:05:05 +03:00
for _, client := range clients[0:] {
rrs, _, err = client.Discover(ctx, "foo1", 10, nil)
require.NoError(t, err)
require.Len(t, rrs, 2)
2018-04-24 17:05:05 +03:00
for j, rr := range rrs {
checkHostRegistration(t, rr, hosts[3+j])
}
}
}
func checkHostRegistration(t *testing.T, rr Registration, host host.Host) {
if rr.Peer.ID != host.ID() {
t.Fatal("bad registration: peer ID doesn't match host ID")
}
addrs := host.Addrs()
raddrs := rr.Peer.Addrs
require.Equal(t, len(addrs), len(raddrs), "bad registration: peer address length mismatch")
2018-04-24 17:05:05 +03:00
for i, addr := range addrs {
raddr := raddrs[i]
require.True(t, addr.Equal(raddr), "bad registration: peer address mismatch")
2018-04-24 17:05:05 +03:00
}
}
2018-04-24 18:16:44 +03:00
const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
func RandStringBytes(n int) string {
b := make([]byte, n)
for i := range b {
b[i] = letterBytes[rand.Intn(len(letterBytes))]
}
return string(b)
}
2018-04-24 18:16:44 +03:00
func TestSVCErrors(t *testing.T) {
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
m := mocknet.New()
defer m.Close()
hosts := getRendezvousHosts(t, ctx, m, 2)
2018-04-24 18:16:44 +03:00
svc, err := makeRendezvousService(ctx, hosts[0], ":memory:")
require.NoError(t, err)
2018-04-24 18:16:44 +03:00
defer svc.DB.Close()
// testable registration errors
privKey := hosts[1].Peerstore().PrivKey(hosts[1].ID())
rm, err := newRegisterMessage(privKey, "", peer.AddrInfo{}, 0)
require.NoError(t, err)
res, err := doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_NAMESPACE, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
badns := RandStringBytes(2 * MaxNamespaceLength)
rm, err = newRegisterMessage(privKey, badns, peer.AddrInfo{}, 0)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_NAMESPACE, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
rm, err = newRegisterMessage(privKey, "foo", peer.AddrInfo{}, 0)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_SIGNED_PEER_RECORD, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
rm, err = newRegisterMessage(privKey, "foo", peer.AddrInfo{ID: peer.ID("blah")}, 0)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_SIGNED_PEER_RECORD, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
p, err := peer.Decode("QmVr26fY1tKyspEJBniVhqxQeEjhF78XerGiqWAwraVLQH")
require.NoError(t, err)
2018-04-24 18:16:44 +03:00
rm, err = newRegisterMessage(privKey, "foo", peer.AddrInfo{ID: p}, 0)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_SIGNED_PEER_RECORD, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
rm, err = newRegisterMessage(privKey, "foo", peer.AddrInfo{ID: hosts[1].ID()}, 0)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_SIGNED_PEER_RECORD, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
rm, err = newRegisterMessage(privKey, "foo", peer.AddrInfo{ID: hosts[1].ID(), Addrs: hosts[1].Addrs()}, 2*MaxTTL)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_TTL, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
// do MaxRegistrations
for i := 0; i < MaxRegistrations+1; i++ {
ns := fmt.Sprintf("foo%d", i)
rm, err = newRegisterMessage(privKey, ns, peer.AddrInfo{ID: hosts[1].ID(), Addrs: hosts[1].Addrs()}, 0)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
2018-04-24 18:16:44 +03:00
if err != nil {
t.Fatal(err)
}
if res.GetRegisterResponse().GetStatus() != pb.Message_OK {
t.Fatal("expected OK")
}
}
// and now fail
rm, err = newRegisterMessage(privKey, "foo", peer.AddrInfo{ID: hosts[1].ID(), Addrs: hosts[1].Addrs()}, 0)
require.NoError(t, err)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), rm)
require.NoError(t, err)
require.Equal(t, pb.Message_E_NOT_AUTHORIZED, res.GetRegisterResponse().GetStatus())
2018-04-24 18:16:44 +03:00
// testable discovery errors
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(), newDiscoverMessage(badns, 0, nil))
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_NAMESPACE, res.GetDiscoverResponse().GetStatus())
2018-04-24 18:16:44 +03:00
badcookie := make([]byte, 10)
rand.Read(badcookie)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(),
newDiscoverMessage("foo", 0, badcookie))
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_COOKIE, res.GetDiscoverResponse().GetStatus())
2018-04-24 18:16:44 +03:00
badcookie = make([]byte, 40)
rand.Read(badcookie)
res, err = doTestRequest(ctx, hosts[1], hosts[0].ID(),
newDiscoverMessage("foo", 0, badcookie))
require.NoError(t, err)
require.Equal(t, pb.Message_E_INVALID_COOKIE, res.GetDiscoverResponse().GetStatus())
2018-04-24 18:16:44 +03:00
}
func doTestRequest(ctx context.Context, host host.Host, rp peer.ID, m *pb.Message) (*pb.Message, error) {
s, err := host.NewStream(ctx, rp, RendezvousProto)
if err != nil {
return nil, err
}
defer s.Close()
r := pbio.NewDelimitedReader(s, inet.MessageSizeMax)
w := pbio.NewDelimitedWriter(s)
2018-04-24 18:16:44 +03:00
err = w.WriteMsg(m)
if err != nil {
return nil, err
}
res := new(pb.Message)
err = r.ReadMsg(res)
if err != nil {
return nil, err
}
return res, nil
}