mirror of
				https://github.com/coredns/coredns.git
				synced 2025-10-29 01:04:15 -04:00 
			
		
		
		
	*add option for resolving headless Services without external IPs in k8s_external Signed-off-by: Tomas Kohout <tomas.kohout1995@gmail.com>
		
			
				
	
	
		
			104 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			104 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package kubernetes
 | |
| 
 | |
| import (
 | |
| 	"net"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/miekg/dns"
 | |
| )
 | |
| 
 | |
| func isDefaultNS(name, zone string) bool {
 | |
| 	return strings.Index(name, defaultNSName) == 0 && strings.Index(name, zone) == len(defaultNSName)
 | |
| }
 | |
| 
 | |
| // nsAddrs returns the A or AAAA records for the CoreDNS service in the cluster. If the service cannot be found,
 | |
| // it returns a record for the local address of the machine we're running on.
 | |
| func (k *Kubernetes) nsAddrs(external, headless bool, zone string) []dns.RR {
 | |
| 	var (
 | |
| 		svcNames      []string
 | |
| 		svcIPs        []net.IP
 | |
| 		foundEndpoint bool
 | |
| 	)
 | |
| 
 | |
| 	// Find the CoreDNS Endpoints
 | |
| 	for _, localIP := range k.localIPs {
 | |
| 		endpoints := k.APIConn.EpIndexReverse(localIP.String())
 | |
| 
 | |
| 		// Collect IPs for all Services of the Endpoints
 | |
| 		for _, endpoint := range endpoints {
 | |
| 			foundEndpoint = true
 | |
| 			svcs := k.APIConn.SvcIndex(endpoint.Index)
 | |
| 			for _, svc := range svcs {
 | |
| 				if external {
 | |
| 					svcName := strings.Join([]string{svc.Name, svc.Namespace, zone}, ".")
 | |
| 
 | |
| 					if headless && svc.Headless() {
 | |
| 						for _, s := range endpoint.Subsets {
 | |
| 							for _, a := range s.Addresses {
 | |
| 								svcNames = append(svcNames, endpointHostname(a, k.endpointNameMode)+"."+svcName)
 | |
| 								svcIPs = append(svcIPs, net.ParseIP(a.IP))
 | |
| 							}
 | |
| 						}
 | |
| 					} else {
 | |
| 						for _, exIP := range svc.ExternalIPs {
 | |
| 							svcNames = append(svcNames, svcName)
 | |
| 							svcIPs = append(svcIPs, net.ParseIP(exIP))
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					continue
 | |
| 				}
 | |
| 				svcName := strings.Join([]string{svc.Name, svc.Namespace, Svc, zone}, ".")
 | |
| 				if svc.Headless() {
 | |
| 					// For a headless service, use the endpoints IPs
 | |
| 					for _, s := range endpoint.Subsets {
 | |
| 						for _, a := range s.Addresses {
 | |
| 							svcNames = append(svcNames, endpointHostname(a, k.endpointNameMode)+"."+svcName)
 | |
| 							svcIPs = append(svcIPs, net.ParseIP(a.IP))
 | |
| 						}
 | |
| 					}
 | |
| 				} else {
 | |
| 					for _, clusterIP := range svc.ClusterIPs {
 | |
| 						svcNames = append(svcNames, svcName)
 | |
| 						svcIPs = append(svcIPs, net.ParseIP(clusterIP))
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// If no CoreDNS endpoints were found, use the localIPs directly
 | |
| 	if !foundEndpoint {
 | |
| 		svcIPs = make([]net.IP, len(k.localIPs))
 | |
| 		svcNames = make([]string, len(k.localIPs))
 | |
| 		for i, localIP := range k.localIPs {
 | |
| 			svcNames[i] = defaultNSName + zone
 | |
| 			svcIPs[i] = localIP
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Create an RR slice of collected IPs
 | |
| 	rrs := make([]dns.RR, len(svcIPs))
 | |
| 	for i, ip := range svcIPs {
 | |
| 		if ip.To4() == nil {
 | |
| 			rr := new(dns.AAAA)
 | |
| 			rr.Hdr.Class = dns.ClassINET
 | |
| 			rr.Hdr.Rrtype = dns.TypeAAAA
 | |
| 			rr.Hdr.Name = svcNames[i]
 | |
| 			rr.AAAA = ip
 | |
| 			rrs[i] = rr
 | |
| 			continue
 | |
| 		}
 | |
| 		rr := new(dns.A)
 | |
| 		rr.Hdr.Class = dns.ClassINET
 | |
| 		rr.Hdr.Rrtype = dns.TypeA
 | |
| 		rr.Hdr.Name = svcNames[i]
 | |
| 		rr.A = ip
 | |
| 		rrs[i] = rr
 | |
| 	}
 | |
| 
 | |
| 	return rrs
 | |
| }
 | |
| 
 | |
| const defaultNSName = "ns.dns."
 |