2021-05-23 19:42:26 +00:00
|
|
|
package core
|
2019-09-25 13:58:19 +00:00
|
|
|
|
|
|
|
import (
|
2019-09-25 18:07:36 +00:00
|
|
|
"bytes"
|
2022-08-06 14:05:12 +00:00
|
|
|
"crypto/ed25519"
|
2019-09-25 18:07:36 +00:00
|
|
|
"math/rand"
|
2021-05-29 16:13:59 +00:00
|
|
|
"net/url"
|
2019-09-25 13:58:19 +00:00
|
|
|
"os"
|
|
|
|
"testing"
|
2019-09-25 18:07:36 +00:00
|
|
|
"time"
|
2019-09-25 13:58:19 +00:00
|
|
|
|
|
|
|
"github.com/gologme/log"
|
|
|
|
)
|
|
|
|
|
2019-11-29 09:45:02 +00:00
|
|
|
// GetLoggerWithPrefix creates a new logger instance with prefix.
|
2019-09-28 12:41:53 +00:00
|
|
|
// If verbose is set to true, three log levels are enabled: "info", "warn", "error".
|
2019-09-28 12:24:54 +00:00
|
|
|
func GetLoggerWithPrefix(prefix string, verbose bool) *log.Logger {
|
2019-09-25 13:58:19 +00:00
|
|
|
l := log.New(os.Stderr, prefix, log.Flags())
|
2019-09-28 12:24:54 +00:00
|
|
|
if !verbose {
|
|
|
|
return l
|
|
|
|
}
|
2019-09-25 13:58:19 +00:00
|
|
|
l.EnableLevel("info")
|
|
|
|
l.EnableLevel("warn")
|
|
|
|
l.EnableLevel("error")
|
|
|
|
return l
|
|
|
|
}
|
|
|
|
|
2019-09-28 12:41:53 +00:00
|
|
|
// CreateAndConnectTwo creates two nodes. nodeB connects to nodeA.
|
|
|
|
// Verbosity flag is passed to logger.
|
|
|
|
func CreateAndConnectTwo(t testing.TB, verbose bool) (nodeA *Core, nodeB *Core) {
|
2022-08-06 14:05:12 +00:00
|
|
|
var err error
|
|
|
|
var skA, skB ed25519.PrivateKey
|
|
|
|
if _, skA, err = ed25519.GenerateKey(nil); err != nil {
|
2019-09-25 13:58:19 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-08-06 14:05:12 +00:00
|
|
|
if _, skB, err = ed25519.GenerateKey(nil); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-09-03 10:42:05 +00:00
|
|
|
logger := GetLoggerWithPrefix("", false)
|
2022-09-17 19:07:00 +00:00
|
|
|
if nodeA, err = New(skA, logger, ListenAddress("tcp://127.0.0.1:0")); err != nil {
|
2022-08-06 14:05:12 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-09-17 19:07:00 +00:00
|
|
|
if nodeB, err = New(skB, logger, ListenAddress("tcp://127.0.0.1:0")); err != nil {
|
2019-09-25 13:58:19 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-05-29 16:13:59 +00:00
|
|
|
u, err := url.Parse("tcp://" + nodeA.links.tcp.getAddr().String())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
err = nodeB.CallPeer(u, "")
|
2019-09-25 13:58:19 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-09-25 18:07:36 +00:00
|
|
|
|
2021-05-29 16:13:59 +00:00
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
|
2019-09-25 18:07:36 +00:00
|
|
|
if l := len(nodeA.GetPeers()); l != 1 {
|
|
|
|
t.Fatal("unexpected number of peers", l)
|
|
|
|
}
|
|
|
|
if l := len(nodeB.GetPeers()); l != 1 {
|
|
|
|
t.Fatal("unexpected number of peers", l)
|
|
|
|
}
|
|
|
|
|
2019-09-28 12:41:53 +00:00
|
|
|
return nodeA, nodeB
|
2019-09-25 18:07:36 +00:00
|
|
|
}
|
|
|
|
|
2019-09-28 12:10:17 +00:00
|
|
|
// WaitConnected blocks until either nodes negotiated DHT or 5 seconds passed.
|
|
|
|
func WaitConnected(nodeA, nodeB *Core) bool {
|
|
|
|
// It may take up to 3 seconds, but let's wait 5.
|
|
|
|
for i := 0; i < 50; i++ {
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
2023-03-26 21:34:49 +00:00
|
|
|
/*
|
|
|
|
if len(nodeA.GetPeers()) > 0 && len(nodeB.GetPeers()) > 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
*/
|
2023-05-13 21:15:04 +00:00
|
|
|
if len(nodeA.GetTree()) > 1 && len(nodeB.GetTree()) > 1 {
|
|
|
|
time.Sleep(3*time.Second) // FIXME hack, there's still stuff happening internally
|
2019-09-28 12:10:17 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2019-09-28 12:41:53 +00:00
|
|
|
// CreateEchoListener creates a routine listening on nodeA. It expects repeats messages of length bufLen.
|
|
|
|
// It returns a channel used to synchronize the routine with caller.
|
2019-09-28 12:20:57 +00:00
|
|
|
func CreateEchoListener(t testing.TB, nodeA *Core, bufLen int, repeats int) chan struct{} {
|
2019-09-28 12:41:53 +00:00
|
|
|
// Start routine
|
2019-09-25 18:07:36 +00:00
|
|
|
done := make(chan struct{})
|
|
|
|
go func() {
|
2019-09-28 12:10:17 +00:00
|
|
|
buf := make([]byte, bufLen)
|
2021-06-13 18:40:20 +00:00
|
|
|
res := make([]byte, bufLen)
|
2019-09-28 12:20:57 +00:00
|
|
|
for i := 0; i < repeats; i++ {
|
2021-07-07 23:36:51 +00:00
|
|
|
n, from, err := nodeA.ReadFrom(buf)
|
2019-09-28 12:20:57 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if n != bufLen {
|
|
|
|
t.Error("missing data")
|
|
|
|
return
|
|
|
|
}
|
2021-06-13 18:40:20 +00:00
|
|
|
copy(res, buf)
|
|
|
|
copy(res[8:24], buf[24:40])
|
|
|
|
copy(res[24:40], buf[8:24])
|
2021-07-07 23:36:51 +00:00
|
|
|
_, err = nodeA.WriteTo(res, from)
|
2019-09-28 12:20:57 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
2019-09-25 18:07:36 +00:00
|
|
|
}
|
|
|
|
done <- struct{}{}
|
|
|
|
}()
|
|
|
|
|
2019-09-28 12:10:17 +00:00
|
|
|
return done
|
|
|
|
}
|
|
|
|
|
2019-09-28 12:41:53 +00:00
|
|
|
// TestCore_Start_Connect checks if two nodes can connect together.
|
|
|
|
func TestCore_Start_Connect(t *testing.T) {
|
|
|
|
CreateAndConnectTwo(t, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestCore_Start_Transfer checks that messages can be passed between nodes (in both directions).
|
2019-09-28 12:10:17 +00:00
|
|
|
func TestCore_Start_Transfer(t *testing.T) {
|
2019-09-28 12:24:54 +00:00
|
|
|
nodeA, nodeB := CreateAndConnectTwo(t, true)
|
2021-05-29 16:13:59 +00:00
|
|
|
defer nodeA.Stop()
|
|
|
|
defer nodeB.Stop()
|
2019-09-28 12:10:17 +00:00
|
|
|
|
|
|
|
msgLen := 1500
|
2019-09-28 12:20:57 +00:00
|
|
|
done := CreateEchoListener(t, nodeA, msgLen, 1)
|
2019-09-28 12:10:17 +00:00
|
|
|
|
|
|
|
if !WaitConnected(nodeA, nodeB) {
|
|
|
|
t.Fatal("nodes did not connect")
|
|
|
|
}
|
|
|
|
|
2021-05-29 16:13:59 +00:00
|
|
|
// Send
|
2019-09-28 12:10:17 +00:00
|
|
|
msg := make([]byte, msgLen)
|
2021-06-13 18:40:20 +00:00
|
|
|
rand.Read(msg[40:])
|
|
|
|
msg[0] = 0x60
|
|
|
|
copy(msg[8:24], nodeB.Address())
|
|
|
|
copy(msg[24:40], nodeA.Address())
|
2021-07-07 23:36:51 +00:00
|
|
|
_, err := nodeB.WriteTo(msg, nodeA.LocalAddr())
|
2019-09-25 18:07:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-09-28 12:10:17 +00:00
|
|
|
buf := make([]byte, msgLen)
|
2021-07-07 23:36:51 +00:00
|
|
|
_, _, err = nodeB.ReadFrom(buf)
|
2019-09-25 18:07:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2021-06-13 18:40:20 +00:00
|
|
|
if !bytes.Equal(msg[40:], buf[40:]) {
|
2019-09-25 18:07:36 +00:00
|
|
|
t.Fatal("expected echo")
|
|
|
|
}
|
|
|
|
<-done
|
2019-09-25 13:58:19 +00:00
|
|
|
}
|
2019-09-28 12:20:57 +00:00
|
|
|
|
2019-09-28 12:41:53 +00:00
|
|
|
// BenchmarkCore_Start_Transfer estimates the possible transfer between nodes (in MB/s).
|
2019-09-28 12:20:57 +00:00
|
|
|
func BenchmarkCore_Start_Transfer(b *testing.B) {
|
2019-09-28 12:24:54 +00:00
|
|
|
nodeA, nodeB := CreateAndConnectTwo(b, false)
|
2019-09-28 12:20:57 +00:00
|
|
|
|
|
|
|
msgLen := 1500 // typical MTU
|
|
|
|
done := CreateEchoListener(b, nodeA, msgLen, b.N)
|
|
|
|
|
|
|
|
if !WaitConnected(nodeA, nodeB) {
|
|
|
|
b.Fatal("nodes did not connect")
|
|
|
|
}
|
|
|
|
|
2021-05-29 16:13:59 +00:00
|
|
|
// Send
|
2019-09-28 12:20:57 +00:00
|
|
|
msg := make([]byte, msgLen)
|
2021-06-13 18:40:20 +00:00
|
|
|
rand.Read(msg[40:])
|
|
|
|
msg[0] = 0x60
|
|
|
|
copy(msg[8:24], nodeB.Address())
|
|
|
|
copy(msg[24:40], nodeA.Address())
|
|
|
|
|
2019-09-28 12:20:57 +00:00
|
|
|
buf := make([]byte, msgLen)
|
|
|
|
|
2021-05-31 11:39:53 +00:00
|
|
|
b.SetBytes(int64(msgLen))
|
2019-09-28 12:20:57 +00:00
|
|
|
b.ResetTimer()
|
|
|
|
|
2021-07-07 23:36:51 +00:00
|
|
|
addr := nodeA.LocalAddr()
|
2019-09-28 12:20:57 +00:00
|
|
|
for i := 0; i < b.N; i++ {
|
2021-07-07 23:36:51 +00:00
|
|
|
_, err := nodeB.WriteTo(msg, addr)
|
2019-09-28 12:20:57 +00:00
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
2021-07-07 23:36:51 +00:00
|
|
|
_, _, err = nodeB.ReadFrom(buf)
|
2019-09-28 12:20:57 +00:00
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
<-done
|
|
|
|
}
|