2019-05-22 20:33:19 -07:00
|
|
|
/*
|
|
|
|
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"
|
|
|
|
)
|
|
|
|
|
2019-08-02 11:20:56 -07:00
|
|
|
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 lengthes
|
|
|
|
if len(cidrs) != len(tc.cidrs) {
|
|
|
|
t.Errorf("cidrs should be of the same lengthes %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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-22 20:33:19 -07:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-02 11:20:56 -07:00
|
|
|
func TestIsIPv6CIDRString(t *testing.T) {
|
2019-05-22 20:33:19 -07:00
|
|
|
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)
|
2019-08-02 11:20:56 -07:00
|
|
|
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)
|
2019-05-22 20:33:19 -07:00
|
|
|
if res != tc.expectResult {
|
|
|
|
t.Errorf("%v: want IsIPv6CIDR=%v, got %v", tc.desc, tc.expectResult, res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|