mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-31 05:40:42 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			269 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			269 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| Copyright 2016 The Kubernetes Authors.
 | |
| 
 | |
| Licensed under the Apache License, Version 2.0 (the "License");
 | |
| you may not use this file except in compliance with the License.
 | |
| You may obtain a copy of the License at
 | |
| 
 | |
|     http://www.apache.org/licenses/LICENSE-2.0
 | |
| 
 | |
| Unless required by applicable law or agreed to in writing, software
 | |
| distributed under the License is distributed on an "AS IS" BASIS,
 | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| See the License for the specific language governing permissions and
 | |
| limitations under the License.
 | |
| */
 | |
| 
 | |
| package node
 | |
| 
 | |
| import (
 | |
| 	"net"
 | |
| 	"reflect"
 | |
| 	"testing"
 | |
| 
 | |
| 	v1 "k8s.io/api/core/v1"
 | |
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	utilfeature "k8s.io/apiserver/pkg/util/feature"
 | |
| 	featuregatetesting "k8s.io/component-base/featuregate/testing"
 | |
| 	"k8s.io/kubernetes/pkg/features"
 | |
| )
 | |
| 
 | |
| func TestGetPreferredAddress(t *testing.T) {
 | |
| 	testcases := map[string]struct {
 | |
| 		Labels      map[string]string
 | |
| 		Addresses   []v1.NodeAddress
 | |
| 		Preferences []v1.NodeAddressType
 | |
| 
 | |
| 		ExpectErr     string
 | |
| 		ExpectAddress string
 | |
| 	}{
 | |
| 		"no addresses": {
 | |
| 			ExpectErr: "no preferred addresses found; known addresses: []",
 | |
| 		},
 | |
| 		"missing address": {
 | |
| 			Addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 			},
 | |
| 			Preferences: []v1.NodeAddressType{v1.NodeHostName},
 | |
| 			ExpectErr:   "no preferred addresses found; known addresses: [{InternalIP 1.2.3.4}]",
 | |
| 		},
 | |
| 		"found address": {
 | |
| 			Addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "1.2.3.5"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "1.2.3.7"},
 | |
| 			},
 | |
| 			Preferences:   []v1.NodeAddressType{v1.NodeHostName, v1.NodeExternalIP},
 | |
| 			ExpectAddress: "1.2.3.5",
 | |
| 		},
 | |
| 		"found hostname address": {
 | |
| 			Labels: map[string]string{v1.LabelHostname: "label-hostname"},
 | |
| 			Addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeExternalIP, Address: "1.2.3.5"},
 | |
| 				{Type: v1.NodeHostName, Address: "status-hostname"},
 | |
| 			},
 | |
| 			Preferences:   []v1.NodeAddressType{v1.NodeHostName, v1.NodeExternalIP},
 | |
| 			ExpectAddress: "status-hostname",
 | |
| 		},
 | |
| 		"label address ignored": {
 | |
| 			Labels: map[string]string{v1.LabelHostname: "label-hostname"},
 | |
| 			Addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeExternalIP, Address: "1.2.3.5"},
 | |
| 			},
 | |
| 			Preferences:   []v1.NodeAddressType{v1.NodeHostName, v1.NodeExternalIP},
 | |
| 			ExpectAddress: "1.2.3.5",
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for k, tc := range testcases {
 | |
| 		node := &v1.Node{
 | |
| 			ObjectMeta: metav1.ObjectMeta{Labels: tc.Labels},
 | |
| 			Status:     v1.NodeStatus{Addresses: tc.Addresses},
 | |
| 		}
 | |
| 		address, err := GetPreferredNodeAddress(node, tc.Preferences)
 | |
| 		errString := ""
 | |
| 		if err != nil {
 | |
| 			errString = err.Error()
 | |
| 		}
 | |
| 		if errString != tc.ExpectErr {
 | |
| 			t.Errorf("%s: expected err=%q, got %q", k, tc.ExpectErr, errString)
 | |
| 		}
 | |
| 		if address != tc.ExpectAddress {
 | |
| 			t.Errorf("%s: expected address=%q, got %q", k, tc.ExpectAddress, address)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestGetNodeHostIPs(t *testing.T) {
 | |
| 	testcases := []struct {
 | |
| 		name      string
 | |
| 		addresses []v1.NodeAddress
 | |
| 		dualStack bool
 | |
| 
 | |
| 		expectIPs []net.IP
 | |
| 	}{
 | |
| 		{
 | |
| 			name:      "no addresses",
 | |
| 			expectIPs: nil,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "no InternalIP/ExternalIP",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeHostName, Address: "example.com"},
 | |
| 			},
 | |
| 			expectIPs: nil,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "IPv4-only, simple",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 			},
 | |
| 			expectIPs: []net.IP{net.ParseIP("1.2.3.4")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "IPv4-only, external-first",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 			},
 | |
| 			expectIPs: []net.IP{net.ParseIP("1.2.3.4")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "IPv4-only, no internal",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 			},
 | |
| 			expectIPs: []net.IP{net.ParseIP("4.3.2.1")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "dual-stack node, single-stack cluster",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 				{Type: v1.NodeInternalIP, Address: "a:b::c:d"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "d:c::b:a"},
 | |
| 			},
 | |
| 			expectIPs: []net.IP{net.ParseIP("1.2.3.4")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "dual-stack node, dual-stack cluster",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 				{Type: v1.NodeInternalIP, Address: "a:b::c:d"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "d:c::b:a"},
 | |
| 			},
 | |
| 			dualStack: true,
 | |
| 			expectIPs: []net.IP{net.ParseIP("1.2.3.4"), net.ParseIP("a:b::c:d")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "dual-stack node, different order, single-stack cluster",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 				{Type: v1.NodeInternalIP, Address: "a:b::c:d"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "d:c::b:a"},
 | |
| 			},
 | |
| 			expectIPs: []net.IP{net.ParseIP("1.2.3.4")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "dual-stack node, different order, dual-stack cluster",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "1.2.3.4"},
 | |
| 				{Type: v1.NodeInternalIP, Address: "a:b::c:d"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "d:c::b:a"},
 | |
| 			},
 | |
| 			dualStack: true,
 | |
| 			expectIPs: []net.IP{net.ParseIP("1.2.3.4"), net.ParseIP("a:b::c:d")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "dual-stack node, IPv6-first, no internal IPv4, single-stack cluster",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "a:b::c:d"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "d:c::b:a"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 			},
 | |
| 			expectIPs: []net.IP{net.ParseIP("a:b::c:d")},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "dual-stack node, IPv6-first, no internal IPv4, dual-stack cluster",
 | |
| 			addresses: []v1.NodeAddress{
 | |
| 				{Type: v1.NodeInternalIP, Address: "a:b::c:d"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "d:c::b:a"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.1"},
 | |
| 				{Type: v1.NodeExternalIP, Address: "4.3.2.2"},
 | |
| 			},
 | |
| 			dualStack: true,
 | |
| 			expectIPs: []net.IP{net.ParseIP("a:b::c:d"), net.ParseIP("4.3.2.1")},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, tc := range testcases {
 | |
| 		t.Run(tc.name, func(t *testing.T) {
 | |
| 			defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.IPv6DualStack, tc.dualStack)()
 | |
| 			node := &v1.Node{
 | |
| 				Status: v1.NodeStatus{Addresses: tc.addresses},
 | |
| 			}
 | |
| 			nodeIPs, err := GetNodeHostIPs(node)
 | |
| 			nodeIP, err2 := GetNodeHostIP(node)
 | |
| 
 | |
| 			if (err == nil && err2 != nil) || (err != nil && err2 == nil) {
 | |
| 				t.Errorf("GetNodeHostIPs() returned error=%q but GetNodeHostIP() returned error=%q", err, err2)
 | |
| 			}
 | |
| 			if err != nil {
 | |
| 				if tc.expectIPs != nil {
 | |
| 					t.Errorf("expected %v, got error (%v)", tc.expectIPs, err)
 | |
| 				}
 | |
| 			} else if tc.expectIPs == nil {
 | |
| 				t.Errorf("expected error, got %v", nodeIPs)
 | |
| 			} else if !reflect.DeepEqual(nodeIPs, tc.expectIPs) {
 | |
| 				t.Errorf("expected %v, got %v", tc.expectIPs, nodeIPs)
 | |
| 			} else if !nodeIP.Equal(nodeIPs[0]) {
 | |
| 				t.Errorf("GetNodeHostIP did not return same primary (%s) as GetNodeHostIPs (%s)", nodeIP.String(), nodeIPs[0].String())
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestGetHostname(t *testing.T) {
 | |
| 	testCases := []struct {
 | |
| 		hostName         string
 | |
| 		expectedHostName string
 | |
| 		expectError      bool
 | |
| 	}{
 | |
| 		{
 | |
| 			hostName:    "   ",
 | |
| 			expectError: true,
 | |
| 		},
 | |
| 		{
 | |
| 			hostName:         " abc  ",
 | |
| 			expectedHostName: "abc",
 | |
| 			expectError:      false,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for idx, test := range testCases {
 | |
| 		hostName, err := GetHostname(test.hostName)
 | |
| 		if err != nil && !test.expectError {
 | |
| 			t.Errorf("[%d]: unexpected error: %s", idx, err)
 | |
| 		}
 | |
| 		if err == nil && test.expectError {
 | |
| 			t.Errorf("[%d]: expected error, got none", idx)
 | |
| 		}
 | |
| 		if test.expectedHostName != hostName {
 | |
| 			t.Errorf("[%d]: expected output %q, got %q", idx, test.expectedHostName, hostName)
 | |
| 		}
 | |
| 
 | |
| 	}
 | |
| }
 |