mirror of
				https://github.com/coredns/coredns.git
				synced 2025-10-31 18:23:13 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			213 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package kubernetes
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"net"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"github.com/coredns/coredns/plugin/test"
 | |
| 	"github.com/miekg/dns"
 | |
| 	api "k8s.io/api/core/v1"
 | |
| 	meta "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	"k8s.io/client-go/kubernetes"
 | |
| 	"k8s.io/client-go/kubernetes/fake"
 | |
| )
 | |
| 
 | |
| func endpointSubsets(addrs ...string) (eps []api.EndpointSubset) {
 | |
| 	for _, ap := range addrs {
 | |
| 		apa := strings.Split(ap, ":")
 | |
| 		address := apa[0]
 | |
| 		port, _ := strconv.Atoi(apa[1])
 | |
| 		eps = append(eps, api.EndpointSubset{Addresses: []api.EndpointAddress{{IP: address}}, Ports: []api.EndpointPort{{Port: int32(port)}}})
 | |
| 	}
 | |
| 	return eps
 | |
| }
 | |
| 
 | |
| func TestEndpointsSubsetDiffs(t *testing.T) {
 | |
| 	var tests = []struct {
 | |
| 		a, b, expected api.Endpoints
 | |
| 	}{
 | |
| 		{ // From a->b: Nothing changes
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.1:80", "10.0.0.2:8080")},
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.1:80", "10.0.0.2:8080")},
 | |
| 			api.Endpoints{},
 | |
| 		},
 | |
| 		{ // From a->b: Everything goes away
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.1:80", "10.0.0.2:8080")},
 | |
| 			api.Endpoints{},
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.1:80", "10.0.0.2:8080")},
 | |
| 		},
 | |
| 		{ // From a->b: Everything is new
 | |
| 			api.Endpoints{},
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.1:80", "10.0.0.2:8080")},
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.1:80", "10.0.0.2:8080")},
 | |
| 		},
 | |
| 		{ // From a->b: One goes away, one is new
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.2:8080")},
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.1:80")},
 | |
| 			api.Endpoints{Subsets: endpointSubsets("10.0.0.2:8080", "10.0.0.1:80")},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for i, te := range tests {
 | |
| 		got := endpointsSubsetDiffs(&te.a, &te.b)
 | |
| 		if !endpointsEquivalent(got, &te.expected) {
 | |
| 			t.Errorf("Expected '%v' for test %v, got '%v'.", te.expected, i, got)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func inc(ip net.IP) {
 | |
| 	for j := len(ip) - 1; j >= 0; j-- {
 | |
| 		ip[j]++
 | |
| 		if ip[j] > 0 {
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func BenchmarkController(b *testing.B) {
 | |
| 	client := fake.NewSimpleClientset()
 | |
| 	dco := dnsControlOpts{
 | |
| 		zones: []string{"cluster.local."},
 | |
| 	}
 | |
| 	controller := newdnsController(client, dco)
 | |
| 	cidr := "10.0.0.0/19"
 | |
| 
 | |
| 	// Add resources
 | |
| 	generateEndpoints(cidr, client)
 | |
| 	generateSvcs(cidr, "all", client)
 | |
| 	m := new(dns.Msg)
 | |
| 	m.SetQuestion("svc1.testns.svc.cluster.local.", dns.TypeA)
 | |
| 	k := New([]string{"cluster.local."})
 | |
| 	k.APIConn = controller
 | |
| 	ctx := context.Background()
 | |
| 	rw := &test.ResponseWriter{}
 | |
| 
 | |
| 	b.ResetTimer()
 | |
| 	for i := 0; i < b.N; i++ {
 | |
| 		k.ServeDNS(ctx, rw, m)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func generateEndpoints(cidr string, client kubernetes.Interface) {
 | |
| 	// https://groups.google.com/d/msg/golang-nuts/zlcYA4qk-94/TWRFHeXJCcYJ
 | |
| 	ip, ipnet, err := net.ParseCIDR(cidr)
 | |
| 	if err != nil {
 | |
| 		log.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	count := 1
 | |
| 	ep := &api.Endpoints{
 | |
| 		Subsets: []api.EndpointSubset{{
 | |
| 			Ports: []api.EndpointPort{
 | |
| 				{
 | |
| 					Port:     80,
 | |
| 					Protocol: "tcp",
 | |
| 					Name:     "http",
 | |
| 				},
 | |
| 			},
 | |
| 		}},
 | |
| 		ObjectMeta: meta.ObjectMeta{
 | |
| 			Namespace: "testns",
 | |
| 		},
 | |
| 	}
 | |
| 	for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); inc(ip) {
 | |
| 		ep.Subsets[0].Addresses = []api.EndpointAddress{
 | |
| 			{
 | |
| 				IP:       ip.String(),
 | |
| 				Hostname: "foo" + strconv.Itoa(count),
 | |
| 			},
 | |
| 		}
 | |
| 		ep.ObjectMeta.Name = "svc" + strconv.Itoa(count)
 | |
| 		_, err = client.Core().Endpoints("testns").Create(ep)
 | |
| 		count += 1
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func generateSvcs(cidr string, svcType string, client kubernetes.Interface) {
 | |
| 	ip, ipnet, err := net.ParseCIDR(cidr)
 | |
| 	if err != nil {
 | |
| 		log.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	count := 1
 | |
| 	switch svcType {
 | |
| 	case "clusterip":
 | |
| 		for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); inc(ip) {
 | |
| 			createClusterIPSvc(count, client, ip)
 | |
| 			count += 1
 | |
| 		}
 | |
| 	case "headless":
 | |
| 		for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); inc(ip) {
 | |
| 			createHeadlessSvc(count, client, ip)
 | |
| 			count += 1
 | |
| 		}
 | |
| 	case "external":
 | |
| 		for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); inc(ip) {
 | |
| 			createExternalSvc(count, client, ip)
 | |
| 			count += 1
 | |
| 		}
 | |
| 	default:
 | |
| 		for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); inc(ip) {
 | |
| 			if count%3 == 0 {
 | |
| 				createClusterIPSvc(count, client, ip)
 | |
| 			} else if count%3 == 1 {
 | |
| 				createHeadlessSvc(count, client, ip)
 | |
| 			} else if count%3 == 2 {
 | |
| 				createExternalSvc(count, client, ip)
 | |
| 			}
 | |
| 			count += 1
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func createClusterIPSvc(suffix int, client kubernetes.Interface, ip net.IP) {
 | |
| 	client.Core().Services("testns").Create(&api.Service{
 | |
| 		ObjectMeta: meta.ObjectMeta{
 | |
| 			Name:      "svc" + strconv.Itoa(suffix),
 | |
| 			Namespace: "testns",
 | |
| 		},
 | |
| 		Spec: api.ServiceSpec{
 | |
| 			ClusterIP: ip.String(),
 | |
| 			Ports: []api.ServicePort{{
 | |
| 				Name:     "http",
 | |
| 				Protocol: "tcp",
 | |
| 				Port:     80,
 | |
| 			}},
 | |
| 		},
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func createHeadlessSvc(suffix int, client kubernetes.Interface, ip net.IP) {
 | |
| 	client.Core().Services("testns").Create(&api.Service{
 | |
| 		ObjectMeta: meta.ObjectMeta{
 | |
| 			Name:      "hdls" + strconv.Itoa(suffix),
 | |
| 			Namespace: "testns",
 | |
| 		},
 | |
| 		Spec: api.ServiceSpec{
 | |
| 			ClusterIP: api.ClusterIPNone,
 | |
| 		},
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func createExternalSvc(suffix int, client kubernetes.Interface, ip net.IP) {
 | |
| 	client.Core().Services("testns").Create(&api.Service{
 | |
| 		ObjectMeta: meta.ObjectMeta{
 | |
| 			Name:      "external" + strconv.Itoa(suffix),
 | |
| 			Namespace: "testns",
 | |
| 		},
 | |
| 		Spec: api.ServiceSpec{
 | |
| 			ExternalName: "coredns" + strconv.Itoa(suffix) + ".io",
 | |
| 			Ports: []api.ServicePort{{
 | |
| 				Name:     "http",
 | |
| 				Protocol: "tcp",
 | |
| 				Port:     80,
 | |
| 			}},
 | |
| 			Type: api.ServiceTypeExternalName,
 | |
| 		},
 | |
| 	})
 | |
| }
 |