@@ -2,8 +2,12 @@ package dns
2
2
3
3
import (
4
4
"fmt"
5
+ "hash/fnv"
6
+ "net"
5
7
"strings"
6
8
9
+ "github.com/golang/glog"
10
+
7
11
kapi "k8s.io/kubernetes/pkg/api"
8
12
"k8s.io/kubernetes/pkg/api/errors"
9
13
kclient "k8s.io/kubernetes/pkg/client/unversioned"
@@ -48,32 +52,58 @@ func NewServiceResolver(config *server.Config, accessor ServiceAccessor, endpoin
48
52
// Records implements the SkyDNS Backend interface and returns standard records for
49
53
// a name.
50
54
//
51
- // The standard pattern is <prefix>.<service_name>.<namespace>.(svc|endpoints).<base>
55
+ // The standard pattern is <prefix>.<service_name>.<namespace>.(svc|endpoints|pod ).<base>
52
56
//
53
57
// * prefix may be any series of prefix values
58
+ // * _endpoints is a special prefix that returns the same as <service_name>.<namespace>.svc.<base>
54
59
// * service_name and namespace must locate a real service
60
+ // * unless a fallback is defined, in which case the fallback name will be looked up
55
61
// * svc indicates standard service rules apply (portalIP or endpoints as A records)
56
62
// * reverse lookup of IP is only possible for portalIP
57
63
// * SRV records are returned for each host+port combination as:
58
64
// _<port_name>._<port_protocol>.<dns>
59
65
// _<port_name>.<endpoint_id>.<dns>
60
- // * endpoint_id is "portal" when portalIP is set
61
66
// * endpoints always returns each individual endpoint as A records
67
+ // * SRV records for endpoints are similar to SVC, but are prefixed with a single label
68
+ // that is a hash of the endpoint IP
69
+ // * pods is of the form <IP_with_dashes>.<namespace>.pod.<base> and resolves to <IP>
62
70
//
63
- func (b * ServiceResolver ) Records (name string , exact bool ) ([]msg.Service , error ) {
64
- if ! strings .HasSuffix (name , b .base ) {
71
+ func (b * ServiceResolver ) Records (dnsName string , exact bool ) ([]msg.Service , error ) {
72
+ if ! strings .HasSuffix (dnsName , b .base ) {
65
73
return nil , nil
66
74
}
67
- prefix := strings .Trim (strings .TrimSuffix (name , b .base ), "." )
75
+ prefix := strings .Trim (strings .TrimSuffix (dnsName , b .base ), "." )
68
76
segments := strings .Split (prefix , "." )
69
77
for i , j := 0 , len (segments )- 1 ; i < j ; i , j = i + 1 , j - 1 {
70
78
segments [i ], segments [j ] = segments [j ], segments [i ]
71
79
}
72
80
if len (segments ) == 0 {
73
81
return nil , nil
74
82
}
83
+ glog .V (4 ).Infof ("Answering query %s:%t" , dnsName , exact )
84
+ switch base := segments [0 ]; base {
85
+ case "pod" :
86
+ if len (segments ) != 3 {
87
+ return nil , nil
88
+ }
89
+ namespace , encodedIP := segments [1 ], segments [2 ]
90
+ ip := convertDashIPToIP (encodedIP )
91
+ if net .ParseIP (ip ) == nil {
92
+ return nil , nil
93
+ }
94
+ return []msg.Service {
95
+ {
96
+ Host : ip ,
97
+ Port : 0 ,
98
+
99
+ Priority : 10 ,
100
+ Weight : 10 ,
101
+ Ttl : 30 ,
102
+
103
+ Key : msg .Path (buildDNSName (b .base , "pod" , namespace , getHash (ip ))),
104
+ },
105
+ }, nil
75
106
76
- switch segments [0 ] {
77
107
case "svc" , "endpoints" :
78
108
if len (segments ) < 3 {
79
109
return nil , nil
@@ -94,56 +124,64 @@ func (b *ServiceResolver) Records(name string, exact bool) ([]msg.Service, error
94
124
return nil , nil
95
125
}
96
126
97
- retrieveEndpoints := segments [0 ] == "endpoints" || (len (segments ) > 3 && segments [3 ] == "_endpoints" )
127
+ subdomain := buildDNSName (b .base , base , namespace , name )
128
+ endpointPrefix := base == "endpoints"
129
+ retrieveEndpoints := endpointPrefix || (len (segments ) > 3 && segments [3 ] == "_endpoints" )
98
130
99
131
// if has a portal IP and looking at svc
100
132
if svc .Spec .ClusterIP != kapi .ClusterIPNone && ! retrieveEndpoints {
101
- if len (svc .Spec .Ports ) == 0 {
102
- return nil , nil
133
+ defaultService := msg.Service {
134
+ Host : svc .Spec .ClusterIP ,
135
+ Port : 0 ,
136
+
137
+ Priority : 10 ,
138
+ Weight : 10 ,
139
+ Ttl : 30 ,
103
140
}
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 ,
141
+ defaultHash := getHash (defaultService .Host )
142
+ defaultName := buildDNSName (subdomain , defaultHash )
143
+ defaultService .Key = msg .Path (defaultName )
126
144
127
- Priority : 10 ,
128
- Weight : 10 ,
129
- Ttl : 30 ,
145
+ if len ( svc . Spec . Ports ) == 0 {
146
+ return []msg. Service { defaultService }, nil
147
+ }
130
148
131
- Text : "" ,
132
- Key : msg .Path (srvName ),
133
- },
134
- msg.Service {
135
- Host : srvName ,
136
- Port : port ,
149
+ services := []msg.Service {}
150
+ if len (segments ) == 3 {
151
+ for _ , p := range svc .Spec .Ports {
152
+ port := p .Port
153
+ if port == 0 {
154
+ port = p .TargetPort .IntVal
155
+ }
156
+ if port == 0 {
157
+ continue
158
+ }
159
+ if len (p .Protocol ) == 0 {
160
+ p .Protocol = kapi .ProtocolTCP
161
+ }
162
+ portName := p .Name
163
+ if len (portName ) == 0 {
164
+ portName = fmt .Sprintf ("unknown-port-%d" , port )
165
+ }
166
+ keyName := buildDNSName (subdomain , "_" + strings .ToLower (string (p .Protocol )), "_" + portName )
167
+ services = append (services ,
168
+ msg.Service {
169
+ Host : svc .Spec .ClusterIP ,
170
+ Port : port ,
137
171
138
- Priority : 10 ,
139
- Weight : 10 ,
140
- Ttl : 30 ,
172
+ Priority : 10 ,
173
+ Weight : 10 ,
174
+ Ttl : 30 ,
141
175
142
- Text : "" ,
143
- Key : msg .Path (keyName ),
144
- },
145
- )
176
+ Key : msg .Path (keyName ),
177
+ },
178
+ )
179
+ }
180
+ }
181
+ if len (services ) == 0 {
182
+ services = append (services , defaultService )
146
183
}
184
+ glog .V (4 ).Infof ("Answered %s:%t with %#v" , dnsName , exact , services )
147
185
return services , nil
148
186
}
149
187
@@ -152,79 +190,53 @@ func (b *ServiceResolver) Records(name string, exact bool) ([]msg.Service, error
152
190
if err != nil {
153
191
return nil , err
154
192
}
155
- targets := make ( map [ string ] int )
193
+
156
194
services := make ([]msg.Service , 0 , len (endpoints .Subsets )* 4 )
157
- count := 1
158
195
for _ , s := range endpoints .Subsets {
159
196
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 ++
197
+ defaultService := msg.Service {
198
+ Host : a .IP ,
199
+ Port : 0 ,
200
+
201
+ Priority : 10 ,
202
+ Weight : 10 ,
203
+ Ttl : 30 ,
173
204
}
174
- hadPort := false
205
+ defaultHash := getHash (defaultService .Host )
206
+ defaultName := buildDNSName (subdomain , defaultHash )
207
+ defaultService .Key = msg .Path (defaultName )
208
+
175
209
for _ , p := range s .Ports {
176
210
port := p .Port
177
211
if port == 0 {
178
212
continue
179
213
}
180
- hadPort = true
181
214
if len (p .Protocol ) == 0 {
182
215
p .Protocol = kapi .ProtocolTCP
183
216
}
184
217
portName := p .Name
185
218
if len (portName ) == 0 {
186
219
portName = fmt .Sprintf ("unknown-port-%d" , port )
187
220
}
188
- srvName := fmt .Sprintf ("%s.%s.%s" , portName , shortName , name )
189
- services = append (services , msg.Service {
190
- Host : a .IP ,
191
- Port : port ,
192
221
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 )
222
+ keyName := buildDNSName (subdomain , "_" + strings .ToLower (string (p .Protocol )), "_" + portName , defaultHash )
201
223
services = append (services , msg.Service {
202
- Host : srvName ,
224
+ Host : a . IP ,
203
225
Port : port ,
204
226
205
227
Priority : 10 ,
206
228
Weight : 10 ,
207
229
Ttl : 30 ,
208
230
209
- Text : "" ,
210
- Key : msg .Path (keyName ),
231
+ Key : msg .Path (keyName ),
211
232
})
212
233
}
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
- })
234
+ if len (services ) == 0 {
235
+ services = append (services , defaultService )
225
236
}
226
237
}
227
238
}
239
+ glog .V (4 ).Infof ("Answered %s:%t with %#v" , dnsName , exact , services )
228
240
return services , nil
229
241
}
230
242
return nil , nil
@@ -246,16 +258,16 @@ func (b *ServiceResolver) ReverseRecord(name string) (*msg.Service, error) {
246
258
if len (svc .Spec .Ports ) > 0 {
247
259
port = svc .Spec .Ports [0 ].Port
248
260
}
261
+ hostName := buildDNSName (b .base , "svc" , svc .Namespace , svc .Name )
249
262
return & msg.Service {
250
- Host : fmt . Sprintf ( "%s.%s.svc.%s" , svc . Name , svc . Namespace , b . base ) ,
263
+ Host : hostName ,
251
264
Port : port ,
252
265
253
266
Priority : 10 ,
254
267
Weight : 10 ,
255
268
Ttl : 30 ,
256
269
257
- Text : "" ,
258
- Key : msg .Path (name ),
270
+ Key : msg .Path (name ),
259
271
}, nil
260
272
}
261
273
@@ -278,3 +290,29 @@ func extractIP(reverseName string) (string, bool) {
278
290
}
279
291
return strings .Join (segments , "." ), true
280
292
}
293
+
294
+ // buildDNSName reverses the labels order and joins them with dots.
295
+ func buildDNSName (labels ... string ) string {
296
+ var res string
297
+ for _ , label := range labels {
298
+ if len (res ) == 0 {
299
+ res = label
300
+ } else {
301
+ res = fmt .Sprintf ("%s.%s" , label , res )
302
+ }
303
+ }
304
+ return res
305
+ }
306
+
307
+ // return a hash for the key name
308
+ func getHash (text string ) string {
309
+ h := fnv .New32a ()
310
+ h .Write ([]byte (text ))
311
+ return fmt .Sprintf ("%x" , h .Sum32 ())
312
+ }
313
+
314
+ // convertDashIPToIP takes an encoded IP (with dashes) and replaces them with
315
+ // dots.
316
+ func convertDashIPToIP (ip string ) string {
317
+ return strings .Join (strings .Split (ip , "-" ), "." )
318
+ }
0 commit comments