2021-01-05 11:02:52 -08:00
|
|
|
// Copyright (c) 2021 Tailscale Inc & AUTHORS All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2021-02-04 13:12:42 -08:00
|
|
|
package ipnlocal
|
2021-01-05 11:02:52 -08:00
|
|
|
|
|
|
|
import (
|
2021-03-04 12:04:31 -08:00
|
|
|
"reflect"
|
2021-02-04 13:12:42 -08:00
|
|
|
"testing"
|
|
|
|
|
2021-01-05 11:02:52 -08:00
|
|
|
"inet.af/netaddr"
|
2021-03-17 17:04:32 -07:00
|
|
|
"tailscale.com/net/interfaces"
|
2021-02-22 20:43:35 -08:00
|
|
|
"tailscale.com/net/tsaddr"
|
2021-01-05 11:02:52 -08:00
|
|
|
"tailscale.com/tailcfg"
|
2021-02-05 15:44:46 -08:00
|
|
|
"tailscale.com/types/netmap"
|
2021-03-04 12:04:31 -08:00
|
|
|
"tailscale.com/wgengine/wgcfg"
|
2021-01-05 11:02:52 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestNetworkMapCompare(t *testing.T) {
|
|
|
|
prefix1, err := netaddr.ParseIPPrefix("192.168.0.0/24")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
node1 := &tailcfg.Node{Addresses: []netaddr.IPPrefix{prefix1}}
|
|
|
|
|
|
|
|
prefix2, err := netaddr.ParseIPPrefix("10.0.0.0/8")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
node2 := &tailcfg.Node{Addresses: []netaddr.IPPrefix{prefix2}}
|
|
|
|
|
|
|
|
tests := []struct {
|
2021-01-08 09:32:55 -08:00
|
|
|
name string
|
2021-02-05 15:44:46 -08:00
|
|
|
a, b *netmap.NetworkMap
|
2021-01-05 11:02:52 -08:00
|
|
|
want bool
|
|
|
|
}{
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"both nil",
|
2021-01-05 11:02:52 -08:00
|
|
|
nil,
|
|
|
|
nil,
|
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"b nil",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{},
|
2021-01-05 11:02:52 -08:00
|
|
|
nil,
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"a nil",
|
2021-01-05 11:02:52 -08:00
|
|
|
nil,
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{},
|
2021-01-05 11:02:52 -08:00
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"both default",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{},
|
|
|
|
&netmap.NetworkMap{},
|
2021-01-05 11:02:52 -08:00
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"names identical",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Name: "map1"},
|
|
|
|
&netmap.NetworkMap{Name: "map1"},
|
2021-01-05 11:02:52 -08:00
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"names differ",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Name: "map1"},
|
|
|
|
&netmap.NetworkMap{Name: "map2"},
|
2021-01-05 11:02:52 -08:00
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"Peers identical",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{}},
|
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{}},
|
2021-01-05 11:02:52 -08:00
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"Peer list length",
|
2021-01-05 11:02:52 -08:00
|
|
|
// length of Peers list differs
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{{}}},
|
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{}},
|
2021-01-05 11:02:52 -08:00
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"Node names identical",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{&tailcfg.Node{Name: "A"}}},
|
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{&tailcfg.Node{Name: "A"}}},
|
2021-01-05 11:02:52 -08:00
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"Node names differ",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{&tailcfg.Node{Name: "A"}}},
|
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{&tailcfg.Node{Name: "B"}}},
|
2021-01-05 11:02:52 -08:00
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"Node lists identical",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{node1, node1}},
|
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{node1, node1}},
|
2021-01-05 11:02:52 -08:00
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"Node lists differ",
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{node1, node1}},
|
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{node1, node2}},
|
2021-01-05 11:02:52 -08:00
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
2021-01-08 09:32:55 -08:00
|
|
|
"Node Users differ",
|
2021-01-05 11:02:52 -08:00
|
|
|
// User field is not checked.
|
2021-02-05 15:44:46 -08:00
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{&tailcfg.Node{User: 0}}},
|
|
|
|
&netmap.NetworkMap{Peers: []*tailcfg.Node{&tailcfg.Node{User: 1}}},
|
2021-01-05 11:02:52 -08:00
|
|
|
true,
|
|
|
|
},
|
|
|
|
}
|
2021-01-08 09:32:55 -08:00
|
|
|
for _, tt := range tests {
|
2021-01-05 11:02:52 -08:00
|
|
|
got := dnsMapsEqual(tt.a, tt.b)
|
|
|
|
if got != tt.want {
|
2021-01-08 09:32:55 -08:00
|
|
|
t.Errorf("%s: Equal = %v; want %v", tt.name, got, tt.want)
|
2021-01-05 11:02:52 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-22 20:43:35 -08:00
|
|
|
|
2021-03-17 17:04:32 -07:00
|
|
|
func inRemove(ip netaddr.IP) bool {
|
|
|
|
for _, pfx := range removeFromDefaultRoute {
|
|
|
|
if pfx.Contains(ip) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-02-22 20:43:35 -08:00
|
|
|
func TestShrinkDefaultRoute(t *testing.T) {
|
|
|
|
tests := []struct {
|
2021-03-17 17:04:32 -07:00
|
|
|
route string
|
|
|
|
in []string
|
|
|
|
out []string
|
|
|
|
localIPFn func(netaddr.IP) bool // true if this machine's local IP address should be "in" after shrinking.
|
2021-02-22 20:43:35 -08:00
|
|
|
}{
|
|
|
|
{
|
|
|
|
route: "0.0.0.0/0",
|
|
|
|
in: []string{"1.2.3.4", "25.0.0.1"},
|
|
|
|
out: []string{
|
|
|
|
"10.0.0.1",
|
|
|
|
"10.255.255.255",
|
|
|
|
"192.168.0.1",
|
|
|
|
"192.168.255.255",
|
|
|
|
"172.16.0.1",
|
|
|
|
"172.31.255.255",
|
|
|
|
"100.101.102.103",
|
2021-03-17 17:04:32 -07:00
|
|
|
"224.0.0.1",
|
|
|
|
"169.254.169.254",
|
2021-02-22 20:43:35 -08:00
|
|
|
// Some random IPv6 stuff that shouldn't be in a v4
|
|
|
|
// default route.
|
|
|
|
"fe80::",
|
|
|
|
"2601::1",
|
|
|
|
},
|
2021-03-17 17:04:32 -07:00
|
|
|
localIPFn: func(ip netaddr.IP) bool { return !inRemove(ip) && ip.Is4() },
|
2021-02-22 20:43:35 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
route: "::/0",
|
|
|
|
in: []string{"::1", "2601::1"},
|
|
|
|
out: []string{
|
|
|
|
"fe80::1",
|
2021-03-17 17:04:32 -07:00
|
|
|
"ff00::1",
|
2021-02-22 20:43:35 -08:00
|
|
|
tsaddr.TailscaleULARange().IP.String(),
|
|
|
|
},
|
2021-03-17 17:04:32 -07:00
|
|
|
localIPFn: func(ip netaddr.IP) bool { return !inRemove(ip) && ip.Is6() },
|
2021-02-22 20:43:35 -08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
def := netaddr.MustParseIPPrefix(test.route)
|
|
|
|
got, err := shrinkDefaultRoute(def)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("shrinkDefaultRoute(%q): %v", test.route, err)
|
|
|
|
}
|
|
|
|
for _, ip := range test.in {
|
|
|
|
if !got.Contains(netaddr.MustParseIP(ip)) {
|
|
|
|
t.Errorf("shrink(%q).Contains(%v) = false, want true", test.route, ip)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, ip := range test.out {
|
|
|
|
if got.Contains(netaddr.MustParseIP(ip)) {
|
|
|
|
t.Errorf("shrink(%q).Contains(%v) = true, want false", test.route, ip)
|
|
|
|
}
|
|
|
|
}
|
2021-03-17 17:04:32 -07:00
|
|
|
ips, _, err := interfaces.LocalAddresses()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
for _, ip := range ips {
|
|
|
|
want := test.localIPFn(ip)
|
|
|
|
if gotContains := got.Contains(ip); gotContains != want {
|
|
|
|
t.Errorf("shrink(%q).Contains(%v) = %v, want %v", test.route, ip, gotContains, want)
|
|
|
|
}
|
|
|
|
}
|
2021-02-22 20:43:35 -08:00
|
|
|
}
|
|
|
|
}
|
2021-03-04 12:04:31 -08:00
|
|
|
|
|
|
|
func TestPeerRoutes(t *testing.T) {
|
|
|
|
pp := netaddr.MustParseIPPrefix
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
peers []wgcfg.Peer
|
|
|
|
want []netaddr.IPPrefix
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "small_v4",
|
|
|
|
peers: []wgcfg.Peer{
|
|
|
|
{
|
|
|
|
AllowedIPs: []netaddr.IPPrefix{
|
|
|
|
pp("100.101.102.103/32"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: []netaddr.IPPrefix{
|
|
|
|
pp("100.101.102.103/32"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "big_v4",
|
|
|
|
peers: []wgcfg.Peer{
|
|
|
|
{
|
|
|
|
AllowedIPs: []netaddr.IPPrefix{
|
|
|
|
pp("100.101.102.103/32"),
|
|
|
|
pp("100.101.102.104/32"),
|
|
|
|
pp("100.101.102.105/32"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: []netaddr.IPPrefix{
|
|
|
|
pp("100.64.0.0/10"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "has_1_v6",
|
|
|
|
peers: []wgcfg.Peer{
|
|
|
|
{
|
|
|
|
AllowedIPs: []netaddr.IPPrefix{
|
|
|
|
pp("fd7a:115c:a1e0:ab12:4843:cd96:6258:b240/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: []netaddr.IPPrefix{
|
|
|
|
pp("fd7a:115c:a1e0::/48"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "has_2_v6",
|
|
|
|
peers: []wgcfg.Peer{
|
|
|
|
{
|
|
|
|
AllowedIPs: []netaddr.IPPrefix{
|
|
|
|
pp("fd7a:115c:a1e0:ab12:4843:cd96:6258:b240/128"),
|
|
|
|
pp("fd7a:115c:a1e0:ab12:4843:cd96:6258:b241/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: []netaddr.IPPrefix{
|
|
|
|
pp("fd7a:115c:a1e0::/48"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "big_v4_big_v6",
|
|
|
|
peers: []wgcfg.Peer{
|
|
|
|
{
|
|
|
|
AllowedIPs: []netaddr.IPPrefix{
|
|
|
|
pp("100.101.102.103/32"),
|
|
|
|
pp("100.101.102.104/32"),
|
|
|
|
pp("100.101.102.105/32"),
|
|
|
|
pp("fd7a:115c:a1e0:ab12:4843:cd96:6258:b240/128"),
|
|
|
|
pp("fd7a:115c:a1e0:ab12:4843:cd96:6258:b241/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: []netaddr.IPPrefix{
|
|
|
|
pp("fd7a:115c:a1e0::/48"),
|
|
|
|
pp("100.64.0.0/10"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
got := peerRoutes(tt.peers, 2)
|
|
|
|
if !reflect.DeepEqual(got, tt.want) {
|
|
|
|
t.Errorf("got = %v; want %v", got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2021-04-04 21:35:52 -07:00
|
|
|
|
|
|
|
func TestPeerAPIBase(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
nm *netmap.NetworkMap
|
|
|
|
peer *tailcfg.Node
|
|
|
|
want string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "nil_netmap",
|
|
|
|
peer: new(tailcfg.Node),
|
|
|
|
want: "",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "nil_peer",
|
|
|
|
nm: new(netmap.NetworkMap),
|
|
|
|
want: "",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "self_only_4_them_both",
|
|
|
|
nm: &netmap.NetworkMap{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.1/32"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
peer: &tailcfg.Node{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.2/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::2/128"),
|
|
|
|
},
|
|
|
|
Hostinfo: tailcfg.Hostinfo{
|
|
|
|
Services: []tailcfg.Service{
|
|
|
|
{Proto: "peerapi4", Port: 444},
|
|
|
|
{Proto: "peerapi6", Port: 666},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: "http://100.64.1.2:444",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "self_only_6_them_both",
|
|
|
|
nm: &netmap.NetworkMap{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("fe70::1/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
peer: &tailcfg.Node{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.2/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::2/128"),
|
|
|
|
},
|
|
|
|
Hostinfo: tailcfg.Hostinfo{
|
|
|
|
Services: []tailcfg.Service{
|
|
|
|
{Proto: "peerapi4", Port: 444},
|
|
|
|
{Proto: "peerapi6", Port: 666},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: "http://[fe70::2]:666",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "self_both_them_only_4",
|
|
|
|
nm: &netmap.NetworkMap{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.1/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::1/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
peer: &tailcfg.Node{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.2/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::2/128"),
|
|
|
|
},
|
|
|
|
Hostinfo: tailcfg.Hostinfo{
|
|
|
|
Services: []tailcfg.Service{
|
|
|
|
{Proto: "peerapi4", Port: 444},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: "http://100.64.1.2:444",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "self_both_them_only_6",
|
|
|
|
nm: &netmap.NetworkMap{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.1/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::1/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
peer: &tailcfg.Node{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.2/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::2/128"),
|
|
|
|
},
|
|
|
|
Hostinfo: tailcfg.Hostinfo{
|
|
|
|
Services: []tailcfg.Service{
|
|
|
|
{Proto: "peerapi6", Port: 666},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: "http://[fe70::2]:666",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "self_both_them_no_peerapi_service",
|
|
|
|
nm: &netmap.NetworkMap{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.1/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::1/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
peer: &tailcfg.Node{
|
|
|
|
Addresses: []netaddr.IPPrefix{
|
|
|
|
netaddr.MustParseIPPrefix("100.64.1.2/32"),
|
|
|
|
netaddr.MustParseIPPrefix("fe70::2/128"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: "",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
got := peerAPIBase(tt.nm, tt.peer)
|
|
|
|
if got != tt.want {
|
|
|
|
t.Errorf("got %q; want %q", got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|