2023-08-30 21:33:57 +07:00
|
|
|
package peermanager
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/rand"
|
|
|
|
"fmt"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/libp2p/go-libp2p/core/host"
|
|
|
|
libp2pProtocol "github.com/libp2p/go-libp2p/core/protocol"
|
|
|
|
"github.com/multiformats/go-multiaddr"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/waku-org/go-waku/tests"
|
|
|
|
wps "github.com/waku-org/go-waku/waku/v2/peerstore"
|
2023-09-29 10:43:25 +05:30
|
|
|
wakuproto "github.com/waku-org/go-waku/waku/v2/protocol"
|
2023-09-14 14:06:08 +05:30
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/relay"
|
2023-08-30 21:33:57 +07:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/utils"
|
|
|
|
)
|
|
|
|
|
|
|
|
func getAddr(h host.Host) multiaddr.Multiaddr {
|
|
|
|
id, _ := multiaddr.NewMultiaddr(fmt.Sprintf("/p2p/%s", h.ID().Pretty()))
|
|
|
|
return h.Network().ListenAddresses()[0].Encapsulate(id)
|
|
|
|
}
|
|
|
|
|
|
|
|
func initTest(t *testing.T) (context.Context, *PeerManager, func()) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
// hosts
|
|
|
|
h1, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h1.Close()
|
2023-09-07 15:01:31 +05:30
|
|
|
|
2023-08-30 21:33:57 +07:00
|
|
|
// host 1 is used by peer manager
|
2023-09-27 12:16:37 +05:30
|
|
|
pm := NewPeerManager(10, 20, utils.Logger())
|
2023-08-30 21:33:57 +07:00
|
|
|
pm.SetHost(h1)
|
2023-09-07 15:01:31 +05:30
|
|
|
|
2023-08-30 21:33:57 +07:00
|
|
|
return ctx, pm, func() {
|
|
|
|
cancel()
|
|
|
|
h1.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestServiceSlots(t *testing.T) {
|
|
|
|
ctx, pm, deferFn := initTest(t)
|
|
|
|
defer deferFn()
|
|
|
|
|
|
|
|
h2, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h2.Close()
|
|
|
|
|
|
|
|
h3, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h3.Close()
|
|
|
|
// protocols
|
|
|
|
protocol := libp2pProtocol.ID("test/protocol")
|
|
|
|
protocol1 := libp2pProtocol.ID("test/protocol1")
|
|
|
|
|
|
|
|
// add h2 peer to peer manager
|
|
|
|
t.Log(h2.ID())
|
2023-09-14 20:30:06 +05:30
|
|
|
_, err = pm.AddPeer(getAddr(h2), wps.Static, []string{""}, libp2pProtocol.ID(protocol))
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
///////////////
|
|
|
|
// getting peer for protocol
|
|
|
|
///////////////
|
|
|
|
|
|
|
|
// select peer from pm, currently only h2 is set in pm
|
2023-09-29 10:43:25 +05:30
|
|
|
peerID, err := pm.SelectPeer(protocol, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, peerID, h2.ID())
|
|
|
|
|
|
|
|
// add h3 peer to peer manager
|
2023-09-14 20:30:06 +05:30
|
|
|
_, err = pm.AddPeer(getAddr(h3), wps.Static, []string{""}, libp2pProtocol.ID(protocol))
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// check that returned peer is h2 or h3 peer
|
2023-09-29 10:43:25 +05:30
|
|
|
peerID, err = pm.SelectPeer(protocol, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
if peerID == h2.ID() || peerID == h3.ID() {
|
|
|
|
//Test success
|
|
|
|
t.Log("Random peer selection per protocol successful")
|
|
|
|
} else {
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////
|
|
|
|
// getting peer for protocol1
|
|
|
|
///////////////
|
|
|
|
h4, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h4.Close()
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
_, err = pm.SelectPeer(protocol1, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.Error(t, err, utils.ErrNoPeersAvailable)
|
|
|
|
|
|
|
|
// add h4 peer for protocol1
|
2023-09-14 20:30:06 +05:30
|
|
|
_, err = pm.AddPeer(getAddr(h4), wps.Static, []string{""}, libp2pProtocol.ID(protocol1))
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
//Test peer selection for protocol1
|
2023-09-29 10:43:25 +05:30
|
|
|
peerID, err = pm.SelectPeer(protocol1, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, peerID, h4.ID())
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
_, err = pm.SelectPeerByContentTopic(protocol1, "")
|
|
|
|
require.Error(t, wakuproto.ErrInvalidFormat, err)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPeerSelection(t *testing.T) {
|
|
|
|
ctx, pm, deferFn := initTest(t)
|
|
|
|
defer deferFn()
|
|
|
|
|
|
|
|
h2, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h2.Close()
|
|
|
|
|
|
|
|
h3, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h3.Close()
|
|
|
|
|
|
|
|
protocol := libp2pProtocol.ID("test/protocol")
|
|
|
|
_, err = pm.AddPeer(getAddr(h2), wps.Static, []string{"/waku/rs/2/1", "/waku/rs/2/2"}, libp2pProtocol.ID(protocol))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = pm.AddPeer(getAddr(h3), wps.Static, []string{"/waku/rs/2/1"}, libp2pProtocol.ID(protocol))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = pm.SelectPeer(protocol, "")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
peerID, err := pm.SelectPeer(protocol, "/waku/rs/2/2")
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, h2.ID(), peerID)
|
|
|
|
|
|
|
|
_, err = pm.SelectPeer(protocol, "/waku/rs/2/3")
|
|
|
|
require.Error(t, utils.ErrNoPeersAvailable, err)
|
|
|
|
|
|
|
|
_, err = pm.SelectPeer(protocol, "/waku/rs/2/1")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2023-08-30 21:33:57 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestDefaultProtocol(t *testing.T) {
|
|
|
|
ctx, pm, deferFn := initTest(t)
|
|
|
|
defer deferFn()
|
|
|
|
///////////////
|
|
|
|
// check peer for default protocol
|
|
|
|
///////////////
|
|
|
|
//Test empty peer selection for relay protocol
|
2023-09-29 10:43:25 +05:30
|
|
|
_, err := pm.SelectPeer(relay.WakuRelayID_v200, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.Error(t, err, utils.ErrNoPeersAvailable)
|
|
|
|
|
|
|
|
///////////////
|
|
|
|
// getting peer for default protocol
|
|
|
|
///////////////
|
|
|
|
h5, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h5.Close()
|
|
|
|
|
|
|
|
//Test peer selection for relay protocol from peer store
|
2023-09-14 20:30:06 +05:30
|
|
|
_, err = pm.AddPeer(getAddr(h5), wps.Static, []string{""}, relay.WakuRelayID_v200)
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// since we are not passing peerList, selectPeer fn using filterByProto checks in PeerStore for peers with same protocol.
|
2023-09-29 10:43:25 +05:30
|
|
|
peerID, err := pm.SelectPeer(relay.WakuRelayID_v200, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, peerID, h5.ID())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAdditionAndRemovalOfPeer(t *testing.T) {
|
|
|
|
ctx, pm, deferFn := initTest(t)
|
|
|
|
defer deferFn()
|
|
|
|
///////////////
|
|
|
|
// set h6 peer for protocol2 and remove that peer and check again
|
|
|
|
///////////////
|
|
|
|
//Test random peer selection
|
|
|
|
protocol2 := libp2pProtocol.ID("test/protocol2")
|
|
|
|
h6, err := tests.MakeHost(ctx, 0, rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer h6.Close()
|
|
|
|
|
2023-09-14 20:30:06 +05:30
|
|
|
_, err = pm.AddPeer(getAddr(h6), wps.Static, []string{""}, protocol2)
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
peerID, err := pm.SelectPeer(protocol2, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, peerID, h6.ID())
|
|
|
|
|
|
|
|
pm.RemovePeer(peerID)
|
2023-09-29 10:43:25 +05:30
|
|
|
_, err = pm.SelectPeer(protocol2, "")
|
2023-08-30 21:33:57 +07:00
|
|
|
require.Error(t, err, utils.ErrNoPeersAvailable)
|
|
|
|
}
|
2023-09-07 06:36:43 +00:00
|
|
|
|
|
|
|
func TestConnectToRelayPeers(t *testing.T) {
|
|
|
|
|
2023-09-07 15:01:31 +05:30
|
|
|
ctx, pm, deferFn := initTest(t)
|
|
|
|
pc, err := NewPeerConnectionStrategy(pm, 120*time.Second, pm.logger)
|
|
|
|
require.NoError(t, err)
|
|
|
|
pm.SetPeerConnector(pc)
|
|
|
|
err = pc.Start(ctx)
|
|
|
|
require.NoError(t, err)
|
|
|
|
pm.Start(ctx)
|
|
|
|
|
2023-09-07 06:36:43 +00:00
|
|
|
defer deferFn()
|
|
|
|
|
|
|
|
pm.connectToRelayPeers()
|
|
|
|
|
|
|
|
}
|