Skip to content

Commit f3fa8fc

Browse files
DNS services are not resolving properly
The change on Sept 15th changed how services resolved in the absence of search paths, which resulted in very long times to resolve DNS in some cases. Change the SRV record style to match upstream Kube (hash of pod ip) Add support for the "pod" range <IP>.namespace.pod.cluster.local resolves to <IP>. Update tests.
1 parent 0e08c78 commit f3fa8fc

File tree

3 files changed

+164
-100
lines changed

3 files changed

+164
-100
lines changed

pkg/dns/server.go

+5-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@ package dns
33
import (
44
client "k8s.io/kubernetes/pkg/client/unversioned"
55

6+
"github.com/golang/glog"
7+
68
"github.com/coreos/go-etcd/etcd"
79
"github.com/prometheus/client_golang/prometheus"
810
backendetcd "github.com/skynetservices/skydns/backends/etcd"
@@ -12,8 +14,9 @@ import (
1214
// NewServerDefaults returns the default SkyDNS server configuration for a DNS server.
1315
func NewServerDefaults() (*server.Config, error) {
1416
config := &server.Config{
15-
Domain: "cluster.local.",
16-
Local: "openshift.default.svc.cluster.local.",
17+
Domain: "cluster.local.",
18+
Local: "openshift.default.svc.cluster.local.",
19+
Verbose: bool(glog.V(4)),
1720
}
1821
return config, server.SetDefaults(config)
1922
}

pkg/dns/serviceresolver.go

+124-92
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,12 @@ package dns
22

33
import (
44
"fmt"
5+
"hash/fnv"
6+
"net"
57
"strings"
68

9+
"github.com/golang/glog"
10+
711
kapi "k8s.io/kubernetes/pkg/api"
812
"k8s.io/kubernetes/pkg/api/errors"
913
kclient "k8s.io/kubernetes/pkg/client/unversioned"
@@ -60,20 +64,42 @@ func NewServiceResolver(config *server.Config, accessor ServiceAccessor, endpoin
6064
// * endpoint_id is "portal" when portalIP is set
6165
// * endpoints always returns each individual endpoint as A records
6266
//
63-
func (b *ServiceResolver) Records(name string, exact bool) ([]msg.Service, error) {
64-
if !strings.HasSuffix(name, b.base) {
67+
func (b *ServiceResolver) Records(dnsName string, exact bool) ([]msg.Service, error) {
68+
if !strings.HasSuffix(dnsName, b.base) {
6569
return nil, nil
6670
}
67-
prefix := strings.Trim(strings.TrimSuffix(name, b.base), ".")
71+
prefix := strings.Trim(strings.TrimSuffix(dnsName, b.base), ".")
6872
segments := strings.Split(prefix, ".")
6973
for i, j := 0, len(segments)-1; i < j; i, j = i+1, j-1 {
7074
segments[i], segments[j] = segments[j], segments[i]
7175
}
7276
if len(segments) == 0 {
7377
return nil, nil
7478
}
79+
glog.V(4).Infof("Answering query %s:%t", dnsName, exact)
80+
switch base := segments[0]; base {
81+
case "pod":
82+
if len(segments) != 3 {
83+
return nil, nil
84+
}
85+
namespace, encodedIP := segments[1], segments[2]
86+
ip := reverseIP(encodedIP)
87+
if net.ParseIP(ip) == nil {
88+
return nil, nil
89+
}
90+
return []msg.Service{
91+
{
92+
Host: ip,
93+
Port: 0,
94+
95+
Priority: 10,
96+
Weight: 10,
97+
Ttl: 30,
98+
99+
Key: msg.Path(buildDNSName(b.base, "pod", namespace, getHash(ip))),
100+
},
101+
}, nil
75102

76-
switch segments[0] {
77103
case "svc", "endpoints":
78104
if len(segments) < 3 {
79105
return nil, nil
@@ -94,56 +120,64 @@ func (b *ServiceResolver) Records(name string, exact bool) ([]msg.Service, error
94120
return nil, nil
95121
}
96122

97-
retrieveEndpoints := segments[0] == "endpoints" || (len(segments) > 3 && segments[3] == "_endpoints")
123+
subdomain := buildDNSName(b.base, base, namespace, name)
124+
endpointPrefix := base == "endpoints"
125+
retrieveEndpoints := endpointPrefix || (len(segments) > 3 && segments[3] == "_endpoints")
98126

99127
// if has a portal IP and looking at svc
100128
if svc.Spec.ClusterIP != kapi.ClusterIPNone && !retrieveEndpoints {
101-
if len(svc.Spec.Ports) == 0 {
102-
return nil, nil
129+
defaultService := msg.Service{
130+
Host: svc.Spec.ClusterIP,
131+
Port: 0,
132+
133+
Priority: 10,
134+
Weight: 10,
135+
Ttl: 30,
103136
}
104-
services := []msg.Service{}
105-
for _, p := range svc.Spec.Ports {
106-
port := p.Port
107-
if port == 0 {
108-
port = p.TargetPort.IntVal
109-
}
110-
if port == 0 {
111-
continue
112-
}
113-
if len(p.Protocol) == 0 {
114-
p.Protocol = kapi.ProtocolTCP
115-
}
116-
portName := p.Name
117-
if len(portName) == 0 {
118-
portName = fmt.Sprintf("unknown-port-%d", port)
119-
}
120-
srvName := fmt.Sprintf("%s.portal.%s", portName, name)
121-
keyName := fmt.Sprintf("_%s._%s.%s", portName, p.Protocol, name)
122-
services = append(services,
123-
msg.Service{
124-
Host: svc.Spec.ClusterIP,
125-
Port: port,
137+
defaultHash := getHash(defaultService.Host)
138+
defaultName := buildDNSName(subdomain, defaultHash)
139+
defaultService.Key = msg.Path(defaultName)
126140

127-
Priority: 10,
128-
Weight: 10,
129-
Ttl: 30,
141+
if len(svc.Spec.Ports) == 0 {
142+
return []msg.Service{defaultService}, nil
143+
}
130144

131-
Text: "",
132-
Key: msg.Path(srvName),
133-
},
134-
msg.Service{
135-
Host: srvName,
136-
Port: port,
145+
services := []msg.Service{}
146+
if len(segments) == 3 {
147+
for _, p := range svc.Spec.Ports {
148+
port := p.Port
149+
if port == 0 {
150+
port = p.TargetPort.IntVal
151+
}
152+
if port == 0 {
153+
continue
154+
}
155+
if len(p.Protocol) == 0 {
156+
p.Protocol = kapi.ProtocolTCP
157+
}
158+
portName := p.Name
159+
if len(portName) == 0 {
160+
portName = fmt.Sprintf("unknown-port-%d", port)
161+
}
162+
keyName := buildDNSName(subdomain, "_"+strings.ToLower(string(p.Protocol)), "_"+portName)
163+
services = append(services,
164+
msg.Service{
165+
Host: svc.Spec.ClusterIP,
166+
Port: port,
137167

138-
Priority: 10,
139-
Weight: 10,
140-
Ttl: 30,
168+
Priority: 10,
169+
Weight: 10,
170+
Ttl: 30,
141171

142-
Text: "",
143-
Key: msg.Path(keyName),
144-
},
145-
)
172+
Key: msg.Path(keyName),
173+
},
174+
)
175+
}
176+
}
177+
if len(services) == 0 {
178+
services = append(services, defaultService)
146179
}
180+
glog.V(4).Infof("Answered %s:%t with %#v", dnsName, exact, services)
147181
return services, nil
148182
}
149183

@@ -152,79 +186,53 @@ func (b *ServiceResolver) Records(name string, exact bool) ([]msg.Service, error
152186
if err != nil {
153187
return nil, err
154188
}
155-
targets := make(map[string]int)
189+
156190
services := make([]msg.Service, 0, len(endpoints.Subsets)*4)
157-
count := 1
158191
for _, s := range endpoints.Subsets {
159192
for _, a := range s.Addresses {
160-
shortName := ""
161-
if a.TargetRef != nil {
162-
name := fmt.Sprintf("%s-%s", a.TargetRef.Name, a.TargetRef.Namespace)
163-
if c, ok := targets[name]; ok {
164-
shortName = fmt.Sprintf("e%d", c)
165-
} else {
166-
shortName = fmt.Sprintf("e%d", count)
167-
targets[name] = count
168-
count++
169-
}
170-
} else {
171-
shortName = fmt.Sprintf("e%d", count)
172-
count++
193+
defaultService := msg.Service{
194+
Host: a.IP,
195+
Port: 0,
196+
197+
Priority: 10,
198+
Weight: 10,
199+
Ttl: 30,
173200
}
174-
hadPort := false
201+
defaultHash := getHash(defaultService.Host)
202+
defaultName := buildDNSName(subdomain, defaultHash)
203+
defaultService.Key = msg.Path(defaultName)
204+
175205
for _, p := range s.Ports {
176206
port := p.Port
177207
if port == 0 {
178208
continue
179209
}
180-
hadPort = true
181210
if len(p.Protocol) == 0 {
182211
p.Protocol = kapi.ProtocolTCP
183212
}
184213
portName := p.Name
185214
if len(portName) == 0 {
186215
portName = fmt.Sprintf("unknown-port-%d", port)
187216
}
188-
srvName := fmt.Sprintf("%s.%s.%s", portName, shortName, name)
189-
services = append(services, msg.Service{
190-
Host: a.IP,
191-
Port: port,
192217

193-
Priority: 10,
194-
Weight: 10,
195-
Ttl: 30,
196-
197-
Text: "",
198-
Key: msg.Path(srvName),
199-
})
200-
keyName := fmt.Sprintf("_%s._%s.%s", portName, p.Protocol, name)
218+
keyName := buildDNSName(subdomain, "_"+strings.ToLower(string(p.Protocol)), "_"+portName, defaultHash)
201219
services = append(services, msg.Service{
202-
Host: srvName,
220+
Host: a.IP,
203221
Port: port,
204222

205223
Priority: 10,
206224
Weight: 10,
207225
Ttl: 30,
208226

209-
Text: "",
210-
Key: msg.Path(keyName),
227+
Key: msg.Path(keyName),
211228
})
212229
}
213-
214-
if !hadPort {
215-
services = append(services, msg.Service{
216-
Host: a.IP,
217-
218-
Priority: 10,
219-
Weight: 10,
220-
Ttl: 30,
221-
222-
Text: "",
223-
Key: msg.Path(name),
224-
})
230+
if len(services) == 0 {
231+
services = append(services, defaultService)
225232
}
226233
}
227234
}
235+
glog.V(4).Infof("Answered %s:%t with %#v", dnsName, exact, services)
228236
return services, nil
229237
}
230238
return nil, nil
@@ -246,16 +254,16 @@ func (b *ServiceResolver) ReverseRecord(name string) (*msg.Service, error) {
246254
if len(svc.Spec.Ports) > 0 {
247255
port = svc.Spec.Ports[0].Port
248256
}
257+
hostName := buildDNSName(b.base, "svc", svc.Namespace, svc.Name)
249258
return &msg.Service{
250-
Host: fmt.Sprintf("%s.%s.svc.%s", svc.Name, svc.Namespace, b.base),
259+
Host: hostName,
251260
Port: port,
252261

253262
Priority: 10,
254263
Weight: 10,
255264
Ttl: 30,
256265

257-
Text: "",
258-
Key: msg.Path(name),
266+
Key: msg.Path(name),
259267
}, nil
260268
}
261269

@@ -278,3 +286,27 @@ func extractIP(reverseName string) (string, bool) {
278286
}
279287
return strings.Join(segments, "."), true
280288
}
289+
290+
// buildDNSName reverses the labels order and joins them with dots.
291+
func buildDNSName(labels ...string) string {
292+
var res string
293+
for _, label := range labels {
294+
if len(res) == 0 {
295+
res = label
296+
} else {
297+
res = fmt.Sprintf("%s.%s", label, res)
298+
}
299+
}
300+
return res
301+
}
302+
303+
// return a hash for the key name
304+
func getHash(text string) string {
305+
h := fnv.New32a()
306+
h.Write([]byte(text))
307+
return fmt.Sprintf("%x", h.Sum32())
308+
}
309+
310+
func reverseIP(ip string) string {
311+
return strings.Join(strings.Split(ip, "-"), ".")
312+
}

0 commit comments

Comments
 (0)