blob: 95e6b37c4bde35e1643d7f486eb79f27a51962a4 [file] [log] [blame]
/*
Copyright 2018 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 net
import (
"net"
"testing"
)
func TestParseCIDRs(t *testing.T) {
testCases := []struct {
cidrs []string
errString string
errorExpected bool
}{
{
cidrs: []string{},
errString: "should not return an error for an empty slice",
errorExpected: false,
},
{
cidrs: []string{"10.0.0.0/8", "not-a-valid-cidr", "2000::/10"},
errString: "should return error for bad cidr",
errorExpected: true,
},
{
cidrs: []string{"10.0.0.0/8", "2000::/10"},
errString: "should not return error for good cidrs",
errorExpected: false,
},
}
for _, tc := range testCases {
cidrs, err := ParseCIDRs(tc.cidrs)
if tc.errorExpected {
if err == nil {
t.Errorf("%v", tc.errString)
}
continue
}
if err != nil {
t.Errorf("%v error:%v", tc.errString, err)
}
// validate lengths
if len(cidrs) != len(tc.cidrs) {
t.Errorf("cidrs should be of the same lengths %v != %v", len(cidrs), len(tc.cidrs))
}
}
}
func TestDualStackIPs(t *testing.T) {
testCases := []struct {
ips []string
errMessage string
expectedResult bool
expectError bool
}{
{
ips: []string{"1.1.1.1"},
errMessage: "should fail because length is not at least 2",
expectedResult: false,
expectError: false,
},
{
ips: []string{},
errMessage: "should fail because length is not at least 2",
expectedResult: false,
expectError: false,
},
{
ips: []string{"1.1.1.1", "2.2.2.2", "3.3.3.3"},
errMessage: "should fail because all are v4",
expectedResult: false,
expectError: false,
},
{
ips: []string{"fd92:20ba:ca:34f7:ffff:ffff:ffff:ffff", "fd92:20ba:ca:34f7:ffff:ffff:ffff:fff0", "fd92:20ba:ca:34f7:ffff:ffff:ffff:fff1"},
errMessage: "should fail because all are v6",
expectedResult: false,
expectError: false,
},
{
ips: []string{"1.1.1.1", "not-a-valid-ip"},
errMessage: "should fail because 2nd ip is invalid",
expectedResult: false,
expectError: true,
},
{
ips: []string{"not-a-valid-ip", "fd92:20ba:ca:34f7:ffff:ffff:ffff:ffff"},
errMessage: "should fail because 1st ip is invalid",
expectedResult: false,
expectError: true,
},
{
ips: []string{"1.1.1.1", "fd92:20ba:ca:34f7:ffff:ffff:ffff:ffff"},
errMessage: "expected success, but found failure",
expectedResult: true,
expectError: false,
},
{
ips: []string{"fd92:20ba:ca:34f7:ffff:ffff:ffff:ffff", "1.1.1.1", "fd92:20ba:ca:34f7:ffff:ffff:ffff:fff0"},
errMessage: "expected success, but found failure",
expectedResult: true,
expectError: false,
},
{
ips: []string{"1.1.1.1", "fd92:20ba:ca:34f7:ffff:ffff:ffff:ffff", "10.0.0.0"},
errMessage: "expected success, but found failure",
expectedResult: true,
expectError: false,
},
{
ips: []string{"fd92:20ba:ca:34f7:ffff:ffff:ffff:ffff", "1.1.1.1"},
errMessage: "expected success, but found failure",
expectedResult: true,
expectError: false,
},
}
// for each test case, test the regular func and the string func
for _, tc := range testCases {
dualStack, err := IsDualStackIPStrings(tc.ips)
if err == nil && tc.expectError {
t.Errorf("%s", tc.errMessage)
continue
}
if err != nil && !tc.expectError {
t.Errorf("failed to run test case for %v, error: %v", tc.ips, err)
continue
}
if dualStack != tc.expectedResult {
t.Errorf("%v for %v", tc.errMessage, tc.ips)
}
}
for _, tc := range testCases {
ips := make([]net.IP, 0, len(tc.ips))
for _, ip := range tc.ips {
parsedIP := net.ParseIP(ip)
ips = append(ips, parsedIP)
}
dualStack, err := IsDualStackIPs(ips)
if err == nil && tc.expectError {
t.Errorf("%s", tc.errMessage)
continue
}
if err != nil && !tc.expectError {
t.Errorf("failed to run test case for %v, error: %v", tc.ips, err)
continue
}
if dualStack != tc.expectedResult {
t.Errorf("%v for %v", tc.errMessage, tc.ips)
}
}
}
func TestDualStackCIDRs(t *testing.T) {
testCases := []struct {
cidrs []string
errMessage string
expectedResult bool
expectError bool
}{
{
cidrs: []string{"10.10.10.10/8"},
errMessage: "should fail because length is not at least 2",
expectedResult: false,
expectError: false,
},
{
cidrs: []string{},
errMessage: "should fail because length is not at least 2",
expectedResult: false,
expectError: false,
},
{
cidrs: []string{"10.10.10.10/8", "20.20.20.20/8", "30.30.30.30/8"},
errMessage: "should fail because all cidrs are v4",
expectedResult: false,
expectError: false,
},
{
cidrs: []string{"2000::/10", "3000::/10"},
errMessage: "should fail because all cidrs are v6",
expectedResult: false,
expectError: false,
},
{
cidrs: []string{"10.10.10.10/8", "not-a-valid-cidr"},
errMessage: "should fail because 2nd cidr is invalid",
expectedResult: false,
expectError: true,
},
{
cidrs: []string{"not-a-valid-ip", "2000::/10"},
errMessage: "should fail because 1st cidr is invalid",
expectedResult: false,
expectError: true,
},
{
cidrs: []string{"10.10.10.10/8", "2000::/10"},
errMessage: "expected success, but found failure",
expectedResult: true,
expectError: false,
},
{
cidrs: []string{"2000::/10", "10.10.10.10/8"},
errMessage: "expected success, but found failure",
expectedResult: true,
expectError: false,
},
{
cidrs: []string{"2000::/10", "10.10.10.10/8", "3000::/10"},
errMessage: "expected success, but found failure",
expectedResult: true,
expectError: false,
},
}
// for each test case, test the regular func and the string func
for _, tc := range testCases {
dualStack, err := IsDualStackCIDRStrings(tc.cidrs)
if err == nil && tc.expectError {
t.Errorf("%s", tc.errMessage)
continue
}
if err != nil && !tc.expectError {
t.Errorf("failed to run test case for %v, error: %v", tc.cidrs, err)
continue
}
if dualStack != tc.expectedResult {
t.Errorf("%v for %v", tc.errMessage, tc.cidrs)
}
}
for _, tc := range testCases {
cidrs := make([]*net.IPNet, 0, len(tc.cidrs))
for _, cidr := range tc.cidrs {
_, parsedCIDR, _ := net.ParseCIDR(cidr)
cidrs = append(cidrs, parsedCIDR)
}
dualStack, err := IsDualStackCIDRs(cidrs)
if err == nil && tc.expectError {
t.Errorf("%s", tc.errMessage)
continue
}
if err != nil && !tc.expectError {
t.Errorf("failed to run test case for %v, error: %v", tc.cidrs, err)
continue
}
if dualStack != tc.expectedResult {
t.Errorf("%v for %v", tc.errMessage, tc.cidrs)
}
}
}
func TestIsIPv6String(t *testing.T) {
testCases := []struct {
ip string
expectIPv6 bool
}{
{
ip: "127.0.0.1",
expectIPv6: false,
},
{
ip: "192.168.0.0",
expectIPv6: false,
},
{
ip: "1.2.3.4",
expectIPv6: false,
},
{
ip: "bad ip",
expectIPv6: false,
},
{
ip: "::1",
expectIPv6: true,
},
{
ip: "fd00::600d:f00d",
expectIPv6: true,
},
{
ip: "2001:db8::5",
expectIPv6: true,
},
}
for i := range testCases {
isIPv6 := IsIPv6String(testCases[i].ip)
if isIPv6 != testCases[i].expectIPv6 {
t.Errorf("[%d] Expect ipv6 %v, got %v", i+1, testCases[i].expectIPv6, isIPv6)
}
}
}
func TestIsIPv6(t *testing.T) {
testCases := []struct {
ip net.IP
expectIPv6 bool
}{
{
ip: net.IPv4zero,
expectIPv6: false,
},
{
ip: net.IPv4bcast,
expectIPv6: false,
},
{
ip: net.ParseIP("127.0.0.1"),
expectIPv6: false,
},
{
ip: net.ParseIP("10.20.40.40"),
expectIPv6: false,
},
{
ip: net.ParseIP("172.17.3.0"),
expectIPv6: false,
},
{
ip: nil,
expectIPv6: false,
},
{
ip: net.IPv6loopback,
expectIPv6: true,
},
{
ip: net.IPv6zero,
expectIPv6: true,
},
{
ip: net.ParseIP("fd00::600d:f00d"),
expectIPv6: true,
},
{
ip: net.ParseIP("2001:db8::5"),
expectIPv6: true,
},
}
for i := range testCases {
isIPv6 := IsIPv6(testCases[i].ip)
if isIPv6 != testCases[i].expectIPv6 {
t.Errorf("[%d] Expect ipv6 %v, got %v", i+1, testCases[i].expectIPv6, isIPv6)
}
}
}
func TestIsIPv6CIDRString(t *testing.T) {
testCases := []struct {
desc string
cidr string
expectResult bool
}{
{
desc: "ipv4 CIDR 1",
cidr: "10.0.0.0/8",
expectResult: false,
},
{
desc: "ipv4 CIDR 2",
cidr: "192.168.0.0/16",
expectResult: false,
},
{
desc: "ipv6 CIDR 1",
cidr: "::/1",
expectResult: true,
},
{
desc: "ipv6 CIDR 2",
cidr: "2000::/10",
expectResult: true,
},
{
desc: "ipv6 CIDR 3",
cidr: "2001:db8::/32",
expectResult: true,
},
}
for _, tc := range testCases {
res := IsIPv6CIDRString(tc.cidr)
if res != tc.expectResult {
t.Errorf("%v: want IsIPv6CIDRString=%v, got %v", tc.desc, tc.expectResult, res)
}
}
}
func TestIsIPv6CIDR(t *testing.T) {
testCases := []struct {
desc string
cidr string
expectResult bool
}{
{
desc: "ipv4 CIDR 1",
cidr: "10.0.0.0/8",
expectResult: false,
},
{
desc: "ipv4 CIDR 2",
cidr: "192.168.0.0/16",
expectResult: false,
},
{
desc: "ipv6 CIDR 1",
cidr: "::/1",
expectResult: true,
},
{
desc: "ipv6 CIDR 2",
cidr: "2000::/10",
expectResult: true,
},
{
desc: "ipv6 CIDR 3",
cidr: "2001:db8::/32",
expectResult: true,
},
}
for _, tc := range testCases {
_, cidr, _ := net.ParseCIDR(tc.cidr)
res := IsIPv6CIDR(cidr)
if res != tc.expectResult {
t.Errorf("%v: want IsIPv6CIDR=%v, got %v", tc.desc, tc.expectResult, res)
}
}
}
func TestIsIPv4String(t *testing.T) {
testCases := []struct {
ip string
expectIPv4 bool
}{
{
ip: "127.0.0.1",
expectIPv4: true,
},
{
ip: "192.168.0.0",
expectIPv4: true,
},
{
ip: "1.2.3.4",
expectIPv4: true,
},
{
ip: "bad ip",
expectIPv4: false,
},
{
ip: "::1",
expectIPv4: false,
},
{
ip: "fd00::600d:f00d",
expectIPv4: false,
},
{
ip: "2001:db8::5",
expectIPv4: false,
},
}
for i := range testCases {
isIPv4 := IsIPv4String(testCases[i].ip)
if isIPv4 != testCases[i].expectIPv4 {
t.Errorf("[%d] Expect ipv4 %v, got %v", i+1, testCases[i].expectIPv4, isIPv4)
}
}
}
func TestIsIPv4(t *testing.T) {
testCases := []struct {
ip net.IP
expectIPv4 bool
}{
{
ip: net.IPv4zero,
expectIPv4: true,
},
{
ip: net.IPv4bcast,
expectIPv4: true,
},
{
ip: net.ParseIP("127.0.0.1"),
expectIPv4: true,
},
{
ip: net.ParseIP("10.20.40.40"),
expectIPv4: true,
},
{
ip: net.ParseIP("172.17.3.0"),
expectIPv4: true,
},
{
ip: nil,
expectIPv4: false,
},
{
ip: net.IPv6loopback,
expectIPv4: false,
},
{
ip: net.IPv6zero,
expectIPv4: false,
},
{
ip: net.ParseIP("fd00::600d:f00d"),
expectIPv4: false,
},
{
ip: net.ParseIP("2001:db8::5"),
expectIPv4: false,
},
}
for i := range testCases {
isIPv4 := IsIPv4(testCases[i].ip)
if isIPv4 != testCases[i].expectIPv4 {
t.Errorf("[%d] Expect ipv4 %v, got %v", i+1, testCases[i].expectIPv4, isIPv4)
}
}
}
func TestIsIPv4CIDRString(t *testing.T) {
testCases := []struct {
desc string
cidr string
expectResult bool
}{
{
desc: "ipv4 CIDR 1",
cidr: "10.0.0.0/8",
expectResult: true,
},
{
desc: "ipv4 CIDR 2",
cidr: "192.168.0.0/16",
expectResult: true,
},
{
desc: "ipv6 CIDR 1",
cidr: "::/1",
expectResult: false,
},
{
desc: "ipv6 CIDR 2",
cidr: "2000::/10",
expectResult: false,
},
{
desc: "ipv6 CIDR 3",
cidr: "2001:db8::/32",
expectResult: false,
},
}
for _, tc := range testCases {
res := IsIPv4CIDRString(tc.cidr)
if res != tc.expectResult {
t.Errorf("%v: want IsIPv4CIDRString=%v, got %v", tc.desc, tc.expectResult, res)
}
}
}
func TestIsIPv4CIDR(t *testing.T) {
testCases := []struct {
desc string
cidr string
expectResult bool
}{
{
desc: "ipv4 CIDR 1",
cidr: "10.0.0.0/8",
expectResult: true,
},
{
desc: "ipv4 CIDR 2",
cidr: "192.168.0.0/16",
expectResult: true,
},
{
desc: "ipv6 CIDR 1",
cidr: "::/1",
expectResult: false,
},
{
desc: "ipv6 CIDR 2",
cidr: "2000::/10",
expectResult: false,
},
{
desc: "ipv6 CIDR 3",
cidr: "2001:db8::/32",
expectResult: false,
},
}
for _, tc := range testCases {
_, cidr, _ := net.ParseCIDR(tc.cidr)
res := IsIPv4CIDR(cidr)
if res != tc.expectResult {
t.Errorf("%v: want IsIPv4CIDR=%v, got %v", tc.desc, tc.expectResult, res)
}
}
}
func TestParsePort(t *testing.T) {
var tests = []struct {
name string
port string
allowZero bool
expectedPort int
expectedError bool
}{
{
name: "valid port: 1",
port: "1",
expectedPort: 1,
},
{
name: "valid port: 1234",
port: "1234",
expectedPort: 1234,
},
{
name: "valid port: 65535",
port: "65535",
expectedPort: 65535,
},
{
name: "invalid port: not a number",
port: "a",
expectedError: true,
allowZero: false,
},
{
name: "invalid port: too small",
port: "0",
expectedError: true,
},
{
name: "invalid port: negative",
port: "-10",
expectedError: true,
},
{
name: "invalid port: too big",
port: "65536",
expectedError: true,
},
{
name: "zero port: allowed",
port: "0",
allowZero: true,
},
{
name: "zero port: not allowed",
port: "0",
expectedError: true,
},
}
for _, rt := range tests {
t.Run(rt.name, func(t *testing.T) {
actualPort, actualError := ParsePort(rt.port, rt.allowZero)
if actualError != nil && !rt.expectedError {
t.Errorf("%s unexpected failure: %v", rt.name, actualError)
return
}
if actualError == nil && rt.expectedError {
t.Errorf("%s passed when expected to fail", rt.name)
return
}
if actualPort != rt.expectedPort {
t.Errorf("%s returned wrong port: got %d, expected %d", rt.name, actualPort, rt.expectedPort)
}
})
}
}
func TestRangeSize(t *testing.T) {
testCases := []struct {
name string
cidr string
addrs int64
}{
{
name: "supported IPv4 cidr",
cidr: "192.168.1.0/24",
addrs: 256,
},
{
name: "unsupported IPv4 cidr",
cidr: "192.168.1.0/1",
addrs: 0,
},
{
name: "unsupported IPv6 mask",
cidr: "2001:db8::/1",
addrs: 0,
},
}
for _, tc := range testCases {
_, cidr, err := net.ParseCIDR(tc.cidr)
if err != nil {
t.Errorf("failed to parse cidr for test %s, unexpected error: '%s'", tc.name, err)
}
if size := RangeSize(cidr); size != tc.addrs {
t.Errorf("test %s failed. %s should have a range size of %d, got %d",
tc.name, tc.cidr, tc.addrs, size)
}
}
}
func TestGetIndexedIP(t *testing.T) {
testCases := []struct {
cidr string
index int
expectError bool
expectedIP string
}{
{
cidr: "192.168.1.0/24",
index: 20,
expectError: false,
expectedIP: "192.168.1.20",
},
{
cidr: "192.168.1.0/30",
index: 10,
expectError: true,
},
{
cidr: "192.168.1.0/24",
index: 255,
expectError: false,
expectedIP: "192.168.1.255",
},
{
cidr: "255.255.255.0/24",
index: 256,
expectError: true,
},
{
cidr: "fd:11:b2:be::/120",
index: 20,
expectError: false,
expectedIP: "fd:11:b2:be::14",
},
{
cidr: "fd:11:b2:be::/126",
index: 10,
expectError: true,
},
{
cidr: "fd:11:b2:be::/120",
index: 255,
expectError: false,
expectedIP: "fd:11:b2:be::ff",
},
{
cidr: "00:00:00:be::/120",
index: 255,
expectError: false,
expectedIP: "::be:0:0:0:ff",
},
}
for _, tc := range testCases {
_, subnet, err := net.ParseCIDR(tc.cidr)
if err != nil {
t.Errorf("failed to parse cidr %s, unexpected error: '%s'", tc.cidr, err)
}
ip, err := GetIndexedIP(subnet, tc.index)
if err == nil && tc.expectError || err != nil && !tc.expectError {
t.Errorf("expectedError is %v and err is %s", tc.expectError, err)
continue
}
if err == nil {
ipString := ip.String()
if ipString != tc.expectedIP {
t.Errorf("expected %s but instead got %s", tc.expectedIP, ipString)
}
}
}
}