2020-11-25 13:01:53 +02:00
|
|
|
package router_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/golang/mock/gomock"
|
2020-12-04 03:36:16 +02:00
|
|
|
. "github.com/xtls/xray-core/app/router"
|
|
|
|
"github.com/xtls/xray-core/common"
|
|
|
|
"github.com/xtls/xray-core/common/net"
|
|
|
|
"github.com/xtls/xray-core/common/session"
|
2021-12-15 02:28:47 +02:00
|
|
|
"github.com/xtls/xray-core/features/dns"
|
2020-12-04 03:36:16 +02:00
|
|
|
"github.com/xtls/xray-core/features/outbound"
|
|
|
|
routing_session "github.com/xtls/xray-core/features/routing/session"
|
|
|
|
"github.com/xtls/xray-core/testing/mocks"
|
2020-11-25 13:01:53 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type mockOutboundManager struct {
|
|
|
|
outbound.Manager
|
|
|
|
outbound.HandlerSelector
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSimpleRouter(t *testing.T) {
|
|
|
|
config := &Config{
|
|
|
|
Rule: []*RoutingRule{
|
|
|
|
{
|
|
|
|
TargetTag: &RoutingRule_Tag{
|
|
|
|
Tag: "test",
|
|
|
|
},
|
|
|
|
Networks: []net.Network{net.Network_TCP},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mockCtl := gomock.NewController(t)
|
|
|
|
defer mockCtl.Finish()
|
|
|
|
|
|
|
|
mockDNS := mocks.NewDNSClient(mockCtl)
|
|
|
|
mockOhm := mocks.NewOutboundManager(mockCtl)
|
|
|
|
mockHs := mocks.NewOutboundHandlerSelector(mockCtl)
|
|
|
|
|
|
|
|
r := new(Router)
|
2021-10-26 08:00:31 +03:00
|
|
|
common.Must(r.Init(context.TODO(), config, mockDNS, &mockOutboundManager{
|
2020-11-25 13:01:53 +02:00
|
|
|
Manager: mockOhm,
|
|
|
|
HandlerSelector: mockHs,
|
2024-02-18 05:51:37 +02:00
|
|
|
}, nil))
|
2020-11-25 13:01:53 +02:00
|
|
|
|
2024-05-14 04:52:24 +03:00
|
|
|
ctx := session.ContextWithOutbounds(context.Background(), []*session.Outbound{{
|
|
|
|
Target: net.TCPDestination(net.DomainAddress("example.com"), 80),
|
|
|
|
}})
|
2020-11-25 13:01:53 +02:00
|
|
|
route, err := r.PickRoute(routing_session.AsRoutingContext(ctx))
|
|
|
|
common.Must(err)
|
|
|
|
if tag := route.GetOutboundTag(); tag != "test" {
|
|
|
|
t.Error("expect tag 'test', bug actually ", tag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSimpleBalancer(t *testing.T) {
|
|
|
|
config := &Config{
|
|
|
|
Rule: []*RoutingRule{
|
|
|
|
{
|
|
|
|
TargetTag: &RoutingRule_BalancingTag{
|
|
|
|
BalancingTag: "balance",
|
|
|
|
},
|
|
|
|
Networks: []net.Network{net.Network_TCP},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BalancingRule: []*BalancingRule{
|
|
|
|
{
|
|
|
|
Tag: "balance",
|
|
|
|
OutboundSelector: []string{"test-"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mockCtl := gomock.NewController(t)
|
|
|
|
defer mockCtl.Finish()
|
|
|
|
|
|
|
|
mockDNS := mocks.NewDNSClient(mockCtl)
|
|
|
|
mockOhm := mocks.NewOutboundManager(mockCtl)
|
|
|
|
mockHs := mocks.NewOutboundHandlerSelector(mockCtl)
|
|
|
|
|
|
|
|
mockHs.EXPECT().Select(gomock.Eq([]string{"test-"})).Return([]string{"test"})
|
|
|
|
|
|
|
|
r := new(Router)
|
2021-10-26 08:00:31 +03:00
|
|
|
common.Must(r.Init(context.TODO(), config, mockDNS, &mockOutboundManager{
|
2020-11-25 13:01:53 +02:00
|
|
|
Manager: mockOhm,
|
|
|
|
HandlerSelector: mockHs,
|
2024-02-18 05:51:37 +02:00
|
|
|
}, nil))
|
2020-11-25 13:01:53 +02:00
|
|
|
|
2024-05-14 04:52:24 +03:00
|
|
|
ctx := session.ContextWithOutbounds(context.Background(), []*session.Outbound{{
|
|
|
|
Target: net.TCPDestination(net.DomainAddress("example.com"), 80),
|
|
|
|
}})
|
2020-11-25 13:01:53 +02:00
|
|
|
route, err := r.PickRoute(routing_session.AsRoutingContext(ctx))
|
|
|
|
common.Must(err)
|
|
|
|
if tag := route.GetOutboundTag(); tag != "test" {
|
|
|
|
t.Error("expect tag 'test', bug actually ", tag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-18 05:51:37 +02:00
|
|
|
/*
|
|
|
|
|
|
|
|
Do not work right now: need a full client setup
|
|
|
|
|
|
|
|
func TestLeastLoadBalancer(t *testing.T) {
|
|
|
|
config := &Config{
|
|
|
|
Rule: []*RoutingRule{
|
|
|
|
{
|
|
|
|
TargetTag: &RoutingRule_BalancingTag{
|
|
|
|
BalancingTag: "balance",
|
|
|
|
},
|
|
|
|
Networks: []net.Network{net.Network_TCP},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BalancingRule: []*BalancingRule{
|
|
|
|
{
|
|
|
|
Tag: "balance",
|
|
|
|
OutboundSelector: []string{"test-"},
|
|
|
|
Strategy: "leastLoad",
|
|
|
|
StrategySettings: serial.ToTypedMessage(&StrategyLeastLoadConfig{
|
|
|
|
Baselines: nil,
|
|
|
|
Expected: 1,
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mockCtl := gomock.NewController(t)
|
|
|
|
defer mockCtl.Finish()
|
|
|
|
|
|
|
|
mockDNS := mocks.NewDNSClient(mockCtl)
|
|
|
|
mockOhm := mocks.NewOutboundManager(mockCtl)
|
|
|
|
mockHs := mocks.NewOutboundHandlerSelector(mockCtl)
|
|
|
|
|
|
|
|
mockHs.EXPECT().Select(gomock.Eq([]string{"test-"})).Return([]string{"test1"})
|
|
|
|
|
|
|
|
r := new(Router)
|
|
|
|
common.Must(r.Init(context.TODO(), config, mockDNS, &mockOutboundManager{
|
|
|
|
Manager: mockOhm,
|
|
|
|
HandlerSelector: mockHs,
|
|
|
|
}, nil))
|
|
|
|
ctx := session.ContextWithOutbound(context.Background(), &session.Outbound{Target: net.TCPDestination(net.DomainAddress("v2ray.com"), 80)})
|
|
|
|
route, err := r.PickRoute(routing_session.AsRoutingContext(ctx))
|
|
|
|
common.Must(err)
|
|
|
|
if tag := route.GetOutboundTag(); tag != "test1" {
|
|
|
|
t.Error("expect tag 'test1', bug actually ", tag)
|
|
|
|
}
|
|
|
|
}*/
|
|
|
|
|
2020-11-25 13:01:53 +02:00
|
|
|
func TestIPOnDemand(t *testing.T) {
|
|
|
|
config := &Config{
|
|
|
|
DomainStrategy: Config_IpOnDemand,
|
|
|
|
Rule: []*RoutingRule{
|
|
|
|
{
|
|
|
|
TargetTag: &RoutingRule_Tag{
|
|
|
|
Tag: "test",
|
|
|
|
},
|
2024-09-19 04:05:59 +03:00
|
|
|
Geoip: []*GeoIP{
|
2020-11-25 13:01:53 +02:00
|
|
|
{
|
2024-09-19 04:05:59 +03:00
|
|
|
Cidr: []*CIDR{
|
|
|
|
{
|
|
|
|
Ip: []byte{192, 168, 0, 0},
|
|
|
|
Prefix: 16,
|
|
|
|
},
|
|
|
|
},
|
2020-11-25 13:01:53 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mockCtl := gomock.NewController(t)
|
|
|
|
defer mockCtl.Finish()
|
|
|
|
|
|
|
|
mockDNS := mocks.NewDNSClient(mockCtl)
|
2021-03-07 06:39:50 +02:00
|
|
|
mockDNS.EXPECT().LookupIP(gomock.Eq("example.com"), dns.IPOption{
|
|
|
|
IPv4Enable: true,
|
|
|
|
IPv6Enable: true,
|
|
|
|
FakeEnable: false,
|
|
|
|
}).Return([]net.IP{{192, 168, 0, 1}}, nil).AnyTimes()
|
2020-11-25 13:01:53 +02:00
|
|
|
|
|
|
|
r := new(Router)
|
2024-02-18 05:51:37 +02:00
|
|
|
common.Must(r.Init(context.TODO(), config, mockDNS, nil, nil))
|
2020-11-25 13:01:53 +02:00
|
|
|
|
2024-05-14 04:52:24 +03:00
|
|
|
ctx := session.ContextWithOutbounds(context.Background(), []*session.Outbound{{
|
|
|
|
Target: net.TCPDestination(net.DomainAddress("example.com"), 80),
|
|
|
|
}})
|
2020-11-25 13:01:53 +02:00
|
|
|
route, err := r.PickRoute(routing_session.AsRoutingContext(ctx))
|
|
|
|
common.Must(err)
|
|
|
|
if tag := route.GetOutboundTag(); tag != "test" {
|
|
|
|
t.Error("expect tag 'test', bug actually ", tag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIPIfNonMatchDomain(t *testing.T) {
|
|
|
|
config := &Config{
|
|
|
|
DomainStrategy: Config_IpIfNonMatch,
|
|
|
|
Rule: []*RoutingRule{
|
|
|
|
{
|
|
|
|
TargetTag: &RoutingRule_Tag{
|
|
|
|
Tag: "test",
|
|
|
|
},
|
2024-09-19 04:05:59 +03:00
|
|
|
Geoip: []*GeoIP{
|
2020-11-25 13:01:53 +02:00
|
|
|
{
|
2024-09-19 04:05:59 +03:00
|
|
|
Cidr: []*CIDR{
|
|
|
|
{
|
|
|
|
Ip: []byte{192, 168, 0, 0},
|
|
|
|
Prefix: 16,
|
|
|
|
},
|
|
|
|
},
|
2020-11-25 13:01:53 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mockCtl := gomock.NewController(t)
|
|
|
|
defer mockCtl.Finish()
|
|
|
|
|
|
|
|
mockDNS := mocks.NewDNSClient(mockCtl)
|
2021-03-07 06:39:50 +02:00
|
|
|
mockDNS.EXPECT().LookupIP(gomock.Eq("example.com"), dns.IPOption{
|
|
|
|
IPv4Enable: true,
|
|
|
|
IPv6Enable: true,
|
|
|
|
FakeEnable: false,
|
|
|
|
}).Return([]net.IP{{192, 168, 0, 1}}, nil).AnyTimes()
|
2020-11-25 13:01:53 +02:00
|
|
|
|
|
|
|
r := new(Router)
|
2024-02-18 05:51:37 +02:00
|
|
|
common.Must(r.Init(context.TODO(), config, mockDNS, nil, nil))
|
2020-11-25 13:01:53 +02:00
|
|
|
|
2024-05-14 04:52:24 +03:00
|
|
|
ctx := session.ContextWithOutbounds(context.Background(), []*session.Outbound{{
|
|
|
|
Target: net.TCPDestination(net.DomainAddress("example.com"), 80),
|
|
|
|
}})
|
2020-11-25 13:01:53 +02:00
|
|
|
route, err := r.PickRoute(routing_session.AsRoutingContext(ctx))
|
|
|
|
common.Must(err)
|
|
|
|
if tag := route.GetOutboundTag(); tag != "test" {
|
|
|
|
t.Error("expect tag 'test', bug actually ", tag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIPIfNonMatchIP(t *testing.T) {
|
|
|
|
config := &Config{
|
|
|
|
DomainStrategy: Config_IpIfNonMatch,
|
|
|
|
Rule: []*RoutingRule{
|
|
|
|
{
|
|
|
|
TargetTag: &RoutingRule_Tag{
|
|
|
|
Tag: "test",
|
|
|
|
},
|
2024-09-19 04:05:59 +03:00
|
|
|
Geoip: []*GeoIP{
|
2020-11-25 13:01:53 +02:00
|
|
|
{
|
2024-09-19 04:05:59 +03:00
|
|
|
Cidr: []*CIDR{
|
|
|
|
{
|
|
|
|
Ip: []byte{127, 0, 0, 0},
|
|
|
|
Prefix: 8,
|
|
|
|
},
|
|
|
|
},
|
2020-11-25 13:01:53 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mockCtl := gomock.NewController(t)
|
|
|
|
defer mockCtl.Finish()
|
|
|
|
|
|
|
|
mockDNS := mocks.NewDNSClient(mockCtl)
|
|
|
|
|
|
|
|
r := new(Router)
|
2024-02-18 05:51:37 +02:00
|
|
|
common.Must(r.Init(context.TODO(), config, mockDNS, nil, nil))
|
2020-11-25 13:01:53 +02:00
|
|
|
|
2024-05-14 04:52:24 +03:00
|
|
|
ctx := session.ContextWithOutbounds(context.Background(), []*session.Outbound{{
|
|
|
|
Target: net.TCPDestination(net.LocalHostIP, 80),
|
|
|
|
}})
|
2020-11-25 13:01:53 +02:00
|
|
|
route, err := r.PickRoute(routing_session.AsRoutingContext(ctx))
|
|
|
|
common.Must(err)
|
|
|
|
if tag := route.GetOutboundTag(); tag != "test" {
|
|
|
|
t.Error("expect tag 'test', bug actually ", tag)
|
|
|
|
}
|
|
|
|
}
|