netbird/client/iface/wgproxy/proxy_test.go

199 lines
4.4 KiB
Go
Raw Normal View History

2025-02-17 21:26:37 +01:00
//go:build linux && !android
package wgproxy
import (
"context"
"net"
"testing"
2025-02-17 21:47:34 +01:00
"github.com/netbirdio/netbird/client/iface/bind"
bindproxy "github.com/netbirdio/netbird/client/iface/wgproxy/bind"
"github.com/netbirdio/netbird/client/iface/wgproxy/ebpf"
2025-02-17 21:26:37 +01:00
"github.com/netbirdio/netbird/client/iface/wgproxy/udp"
2025-02-17 21:47:34 +01:00
"github.com/netbirdio/netbird/util"
)
2025-02-17 21:26:37 +01:00
func init() {
_ = util.InitLog("debug", "console")
}
2025-02-17 21:47:34 +01:00
2025-02-17 21:26:37 +01:00
func TestProxyRedirect(t *testing.T) {
ebpfProxy := ebpf.NewWGEBPFProxy(51831)
if err := ebpfProxy.Listen(); err != nil {
t.Fatalf("failed to initialize ebpf proxy: %s", err)
}
2025-02-17 21:26:37 +01:00
defer func() {
if err := ebpfProxy.Free(); err != nil {
t.Errorf("failed to free ebpf proxy: %s", err)
}
}()
2025-02-17 21:26:37 +01:00
tests := []struct {
2025-02-17 21:47:34 +01:00
name string
proxy Proxy
wgPort int
endpointAddr *net.UDPAddr
2025-02-17 21:26:37 +01:00
}{
{
name: "ebpf kernel proxy",
proxy: ebpf.NewProxyWrapper(ebpfProxy),
wgPort: 51831,
},
{
name: "udp kernel proxy",
proxy: udp.NewWGUDPProxy(51832),
wgPort: 51832,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
2025-02-17 21:47:34 +01:00
redirectTraffic(t, tt.proxy, tt.wgPort, tt.endpointAddr)
2025-02-17 21:26:37 +01:00
})
}
}
2025-02-17 21:47:34 +01:00
func redirectTraffic(t *testing.T, proxy Proxy, wgPort int, endPointAddr *net.UDPAddr) {
2025-02-17 21:26:37 +01:00
t.Helper()
2025-02-17 21:26:37 +01:00
msgHelloFromRelay := []byte("hello from relay")
msgRedirected := [][]byte{
[]byte("hello 1. to p2p"),
[]byte("hello 2. to p2p"),
[]byte("hello 3. to p2p"),
}
2025-02-17 21:26:37 +01:00
dummyWgListener, err := net.ListenUDP("udp", &net.UDPAddr{
IP: net.IPv4(127, 0, 0, 1),
Port: wgPort})
if err != nil {
t.Fatalf("failed to listen on udp port: %s", err)
}
2025-02-17 22:06:04 +01:00
relayedServer, _ := net.ListenUDP("udp",
2025-02-17 21:26:37 +01:00
&net.UDPAddr{
IP: net.IPv4(127, 0, 0, 1),
Port: 1234,
},
)
2025-02-17 22:06:04 +01:00
relayedConn, _ := net.Dial("udp", "127.0.0.1:1234")
2025-02-17 21:26:37 +01:00
defer func() {
_ = dummyWgListener.Close()
_ = relayedConn.Close()
_ = relayedServer.Close()
}()
2025-02-17 21:47:34 +01:00
if err := proxy.AddTurnConn(context.Background(), endPointAddr, relayedConn); err != nil {
2025-02-17 21:26:37 +01:00
t.Errorf("error: %v", err)
}
2025-02-17 21:26:37 +01:00
defer func() {
if err := proxy.CloseConn(); err != nil {
t.Errorf("error: %v", err)
}
}()
2025-02-17 21:26:37 +01:00
proxy.Work()
2025-02-17 21:26:37 +01:00
if _, err := relayedServer.WriteTo(msgHelloFromRelay, relayedConn.LocalAddr()); err != nil {
t.Errorf("error relayedServer.Write(msgHelloFromRelay): %v", err)
}
n, err := dummyWgListener.Read(make([]byte, 1024))
if err != nil {
t.Errorf("error: %v", err)
}
2025-02-17 21:26:37 +01:00
if n != len(msgHelloFromRelay) {
t.Errorf("expected %d bytes, got %d", len(msgHelloFromRelay), n)
}
p2pEndpointAddr := &net.UDPAddr{
IP: net.IPv4(192, 168, 0, 56),
Port: 1234,
}
proxy.RedirectAs(p2pEndpointAddr)
for _, msg := range msgRedirected {
if _, err := relayedServer.WriteTo(msg, relayedConn.LocalAddr()); err != nil {
t.Errorf("error: %v", err)
}
}
for i := 0; i < len(msgRedirected); i++ {
buf := make([]byte, 1024)
n, rAddr, err := dummyWgListener.ReadFrom(buf)
if err != nil {
t.Errorf("error: %v", err)
}
if rAddr.String() != p2pEndpointAddr.String() {
t.Errorf("expected %s, got %s", p2pEndpointAddr.String(), rAddr.String())
}
if string(buf[:n]) != string(msgRedirected[i]) {
t.Errorf("expected %s, got %s", string(msgRedirected[i]), string(buf[:n]))
}
}
}
2025-02-17 21:47:34 +01:00
func TestProxyCloseByRemoteConn(t *testing.T) {
ctx := context.Background()
2025-02-17 21:26:37 +01:00
ebpfProxy := ebpf.NewWGEBPFProxy(51831)
if err := ebpfProxy.Listen(); err != nil {
t.Fatalf("failed to initialize ebpf proxy: %s", err)
}
defer func() {
if err := ebpfProxy.Free(); err != nil {
t.Errorf("failed to free ebpf proxy: %s", err)
}
}()
2025-02-17 21:47:34 +01:00
iceBind := bind.NewICEBind(nil, nil)
endpointAddress := &net.UDPAddr{
IP: net.IPv4(10, 0, 0, 1),
Port: 1234,
}
tests := []struct {
2025-02-17 21:47:34 +01:00
name string
proxy Proxy
endpointAddress *net.UDPAddr
}{
{
name: "ebpf proxy",
2025-02-17 21:26:37 +01:00
proxy: ebpf.NewProxyWrapper(ebpfProxy),
},
{
name: "udp proxy",
proxy: udp.NewWGUDPProxy(51832),
},
2025-02-17 21:47:34 +01:00
{
name: "bind proxy",
proxy: bindproxy.NewProxyBind(iceBind),
endpointAddress: endpointAddress,
},
}
2025-02-17 21:26:37 +01:00
relayedConn, _ := net.Dial("udp", "127.0.0.1:1234")
defer func() {
_ = relayedConn.Close()
}()
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
2025-02-17 21:47:34 +01:00
err := tt.proxy.AddTurnConn(ctx, endpointAddress, relayedConn)
if err != nil {
t.Errorf("error: %v", err)
}
_ = relayedConn.Close()
if err := tt.proxy.CloseConn(); err != nil {
t.Errorf("error: %v", err)
}
})
}
}