@@ -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"
@@ -60,20 +64,42 @@ func NewServiceResolver(config *server.Config, accessor ServiceAccessor, endpoin
60
64
// * endpoint_id is "portal" when portalIP is set
61
65
// * endpoints always returns each individual endpoint as A records
62
66
//
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 ) {
65
69
return nil , nil
66
70
}
67
- prefix := strings .Trim (strings .TrimSuffix (name , b .base ), "." )
71
+ prefix := strings .Trim (strings .TrimSuffix (dnsName , b .base ), "." )
68
72
segments := strings .Split (prefix , "." )
69
73
for i , j := 0 , len (segments )- 1 ; i < j ; i , j = i + 1 , j - 1 {
70
74
segments [i ], segments [j ] = segments [j ], segments [i ]
71
75
}
72
76
if len (segments ) == 0 {
73
77
return nil , nil
74
78
}
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
75
102
76
- switch segments [0 ] {
77
103
case "svc" , "endpoints" :
78
104
if len (segments ) < 3 {
79
105
return nil , nil
@@ -94,56 +120,64 @@ func (b *ServiceResolver) Records(name string, exact bool) ([]msg.Service, error
94
120
return nil , nil
95
121
}
96
122
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" )
98
126
99
127
// if has a portal IP and looking at svc
100
128
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 ,
103
136
}
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 )
126
140
127
- Priority : 10 ,
128
- Weight : 10 ,
129
- Ttl : 30 ,
141
+ if len ( svc . Spec . Ports ) == 0 {
142
+ return []msg. Service { defaultService }, nil
143
+ }
130
144
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 ,
137
167
138
- Priority : 10 ,
139
- Weight : 10 ,
140
- Ttl : 30 ,
168
+ Priority : 10 ,
169
+ Weight : 10 ,
170
+ Ttl : 30 ,
141
171
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 )
146
179
}
180
+ glog .V (4 ).Infof ("Answered %s:%t with %#v" , dnsName , exact , services )
147
181
return services , nil
148
182
}
149
183
@@ -152,79 +186,53 @@ func (b *ServiceResolver) Records(name string, exact bool) ([]msg.Service, error
152
186
if err != nil {
153
187
return nil , err
154
188
}
155
- targets := make ( map [ string ] int )
189
+
156
190
services := make ([]msg.Service , 0 , len (endpoints .Subsets )* 4 )
157
- count := 1
158
191
for _ , s := range endpoints .Subsets {
159
192
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 ,
173
200
}
174
- hadPort := false
201
+ defaultHash := getHash (defaultService .Host )
202
+ defaultName := buildDNSName (subdomain , defaultHash )
203
+ defaultService .Key = msg .Path (defaultName )
204
+
175
205
for _ , p := range s .Ports {
176
206
port := p .Port
177
207
if port == 0 {
178
208
continue
179
209
}
180
- hadPort = true
181
210
if len (p .Protocol ) == 0 {
182
211
p .Protocol = kapi .ProtocolTCP
183
212
}
184
213
portName := p .Name
185
214
if len (portName ) == 0 {
186
215
portName = fmt .Sprintf ("unknown-port-%d" , port )
187
216
}
188
- srvName := fmt .Sprintf ("%s.%s.%s" , portName , shortName , name )
189
- services = append (services , msg.Service {
190
- Host : a .IP ,
191
- Port : port ,
192
217
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 )
201
219
services = append (services , msg.Service {
202
- Host : srvName ,
220
+ Host : a . IP ,
203
221
Port : port ,
204
222
205
223
Priority : 10 ,
206
224
Weight : 10 ,
207
225
Ttl : 30 ,
208
226
209
- Text : "" ,
210
- Key : msg .Path (keyName ),
227
+ Key : msg .Path (keyName ),
211
228
})
212
229
}
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 )
225
232
}
226
233
}
227
234
}
235
+ glog .V (4 ).Infof ("Answered %s:%t with %#v" , dnsName , exact , services )
228
236
return services , nil
229
237
}
230
238
return nil , nil
@@ -246,16 +254,16 @@ func (b *ServiceResolver) ReverseRecord(name string) (*msg.Service, error) {
246
254
if len (svc .Spec .Ports ) > 0 {
247
255
port = svc .Spec .Ports [0 ].Port
248
256
}
257
+ hostName := buildDNSName (b .base , "svc" , svc .Namespace , svc .Name )
249
258
return & msg.Service {
250
- Host : fmt . Sprintf ( "%s.%s.svc.%s" , svc . Name , svc . Namespace , b . base ) ,
259
+ Host : hostName ,
251
260
Port : port ,
252
261
253
262
Priority : 10 ,
254
263
Weight : 10 ,
255
264
Ttl : 30 ,
256
265
257
- Text : "" ,
258
- Key : msg .Path (name ),
266
+ Key : msg .Path (name ),
259
267
}, nil
260
268
}
261
269
@@ -278,3 +286,27 @@ func extractIP(reverseName string) (string, bool) {
278
286
}
279
287
return strings .Join (segments , "." ), true
280
288
}
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