2023-01-23 18:54:21 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
2024-01-04 16:28:16 +07:00
|
|
|
* Copyright (c) 2022-2024, daeuniverse Organization <dae@v2raya.org>
|
2023-01-23 18:54:21 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
package outbound
|
|
|
|
|
|
|
|
import (
|
2023-04-23 12:27:29 +07:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2023-03-14 14:01:55 +07:00
|
|
|
"github.com/daeuniverse/dae/common/consts"
|
|
|
|
"github.com/daeuniverse/dae/component/outbound/dialer"
|
|
|
|
"github.com/daeuniverse/dae/pkg/logger"
|
2024-04-24 01:22:50 +07:00
|
|
|
"github.com/daeuniverse/outbound/pkg/fastrand"
|
2023-01-23 18:54:21 +07:00
|
|
|
)
|
|
|
|
|
2023-01-28 00:50:21 +07:00
|
|
|
const (
|
2023-02-08 19:15:24 +07:00
|
|
|
testTcpCheckUrl = "https://connectivitycheck.gstatic.com/generate_204"
|
|
|
|
testUdpCheckDns = "https://connectivitycheck.gstatic.com/generate_204"
|
2023-01-28 00:50:21 +07:00
|
|
|
)
|
|
|
|
|
2023-02-12 14:39:00 +07:00
|
|
|
var TestNetworkType = &dialer.NetworkType{
|
|
|
|
L4Proto: consts.L4ProtoStr_TCP,
|
|
|
|
IpVersion: consts.IpVersionStr_4,
|
|
|
|
IsDns: false,
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:23:36 +07:00
|
|
|
var log = logger.NewLogger("trace", false, nil)
|
|
|
|
|
2024-01-01 16:19:18 +07:00
|
|
|
func newDirectDialer(option *dialer.GlobalOption, fullcone bool) *dialer.Dialer {
|
|
|
|
_d, p := dialer.NewDirectDialer(option, true)
|
2024-01-11 20:47:05 +07:00
|
|
|
d := dialer.NewDialer(_d, option, dialer.InstanceOption{DisableCheck: false}, p)
|
2024-01-01 16:19:18 +07:00
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
2023-01-23 18:54:21 +07:00
|
|
|
func TestDialerGroup_Select_Fixed(t *testing.T) {
|
2023-01-28 00:50:21 +07:00
|
|
|
option := &dialer.GlobalOption{
|
2023-02-12 14:39:00 +07:00
|
|
|
Log: log,
|
2023-07-09 15:02:17 +07:00
|
|
|
TcpCheckOptionRaw: dialer.TcpCheckOptionRaw{Raw: []string{testTcpCheckUrl}},
|
|
|
|
CheckDnsOptionRaw: dialer.CheckDnsOptionRaw{Raw: []string{testUdpCheckDns}},
|
2023-02-12 14:39:00 +07:00
|
|
|
CheckInterval: 15 * time.Second,
|
|
|
|
CheckTolerance: 0,
|
|
|
|
CheckDnsTcp: false,
|
2023-01-28 00:50:21 +07:00
|
|
|
}
|
2023-01-23 18:54:21 +07:00
|
|
|
dialers := []*dialer.Dialer{
|
2024-01-01 16:19:18 +07:00
|
|
|
newDirectDialer(option, true),
|
|
|
|
newDirectDialer(option, false),
|
2023-01-23 18:54:21 +07:00
|
|
|
}
|
|
|
|
fixedIndex := 1
|
2024-01-01 16:19:18 +07:00
|
|
|
g := NewDialerGroup(option, "test-group", dialers, []*dialer.Annotation{{}},
|
|
|
|
DialerSelectionPolicy{
|
|
|
|
Policy: consts.DialerSelectionPolicy_Fixed,
|
|
|
|
FixedIndex: fixedIndex,
|
|
|
|
}, func(alive bool, networkType *dialer.NetworkType, isInit bool) {})
|
2023-01-23 18:54:21 +07:00
|
|
|
for i := 0; i < 10; i++ {
|
2023-07-09 15:02:17 +07:00
|
|
|
d, _, err := g.Select(TestNetworkType, false)
|
2023-01-23 18:54:21 +07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if d != dialers[fixedIndex] {
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fixedIndex = 0
|
|
|
|
g.selectionPolicy.FixedIndex = fixedIndex
|
|
|
|
for i := 0; i < 10; i++ {
|
2023-07-09 15:02:17 +07:00
|
|
|
d, _, err := g.Select(TestNetworkType, false)
|
2023-01-23 18:54:21 +07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if d != dialers[fixedIndex] {
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDialerGroup_Select_MinLastLatency(t *testing.T) {
|
2023-02-12 14:39:00 +07:00
|
|
|
|
2023-01-28 00:50:21 +07:00
|
|
|
option := &dialer.GlobalOption{
|
2023-02-12 14:39:00 +07:00
|
|
|
Log: log,
|
2023-07-09 15:02:17 +07:00
|
|
|
TcpCheckOptionRaw: dialer.TcpCheckOptionRaw{Raw: []string{testTcpCheckUrl}},
|
|
|
|
CheckDnsOptionRaw: dialer.CheckDnsOptionRaw{Raw: []string{testUdpCheckDns}},
|
2023-02-12 14:39:00 +07:00
|
|
|
CheckInterval: 15 * time.Second,
|
2023-01-28 00:50:21 +07:00
|
|
|
}
|
2023-01-23 18:54:21 +07:00
|
|
|
dialers := []*dialer.Dialer{
|
2024-01-01 16:19:18 +07:00
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
}
|
|
|
|
g := NewDialerGroup(option, "test-group", dialers, []*dialer.Annotation{{}},
|
|
|
|
DialerSelectionPolicy{
|
|
|
|
Policy: consts.DialerSelectionPolicy_MinLastLatency,
|
|
|
|
}, func(alive bool, networkType *dialer.NetworkType, isInit bool) {})
|
2023-01-23 18:54:21 +07:00
|
|
|
|
|
|
|
// Test 1000 times.
|
|
|
|
for i := 0; i < 1000; i++ {
|
|
|
|
var minLatency time.Duration
|
|
|
|
jMinLatency := -1
|
|
|
|
for j, d := range dialers {
|
|
|
|
// Simulate a latency test.
|
|
|
|
var (
|
|
|
|
latency time.Duration
|
|
|
|
alive bool
|
|
|
|
)
|
|
|
|
// 20% chance for timeout.
|
|
|
|
if fastrand.Intn(5) == 0 {
|
|
|
|
// Simulate a timeout test.
|
|
|
|
latency = 1000 * time.Millisecond
|
|
|
|
alive = false
|
|
|
|
} else {
|
|
|
|
// Simulate a normal test.
|
|
|
|
latency = time.Duration(fastrand.Int63n(int64(1000 * time.Millisecond)))
|
|
|
|
alive = true
|
|
|
|
}
|
2023-02-12 14:39:00 +07:00
|
|
|
d.MustGetLatencies10(TestNetworkType).AppendLatency(latency)
|
2023-01-23 18:54:21 +07:00
|
|
|
if jMinLatency == -1 || latency < minLatency {
|
|
|
|
jMinLatency = j
|
|
|
|
minLatency = latency
|
|
|
|
}
|
2023-02-12 14:39:00 +07:00
|
|
|
g.MustGetAliveDialerSet(TestNetworkType).NotifyLatencyChange(d, alive)
|
2023-01-23 18:54:21 +07:00
|
|
|
}
|
2023-07-09 15:02:17 +07:00
|
|
|
d, _, err := g.Select(TestNetworkType, false)
|
2023-01-23 18:54:21 +07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if d != dialers[jMinLatency] {
|
|
|
|
// Get index of d.
|
|
|
|
indexD := -1
|
|
|
|
for j := range dialers {
|
|
|
|
if d == dialers[j] {
|
|
|
|
indexD = j
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
t.Errorf("dialers[%v] expected, but dialers[%v] selected", jMinLatency, indexD)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDialerGroup_Select_Random(t *testing.T) {
|
2023-02-12 14:39:00 +07:00
|
|
|
|
2023-01-28 00:50:21 +07:00
|
|
|
option := &dialer.GlobalOption{
|
2023-02-12 14:39:00 +07:00
|
|
|
Log: log,
|
2023-07-09 15:02:17 +07:00
|
|
|
TcpCheckOptionRaw: dialer.TcpCheckOptionRaw{Raw: []string{testTcpCheckUrl}},
|
|
|
|
CheckDnsOptionRaw: dialer.CheckDnsOptionRaw{Raw: []string{testUdpCheckDns}},
|
2023-02-12 14:39:00 +07:00
|
|
|
CheckInterval: 15 * time.Second,
|
2023-01-28 00:50:21 +07:00
|
|
|
}
|
2023-01-23 18:54:21 +07:00
|
|
|
dialers := []*dialer.Dialer{
|
2024-01-01 16:19:18 +07:00
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
}
|
|
|
|
g := NewDialerGroup(option, "test-group", dialers, []*dialer.Annotation{{}},
|
|
|
|
DialerSelectionPolicy{
|
|
|
|
Policy: consts.DialerSelectionPolicy_Random,
|
|
|
|
}, func(alive bool, networkType *dialer.NetworkType, isInit bool) {})
|
2023-01-23 18:54:21 +07:00
|
|
|
count := make([]int, len(dialers))
|
|
|
|
for i := 0; i < 100; i++ {
|
2023-07-09 15:02:17 +07:00
|
|
|
d, _, err := g.Select(TestNetworkType, false)
|
2023-01-23 18:54:21 +07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
for j, dd := range dialers {
|
|
|
|
if d == dd {
|
|
|
|
count[j]++
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for i, c := range count {
|
|
|
|
if c == 0 {
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
t.Logf("count[%v]: %v", i, c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDialerGroup_SetAlive(t *testing.T) {
|
2023-02-12 14:39:00 +07:00
|
|
|
|
2023-01-28 00:50:21 +07:00
|
|
|
option := &dialer.GlobalOption{
|
2023-02-12 14:39:00 +07:00
|
|
|
Log: log,
|
2023-07-09 15:02:17 +07:00
|
|
|
TcpCheckOptionRaw: dialer.TcpCheckOptionRaw{Raw: []string{testTcpCheckUrl}},
|
|
|
|
CheckDnsOptionRaw: dialer.CheckDnsOptionRaw{Raw: []string{testUdpCheckDns}},
|
2023-02-12 14:39:00 +07:00
|
|
|
CheckInterval: 15 * time.Second,
|
2023-01-28 00:50:21 +07:00
|
|
|
}
|
2023-01-23 18:54:21 +07:00
|
|
|
dialers := []*dialer.Dialer{
|
2024-01-01 16:19:18 +07:00
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
newDirectDialer(option, false),
|
|
|
|
}
|
|
|
|
g := NewDialerGroup(option, "test-group", dialers, []*dialer.Annotation{{}},
|
|
|
|
DialerSelectionPolicy{
|
|
|
|
Policy: consts.DialerSelectionPolicy_Random,
|
|
|
|
}, func(alive bool, networkType *dialer.NetworkType, isInit bool) {})
|
2023-01-23 18:54:21 +07:00
|
|
|
zeroTarget := 3
|
2023-02-12 14:39:00 +07:00
|
|
|
g.MustGetAliveDialerSet(TestNetworkType).NotifyLatencyChange(dialers[zeroTarget], false)
|
2023-01-23 18:54:21 +07:00
|
|
|
count := make([]int, len(dialers))
|
|
|
|
for i := 0; i < 100; i++ {
|
2023-07-09 15:02:17 +07:00
|
|
|
d, _, err := g.Select(TestNetworkType, false)
|
2023-01-23 18:54:21 +07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
for j, dd := range dialers {
|
|
|
|
if d == dd {
|
|
|
|
count[j]++
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for i, c := range count {
|
|
|
|
if c == 0 && i != zeroTarget {
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
t.Logf("count[%v]: %v", i, c)
|
|
|
|
}
|
|
|
|
if count[zeroTarget] != 0 {
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
}
|