mirror of
				https://github.com/coredns/coredns.git
				synced 2025-11-04 03:03:14 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			239 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			239 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
 package kubernetes
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"net"
 | 
						|
	"strconv"
 | 
						|
	"testing"
 | 
						|
 | 
						|
	"github.com/coredns/coredns/plugin/kubernetes/object"
 | 
						|
	"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 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."},
 | 
						|
	}
 | 
						|
	ctx := context.Background()
 | 
						|
	controller := newdnsController(ctx, 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
 | 
						|
	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",
 | 
						|
		},
 | 
						|
	}
 | 
						|
	ctx := context.TODO()
 | 
						|
	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)
 | 
						|
		client.CoreV1().Endpoints("testns").Create(ctx, ep, meta.CreateOptions{})
 | 
						|
		count++
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
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++
 | 
						|
		}
 | 
						|
	case "headless":
 | 
						|
		for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); inc(ip) {
 | 
						|
			createHeadlessSvc(count, client, ip)
 | 
						|
			count++
 | 
						|
		}
 | 
						|
	case "external":
 | 
						|
		for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); inc(ip) {
 | 
						|
			createExternalSvc(count, client, ip)
 | 
						|
			count++
 | 
						|
		}
 | 
						|
	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++
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func createClusterIPSvc(suffix int, client kubernetes.Interface, ip net.IP) {
 | 
						|
	ctx := context.TODO()
 | 
						|
	client.CoreV1().Services("testns").Create(ctx, &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,
 | 
						|
			}},
 | 
						|
		},
 | 
						|
	}, meta.CreateOptions{})
 | 
						|
}
 | 
						|
 | 
						|
func createHeadlessSvc(suffix int, client kubernetes.Interface, ip net.IP) {
 | 
						|
	ctx := context.TODO()
 | 
						|
	client.CoreV1().Services("testns").Create(ctx, &api.Service{
 | 
						|
		ObjectMeta: meta.ObjectMeta{
 | 
						|
			Name:      "hdls" + strconv.Itoa(suffix),
 | 
						|
			Namespace: "testns",
 | 
						|
		},
 | 
						|
		Spec: api.ServiceSpec{
 | 
						|
			ClusterIP: api.ClusterIPNone,
 | 
						|
		},
 | 
						|
	}, meta.CreateOptions{})
 | 
						|
}
 | 
						|
 | 
						|
func createExternalSvc(suffix int, client kubernetes.Interface, ip net.IP) {
 | 
						|
	ctx := context.TODO()
 | 
						|
	client.CoreV1().Services("testns").Create(ctx, &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,
 | 
						|
		},
 | 
						|
	}, meta.CreateOptions{})
 | 
						|
}
 | 
						|
 | 
						|
func TestServiceModified(t *testing.T) {
 | 
						|
	var tests = []struct {
 | 
						|
		oldSvc   interface{}
 | 
						|
		newSvc   interface{}
 | 
						|
		ichanged bool
 | 
						|
		echanged bool
 | 
						|
	}{
 | 
						|
		{
 | 
						|
			oldSvc:   nil,
 | 
						|
			newSvc:   &object.Service{},
 | 
						|
			ichanged: true,
 | 
						|
			echanged: false,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			oldSvc:   &object.Service{},
 | 
						|
			newSvc:   nil,
 | 
						|
			ichanged: true,
 | 
						|
			echanged: false,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			oldSvc:   nil,
 | 
						|
			newSvc:   &object.Service{ExternalIPs: []string{"10.0.0.1"}},
 | 
						|
			ichanged: true,
 | 
						|
			echanged: true,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			oldSvc:   &object.Service{ExternalIPs: []string{"10.0.0.1"}},
 | 
						|
			newSvc:   nil,
 | 
						|
			ichanged: true,
 | 
						|
			echanged: true,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			oldSvc:   &object.Service{ExternalIPs: []string{"10.0.0.1"}},
 | 
						|
			newSvc:   &object.Service{ExternalIPs: []string{"10.0.0.2"}},
 | 
						|
			ichanged: false,
 | 
						|
			echanged: true,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			oldSvc:   &object.Service{ExternalName: "10.0.0.1"},
 | 
						|
			newSvc:   &object.Service{ExternalName: "10.0.0.2"},
 | 
						|
			ichanged: true,
 | 
						|
			echanged: false,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			oldSvc:   &object.Service{Ports: []api.ServicePort{{Name: "test1"}}},
 | 
						|
			newSvc:   &object.Service{Ports: []api.ServicePort{{Name: "test2"}}},
 | 
						|
			ichanged: true,
 | 
						|
			echanged: true,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			oldSvc:   &object.Service{Ports: []api.ServicePort{{Name: "test1"}}},
 | 
						|
			newSvc:   &object.Service{Ports: []api.ServicePort{{Name: "test2"}, {Name: "test3"}}},
 | 
						|
			ichanged: true,
 | 
						|
			echanged: true,
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	for i, test := range tests {
 | 
						|
		ichanged, echanged := serviceModified(test.oldSvc, test.newSvc)
 | 
						|
		if test.ichanged != ichanged || test.echanged != echanged {
 | 
						|
			t.Errorf("Expected %v, %v for test %v. Got %v, %v", test.ichanged, test.echanged, i, ichanged, echanged)
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 |