mirror of
https://github.com/netbirdio/netbird.git
synced 2025-01-22 05:49:12 +01:00
101 lines
2.6 KiB
Go
101 lines
2.6 KiB
Go
|
//go:build !ios
|
||
|
|
||
|
package routemanager
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"net"
|
||
|
"os/exec"
|
||
|
"regexp"
|
||
|
"testing"
|
||
|
|
||
|
"github.com/stretchr/testify/assert"
|
||
|
"github.com/stretchr/testify/require"
|
||
|
)
|
||
|
|
||
|
var expectedVPNint = "utun100"
|
||
|
var expectedExternalInt = "lo0"
|
||
|
var expectedInternalInt = "lo0"
|
||
|
|
||
|
func init() {
|
||
|
testCases = append(testCases, []testCase{
|
||
|
{
|
||
|
name: "To more specific route without custom dialer via vpn",
|
||
|
destination: "10.10.0.2:53",
|
||
|
expectedInterface: expectedVPNint,
|
||
|
dialer: &net.Dialer{},
|
||
|
expectedPacket: createPacketExpectation("100.64.0.1", 12345, "10.10.0.2", 53),
|
||
|
},
|
||
|
}...)
|
||
|
}
|
||
|
|
||
|
func createAndSetupDummyInterface(t *testing.T, intf string, ipAddressCIDR string) string {
|
||
|
t.Helper()
|
||
|
|
||
|
err := exec.Command("ifconfig", intf, "alias", ipAddressCIDR).Run()
|
||
|
require.NoError(t, err, "Failed to create loopback alias")
|
||
|
|
||
|
t.Cleanup(func() {
|
||
|
err := exec.Command("ifconfig", intf, ipAddressCIDR, "-alias").Run()
|
||
|
assert.NoError(t, err, "Failed to remove loopback alias")
|
||
|
})
|
||
|
|
||
|
return "lo0"
|
||
|
}
|
||
|
|
||
|
func addDummyRoute(t *testing.T, dstCIDR string, gw net.IP, _ string) {
|
||
|
t.Helper()
|
||
|
|
||
|
var originalNexthop net.IP
|
||
|
if dstCIDR == "0.0.0.0/0" {
|
||
|
var err error
|
||
|
originalNexthop, err = fetchOriginalGateway()
|
||
|
if err != nil {
|
||
|
t.Logf("Failed to fetch original gateway: %v", err)
|
||
|
}
|
||
|
|
||
|
if output, err := exec.Command("route", "delete", "-net", dstCIDR).CombinedOutput(); err != nil {
|
||
|
t.Logf("Failed to delete route: %v, output: %s", err, output)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
t.Cleanup(func() {
|
||
|
if originalNexthop != nil {
|
||
|
err := exec.Command("route", "add", "-net", dstCIDR, originalNexthop.String()).Run()
|
||
|
assert.NoError(t, err, "Failed to restore original route")
|
||
|
}
|
||
|
})
|
||
|
|
||
|
err := exec.Command("route", "add", "-net", dstCIDR, gw.String()).Run()
|
||
|
require.NoError(t, err, "Failed to add route")
|
||
|
|
||
|
t.Cleanup(func() {
|
||
|
err := exec.Command("route", "delete", "-net", dstCIDR).Run()
|
||
|
assert.NoError(t, err, "Failed to remove route")
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func fetchOriginalGateway() (net.IP, error) {
|
||
|
output, err := exec.Command("route", "-n", "get", "default").CombinedOutput()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
matches := regexp.MustCompile(`gateway: (\S+)`).FindStringSubmatch(string(output))
|
||
|
if len(matches) == 0 {
|
||
|
return nil, fmt.Errorf("gateway not found")
|
||
|
}
|
||
|
|
||
|
return net.ParseIP(matches[1]), nil
|
||
|
}
|
||
|
|
||
|
func setupDummyInterfacesAndRoutes(t *testing.T) {
|
||
|
t.Helper()
|
||
|
|
||
|
defaultDummy := createAndSetupDummyInterface(t, expectedExternalInt, "192.168.0.1/24")
|
||
|
addDummyRoute(t, "0.0.0.0/0", net.IPv4(192, 168, 0, 1), defaultDummy)
|
||
|
|
||
|
otherDummy := createAndSetupDummyInterface(t, expectedInternalInt, "192.168.1.1/24")
|
||
|
addDummyRoute(t, "10.0.0.0/8", net.IPv4(192, 168, 1, 1), otherDummy)
|
||
|
}
|