mirror of
				https://github.com/juanfont/headscale.git
				synced 2025-10-28 10:51:44 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			469 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			469 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package routes
 | |
| 
 | |
| import (
 | |
| 	"net/netip"
 | |
| 	"sync"
 | |
| 	"testing"
 | |
| 
 | |
| 	"github.com/google/go-cmp/cmp"
 | |
| 	"github.com/google/go-cmp/cmp/cmpopts"
 | |
| 	"github.com/juanfont/headscale/hscontrol/types"
 | |
| 	"github.com/juanfont/headscale/hscontrol/util"
 | |
| 	"tailscale.com/util/set"
 | |
| )
 | |
| 
 | |
| // mp is a helper function that wraps netip.MustParsePrefix.
 | |
| func mp(prefix string) netip.Prefix {
 | |
| 	return netip.MustParsePrefix(prefix)
 | |
| }
 | |
| 
 | |
| func TestPrimaryRoutes(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		name              string
 | |
| 		operations        func(pr *PrimaryRoutes) bool
 | |
| 		expectedRoutes    map[types.NodeID]set.Set[netip.Prefix]
 | |
| 		expectedPrimaries map[netip.Prefix]types.NodeID
 | |
| 		expectedIsPrimary map[types.NodeID]bool
 | |
| 		expectedChange    bool
 | |
| 
 | |
| 		// primaries is a map of prefixes to the node that is the primary for that prefix.
 | |
| 		primaries map[netip.Prefix]types.NodeID
 | |
| 		isPrimary map[types.NodeID]bool
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "single-node-registers-single-route",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				return pr.SetRoutes(1, mp("192.168.1.0/24"))
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 1,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "multiple-nodes-register-different-routes",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24"))
 | |
| 				return pr.SetRoutes(2, mp("192.168.2.0/24"))
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.2.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 1,
 | |
| 				mp("192.168.2.0/24"): 2,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 				2: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "multiple-nodes-register-overlapping-routes",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24"))        // true
 | |
| 				return pr.SetRoutes(2, mp("192.168.1.0/24")) // false
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 1,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 			},
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "node-deregisters-a-route",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24"))
 | |
| 				return pr.SetRoutes(1) // Deregister by setting no routes
 | |
| 			},
 | |
| 			expectedRoutes:    nil,
 | |
| 			expectedPrimaries: nil,
 | |
| 			expectedIsPrimary: nil,
 | |
| 			expectedChange:    true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "node-deregisters-one-of-multiple-routes",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24"), mp("192.168.2.0/24"))
 | |
| 				return pr.SetRoutes(1, mp("192.168.2.0/24")) // Deregister one route by setting the remaining route
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.2.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.2.0/24"): 1,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "node-registers-and-deregisters-routes-in-sequence",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24"))
 | |
| 				pr.SetRoutes(2, mp("192.168.2.0/24"))
 | |
| 				pr.SetRoutes(1) // Deregister by setting no routes
 | |
| 				return pr.SetRoutes(1, mp("192.168.3.0/24"))
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.3.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.2.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.2.0/24"): 2,
 | |
| 				mp("192.168.3.0/24"): 1,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 				2: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "multiple-nodes-register-same-route",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24"))        // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24"))        // true
 | |
| 				return pr.SetRoutes(3, mp("192.168.1.0/24")) // false
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				3: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 1,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 			},
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "register-multiple-routes-shift-primary-check-primary",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 1 primary
 | |
| 				pr.SetRoutes(3, mp("192.168.1.0/24")) // false, 1 primary
 | |
| 				return pr.SetRoutes(1)                // true, 2 primary
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				2: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				3: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 2,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				2: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "primary-route-map-is-cleared-up-no-primary",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 1 primary
 | |
| 				pr.SetRoutes(3, mp("192.168.1.0/24")) // false, 1 primary
 | |
| 				pr.SetRoutes(1)                       // true, 2 primary
 | |
| 
 | |
| 				return pr.SetRoutes(2) // true, no primary
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				3: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 3,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				3: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "primary-route-map-is-cleared-up-all-no-primary",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 1 primary
 | |
| 				pr.SetRoutes(3, mp("192.168.1.0/24")) // false, 1 primary
 | |
| 				pr.SetRoutes(1)                       // true, 2 primary
 | |
| 				pr.SetRoutes(2)                       // true, no primary
 | |
| 
 | |
| 				return pr.SetRoutes(3) // false, no primary
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "primary-route-map-is-cleared-up",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 1 primary
 | |
| 				pr.SetRoutes(3, mp("192.168.1.0/24")) // false, 1 primary
 | |
| 				pr.SetRoutes(1)                       // true, 2 primary
 | |
| 
 | |
| 				return pr.SetRoutes(2) // true, no primary
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				3: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 3,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				3: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "primary-route-no-flake",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 1 primary
 | |
| 				pr.SetRoutes(3, mp("192.168.1.0/24")) // false, 1 primary
 | |
| 				pr.SetRoutes(1)                       // true, 2 primary
 | |
| 
 | |
| 				return pr.SetRoutes(1, mp("192.168.1.0/24")) // false, 2 primary
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				3: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 2,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				2: true,
 | |
| 			},
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "primary-route-no-flake-check-old-primary",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 1 primary
 | |
| 				pr.SetRoutes(3, mp("192.168.1.0/24")) // false, 1 primary
 | |
| 				pr.SetRoutes(1)                       // true, 2 primary
 | |
| 
 | |
| 				return pr.SetRoutes(1, mp("192.168.1.0/24")) // false, 2 primary
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				3: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 2,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				2: true,
 | |
| 			},
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "primary-route-no-flake-full-integration",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // false
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 1 primary
 | |
| 				pr.SetRoutes(3, mp("192.168.1.0/24")) // false, 1 primary
 | |
| 				pr.SetRoutes(1)                       // true, 2 primary
 | |
| 				pr.SetRoutes(2)                       // true, 3 primary
 | |
| 				pr.SetRoutes(1, mp("192.168.1.0/24")) // true, 3 primary
 | |
| 				pr.SetRoutes(2, mp("192.168.1.0/24")) // true, 3 primary
 | |
| 				pr.SetRoutes(1)                       // true, 3 primary
 | |
| 
 | |
| 				return pr.SetRoutes(1, mp("192.168.1.0/24")) // false, 3 primary
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				3: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 3,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				3: true,
 | |
| 			},
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "multiple-nodes-register-same-route-and-exit",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("0.0.0.0/0"), mp("192.168.1.0/24"))
 | |
| 				return pr.SetRoutes(2, mp("192.168.1.0/24"))
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 1,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 			},
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "deregister-non-existent-route",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				return pr.SetRoutes(1) // Deregister by setting no routes
 | |
| 			},
 | |
| 			expectedRoutes: nil,
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "register-empty-prefix-list",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				return pr.SetRoutes(1)
 | |
| 			},
 | |
| 			expectedRoutes: nil,
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "exit-nodes",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				pr.SetRoutes(1, mp("10.0.0.0/16"), mp("0.0.0.0/0"), mp("::/0"))
 | |
| 				pr.SetRoutes(3, mp("0.0.0.0/0"), mp("::/0"))
 | |
| 				return pr.SetRoutes(2, mp("0.0.0.0/0"), mp("::/0"))
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("10.0.0.0/16"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("10.0.0.0/16"): 1,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 			},
 | |
| 			expectedChange: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "concurrent-access",
 | |
| 			operations: func(pr *PrimaryRoutes) bool {
 | |
| 				var wg sync.WaitGroup
 | |
| 				wg.Add(2)
 | |
| 				var change1, change2 bool
 | |
| 				go func() {
 | |
| 					defer wg.Done()
 | |
| 					change1 = pr.SetRoutes(1, mp("192.168.1.0/24"))
 | |
| 				}()
 | |
| 				go func() {
 | |
| 					defer wg.Done()
 | |
| 					change2 = pr.SetRoutes(2, mp("192.168.2.0/24"))
 | |
| 				}()
 | |
| 				wg.Wait()
 | |
| 
 | |
| 				return change1 || change2
 | |
| 			},
 | |
| 			expectedRoutes: map[types.NodeID]set.Set[netip.Prefix]{
 | |
| 				1: {
 | |
| 					mp("192.168.1.0/24"): {},
 | |
| 				},
 | |
| 				2: {
 | |
| 					mp("192.168.2.0/24"): {},
 | |
| 				},
 | |
| 			},
 | |
| 			expectedPrimaries: map[netip.Prefix]types.NodeID{
 | |
| 				mp("192.168.1.0/24"): 1,
 | |
| 				mp("192.168.2.0/24"): 2,
 | |
| 			},
 | |
| 			expectedIsPrimary: map[types.NodeID]bool{
 | |
| 				1: true,
 | |
| 				2: true,
 | |
| 			},
 | |
| 			expectedChange: true,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, tt := range tests {
 | |
| 		t.Run(tt.name, func(t *testing.T) {
 | |
| 			pr := New()
 | |
| 			change := tt.operations(pr)
 | |
| 			if change != tt.expectedChange {
 | |
| 				t.Errorf("change = %v, want %v", change, tt.expectedChange)
 | |
| 			}
 | |
| 			comps := append(util.Comparers, cmpopts.EquateEmpty())
 | |
| 			if diff := cmp.Diff(tt.expectedRoutes, pr.routes, comps...); diff != "" {
 | |
| 				t.Errorf("routes mismatch (-want +got):\n%s", diff)
 | |
| 			}
 | |
| 			if diff := cmp.Diff(tt.expectedPrimaries, pr.primaries, comps...); diff != "" {
 | |
| 				t.Errorf("primaries mismatch (-want +got):\n%s", diff)
 | |
| 			}
 | |
| 			if diff := cmp.Diff(tt.expectedIsPrimary, pr.isPrimary, comps...); diff != "" {
 | |
| 				t.Errorf("isPrimary mismatch (-want +got):\n%s", diff)
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 |