| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | package kubernetes
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import (
 | 
					
						
							|  |  |  | 	"net"
 | 
					
						
							|  |  |  | 	"strings"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/middleware/etcd/msg"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/miekg/dns"
 | 
					
						
							|  |  |  | 	"k8s.io/client-go/1.5/pkg/api"
 | 
					
						
							|  |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-02 23:54:33 -04:00
										 |  |  | const defaultNSName = "ns.dns."
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | var corednsRecord dns.A
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-02 02:25:33 -04:00
										 |  |  | type interfaceAddrser interface {
 | 
					
						
							|  |  |  | 	interfaceAddrs() ([]net.Addr, error)
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-02 02:25:33 -04:00
										 |  |  | type interfaceAddrs struct{}
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-02 02:25:33 -04:00
										 |  |  | func (i interfaceAddrs) interfaceAddrs() ([]net.Addr, error) {
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 	return net.InterfaceAddrs()
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // DefaultNSMsg returns an msg.Service representing an A record for
 | 
					
						
							|  |  |  | // ns.dns.[zone] -> dns service ip. This A record is needed to legitimize
 | 
					
						
							|  |  |  | // the SOA response in middleware.NS(), which is hardcoded at ns.dns.[zone].
 | 
					
						
							|  |  |  | func (k *Kubernetes) defaultNSMsg(r recordRequest) msg.Service {
 | 
					
						
							| 
									
										
										
										
											2017-06-02 23:54:33 -04:00
										 |  |  | 	ns := k.coreDNSRecord()
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 	s := msg.Service{
 | 
					
						
							| 
									
										
										
										
											2017-06-02 23:54:33 -04:00
										 |  |  | 		Key:  msg.Path(strings.Join([]string{defaultNSName, r.zone}, "."), "coredns"),
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 		Host: ns.A.String(),
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return s
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func isDefaultNS(name string, r recordRequest) bool {
 | 
					
						
							| 
									
										
										
										
											2017-06-02 23:54:33 -04:00
										 |  |  | 	return strings.Index(name, defaultNSName) == 0 && strings.Index(name, r.zone) == len(defaultNSName)
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-02 23:54:33 -04:00
										 |  |  | func (k *Kubernetes) coreDNSRecord() dns.A {
 | 
					
						
							| 
									
										
										
										
											2017-08-03 23:14:11 -07:00
										 |  |  | 	var (
 | 
					
						
							|  |  |  | 		svcName      string
 | 
					
						
							|  |  |  | 		svcNamespace string
 | 
					
						
							|  |  |  | 		dnsIP        net.IP
 | 
					
						
							|  |  |  | 	)
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if len(corednsRecord.Hdr.Name) == 0 || corednsRecord.A == nil {
 | 
					
						
							|  |  |  | 		// get local Pod IP
 | 
					
						
							| 
									
										
										
										
											2017-08-03 23:14:11 -07:00
										 |  |  | 		localIP := k.interfaceAddrsFunc()
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 		// Find endpoint matching IP to get service and namespace
 | 
					
						
							|  |  |  | 		endpointsList := k.APIConn.EndpointsList()
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	FindEndpoint:
 | 
					
						
							|  |  |  | 		for _, ep := range endpointsList.Items {
 | 
					
						
							|  |  |  | 			for _, eps := range ep.Subsets {
 | 
					
						
							|  |  |  | 				for _, addr := range eps.Addresses {
 | 
					
						
							|  |  |  | 					if localIP.Equal(net.ParseIP(addr.IP)) {
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						svcNamespace = ep.ObjectMeta.Namespace
 | 
					
						
							|  |  |  | 						svcName = ep.ObjectMeta.Name
 | 
					
						
							|  |  |  | 						break FindEndpoint
 | 
					
						
							|  |  |  | 					}
 | 
					
						
							|  |  |  | 				}
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if len(svcName) == 0 {
 | 
					
						
							| 
									
										
										
										
											2017-06-02 23:54:33 -04:00
										 |  |  | 			corednsRecord.Hdr.Name = defaultNSName
 | 
					
						
							| 
									
										
										
										
											2017-05-22 16:05:48 -04:00
										 |  |  | 			corednsRecord.A = localIP
 | 
					
						
							|  |  |  | 			return corednsRecord
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		// Find service to get ClusterIP
 | 
					
						
							|  |  |  | 		serviceList := k.APIConn.ServiceList()
 | 
					
						
							|  |  |  | 	FindService:
 | 
					
						
							|  |  |  | 		for _, svc := range serviceList {
 | 
					
						
							|  |  |  | 			if svcName == svc.Name && svcNamespace == svc.Namespace {
 | 
					
						
							|  |  |  | 				if svc.Spec.ClusterIP == api.ClusterIPNone {
 | 
					
						
							|  |  |  | 					dnsIP = localIP
 | 
					
						
							|  |  |  | 				} else {
 | 
					
						
							|  |  |  | 					dnsIP = net.ParseIP(svc.Spec.ClusterIP)
 | 
					
						
							|  |  |  | 				}
 | 
					
						
							|  |  |  | 				break FindService
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		if dnsIP == nil {
 | 
					
						
							|  |  |  | 			dnsIP = localIP
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		corednsRecord.Hdr.Name = strings.Join([]string{svcName, svcNamespace, "svc."}, ".")
 | 
					
						
							|  |  |  | 		corednsRecord.A = dnsIP
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return corednsRecord
 | 
					
						
							|  |  |  | }
 |