2021-11-12 09:19:42 +00:00
|
|
|
package node
|
|
|
|
|
|
|
|
import (
|
2022-03-18 19:50:10 +00:00
|
|
|
"bytes"
|
2021-11-12 09:19:42 +00:00
|
|
|
"context"
|
2022-03-18 19:50:10 +00:00
|
|
|
"math/big"
|
2021-11-12 09:19:42 +00:00
|
|
|
"net"
|
2022-02-18 17:49:11 +00:00
|
|
|
"sync"
|
2021-11-12 09:19:42 +00:00
|
|
|
"testing"
|
2022-02-18 17:49:11 +00:00
|
|
|
"time"
|
2021-11-12 09:19:42 +00:00
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
|
|
|
"github.com/stretchr/testify/require"
|
2022-11-09 19:53:01 +00:00
|
|
|
"github.com/waku-org/go-waku/tests"
|
|
|
|
"github.com/waku-org/go-waku/waku/persistence"
|
|
|
|
"github.com/waku-org/go-waku/waku/persistence/sqlite"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/filter"
|
2022-12-16 00:47:14 +00:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/pb"
|
2022-11-09 19:53:01 +00:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/relay"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/store"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/utils"
|
2021-11-12 09:19:42 +00:00
|
|
|
)
|
|
|
|
|
2022-12-16 00:47:14 +00:00
|
|
|
func createTestMsg(version uint32) *pb.WakuMessage {
|
|
|
|
message := new(pb.WakuMessage)
|
|
|
|
message.Payload = []byte{0, 1, 2}
|
|
|
|
message.Version = version
|
|
|
|
message.Timestamp = 123456
|
|
|
|
return message
|
|
|
|
}
|
|
|
|
|
2021-11-12 09:19:42 +00:00
|
|
|
func TestWakuNode2(t *testing.T) {
|
|
|
|
hostAddr, _ := net.ResolveTCPAddr("tcp", "0.0.0.0:0")
|
|
|
|
|
|
|
|
key, err := tests.RandomHex(32)
|
|
|
|
require.NoError(t, err)
|
|
|
|
prvKey, err := crypto.HexToECDSA(key)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
wakuNode, err := New(ctx,
|
|
|
|
WithPrivateKey(prvKey),
|
2021-11-17 16:19:42 +00:00
|
|
|
WithHostAddress(hostAddr),
|
2021-11-12 09:19:42 +00:00
|
|
|
WithWakuRelay(),
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = wakuNode.Start()
|
|
|
|
defer wakuNode.Stop()
|
|
|
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
2022-02-18 17:49:11 +00:00
|
|
|
|
2022-03-18 19:50:10 +00:00
|
|
|
func int2Bytes(i int) []byte {
|
|
|
|
if i > 0 {
|
|
|
|
return append(big.NewInt(int64(i)).Bytes(), byte(1))
|
|
|
|
}
|
|
|
|
return append(big.NewInt(int64(i)).Bytes(), byte(0))
|
|
|
|
}
|
|
|
|
|
|
|
|
func Test5000(t *testing.T) {
|
|
|
|
maxMsgs := 5000
|
|
|
|
maxMsgBytes := int2Bytes(maxMsgs)
|
|
|
|
|
2022-05-27 19:58:38 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
2022-02-18 17:49:11 +00:00
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
hostAddr1, _ := net.ResolveTCPAddr("tcp", "0.0.0.0:0")
|
|
|
|
key1, _ := tests.RandomHex(32)
|
|
|
|
prvKey1, _ := crypto.HexToECDSA(key1)
|
|
|
|
|
|
|
|
hostAddr2, _ := net.ResolveTCPAddr("tcp", "0.0.0.0:0")
|
|
|
|
key2, _ := tests.RandomHex(32)
|
|
|
|
prvKey2, _ := crypto.HexToECDSA(key2)
|
|
|
|
|
|
|
|
wakuNode1, err := New(ctx,
|
|
|
|
WithPrivateKey(prvKey1),
|
|
|
|
WithHostAddress(hostAddr1),
|
|
|
|
WithWakuRelay(),
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
err = wakuNode1.Start()
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer wakuNode1.Stop()
|
|
|
|
|
|
|
|
wakuNode2, err := New(ctx,
|
|
|
|
WithPrivateKey(prvKey2),
|
|
|
|
WithHostAddress(hostAddr2),
|
|
|
|
WithWakuRelay(),
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
err = wakuNode2.Start()
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer wakuNode2.Stop()
|
|
|
|
|
|
|
|
err = wakuNode2.DialPeer(ctx, wakuNode1.ListenAddresses()[0].String())
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
time.Sleep(2 * time.Second)
|
|
|
|
|
|
|
|
sub1, err := wakuNode1.Relay().Subscribe(ctx)
|
|
|
|
require.NoError(t, err)
|
|
|
|
sub2, err := wakuNode1.Relay().Subscribe(ctx)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
wg.Add(3)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2022-11-25 20:54:11 +00:00
|
|
|
ticker := time.NewTimer(30 * time.Second)
|
2022-02-18 17:49:11 +00:00
|
|
|
defer ticker.Stop()
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ticker.C:
|
2022-03-18 19:50:10 +00:00
|
|
|
require.Fail(t, "Timeout Sub1")
|
|
|
|
case msg := <-sub1.C:
|
2022-12-21 18:56:58 +00:00
|
|
|
if msg == nil {
|
|
|
|
return
|
|
|
|
}
|
2022-03-18 19:50:10 +00:00
|
|
|
if bytes.Equal(msg.Message().Payload, maxMsgBytes) {
|
2022-02-18 17:49:11 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2022-11-25 20:54:11 +00:00
|
|
|
ticker := time.NewTimer(30 * time.Second)
|
2022-02-18 17:49:11 +00:00
|
|
|
defer ticker.Stop()
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ticker.C:
|
2022-03-18 19:50:10 +00:00
|
|
|
require.Fail(t, "Timeout Sub2")
|
|
|
|
case msg := <-sub2.C:
|
2022-12-21 18:56:58 +00:00
|
|
|
if msg == nil {
|
|
|
|
return
|
|
|
|
}
|
2022-03-18 19:50:10 +00:00
|
|
|
if bytes.Equal(msg.Message().Payload, maxMsgBytes) {
|
2022-02-18 17:49:11 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-03-18 19:50:10 +00:00
|
|
|
for i := 1; i <= maxMsgs; i++ {
|
2022-02-18 17:49:11 +00:00
|
|
|
msg := createTestMsg(0)
|
2022-03-18 19:50:10 +00:00
|
|
|
msg.Payload = int2Bytes(i)
|
2022-02-23 15:01:53 +00:00
|
|
|
msg.Timestamp = int64(i)
|
2022-02-18 17:49:11 +00:00
|
|
|
if err := wakuNode2.Publish(ctx, msg); err != nil {
|
|
|
|
require.Fail(t, "Could not publish all messages")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
}
|
2022-06-14 19:53:48 +00:00
|
|
|
|
|
|
|
func TestDecoupledStoreFromRelay(t *testing.T) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
// NODE1: Relay Node + Filter Server
|
|
|
|
hostAddr1, err := net.ResolveTCPAddr("tcp", "0.0.0.0:0")
|
|
|
|
require.NoError(t, err)
|
|
|
|
wakuNode1, err := New(ctx,
|
|
|
|
WithHostAddress(hostAddr1),
|
|
|
|
WithWakuRelay(),
|
|
|
|
WithWakuFilter(true),
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
err = wakuNode1.Start()
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer wakuNode1.Stop()
|
|
|
|
|
|
|
|
// NODE2: Filter Client/Store
|
2023-01-04 17:58:14 +00:00
|
|
|
db, migration, err := sqlite.NewDB(":memory:")
|
2022-06-14 19:53:48 +00:00
|
|
|
require.NoError(t, err)
|
2023-01-04 17:58:14 +00:00
|
|
|
dbStore, err := persistence.NewDBStore(utils.Logger(), persistence.WithDB(db), persistence.WithMigrations(migration))
|
2022-06-14 19:53:48 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
hostAddr2, err := net.ResolveTCPAddr("tcp", "0.0.0.0:0")
|
|
|
|
require.NoError(t, err)
|
|
|
|
wakuNode2, err := New(ctx,
|
|
|
|
WithHostAddress(hostAddr2),
|
|
|
|
WithWakuFilter(false),
|
2023-01-03 15:17:25 +00:00
|
|
|
WithWakuStore(),
|
2022-06-14 19:53:48 +00:00
|
|
|
WithMessageProvider(dbStore),
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
err = wakuNode2.Start()
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer wakuNode2.Stop()
|
|
|
|
|
|
|
|
err = wakuNode2.DialPeerWithMultiAddress(ctx, wakuNode1.ListenAddresses()[0])
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
time.Sleep(2 * time.Second)
|
|
|
|
|
|
|
|
_, filter, err := wakuNode2.Filter().Subscribe(ctx, filter.ContentFilter{
|
|
|
|
Topic: string(relay.DefaultWakuTopic),
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Sleep to make sure the filter is subscribed
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
|
|
|
|
// Send MSG1 on NODE1
|
|
|
|
msg := createTestMsg(0)
|
|
|
|
msg.Payload = []byte{1, 2, 3, 4, 5}
|
|
|
|
msg.Timestamp = utils.GetUnixEpoch()
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
// MSG1 should be pushed in NODE2 via filter
|
|
|
|
defer wg.Done()
|
|
|
|
env := <-filter.Chan
|
|
|
|
require.Equal(t, msg.Timestamp, env.Message().Timestamp)
|
|
|
|
}()
|
|
|
|
|
|
|
|
time.Sleep(500 * time.Millisecond)
|
|
|
|
|
|
|
|
if err := wakuNode1.Publish(ctx, msg); err != nil {
|
|
|
|
require.Fail(t, "Could not publish all messages")
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
// NODE3: Query from NODE2
|
|
|
|
hostAddr3, err := net.ResolveTCPAddr("tcp", "0.0.0.0:0")
|
|
|
|
require.NoError(t, err)
|
|
|
|
wakuNode3, err := New(ctx,
|
|
|
|
WithHostAddress(hostAddr3),
|
|
|
|
WithWakuFilter(false),
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
err = wakuNode3.Start()
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer wakuNode3.Stop()
|
|
|
|
|
|
|
|
err = wakuNode3.DialPeerWithMultiAddress(ctx, wakuNode2.ListenAddresses()[0])
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// NODE2 should have returned the message received via filter
|
|
|
|
result, err := wakuNode3.Store().Query(ctx, store.Query{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Len(t, result.Messages, 1)
|
|
|
|
require.Equal(t, msg.Timestamp, result.Messages[0].Timestamp)
|
|
|
|
}
|