@@ -18,10 +18,15 @@ import (
18
18
)
19
19
20
20
type nodeEgress struct {
21
- nodeIP string
22
- sdnIP string
23
- requestedIPs sets.String
24
- offline bool
21
+ nodeName string
22
+ nodeIP string
23
+ sdnIP string
24
+
25
+ requestedIPs sets.String
26
+ requestedCIDRs sets.String
27
+ parsedCIDRs map [string ]* net.IPNet
28
+
29
+ offline bool
25
30
}
26
31
27
32
type namespaceEgress struct {
@@ -32,7 +37,8 @@ type namespaceEgress struct {
32
37
}
33
38
34
39
type egressIPInfo struct {
35
- ip string
40
+ ip string
41
+ parsed net.IP
36
42
37
43
nodes []* nodeEgress
38
44
namespaces []* namespaceEgress
@@ -48,6 +54,8 @@ type EgressIPWatcher interface {
48
54
SetNamespaceEgressNormal (vnid uint32 )
49
55
SetNamespaceEgressDropped (vnid uint32 )
50
56
SetNamespaceEgressViaEgressIP (vnid uint32 , egressIP , nodeIP string )
57
+
58
+ UpdateEgressCIDRs ()
51
59
}
52
60
53
61
type EgressIPTracker struct {
@@ -58,9 +66,11 @@ type EgressIPTracker struct {
58
66
nodesByNodeIP map [string ]* nodeEgress
59
67
namespacesByVNID map [uint32 ]* namespaceEgress
60
68
egressIPs map [string ]* egressIPInfo
69
+ nodesWithCIDRs int
61
70
62
71
changedEgressIPs map [* egressIPInfo ]bool
63
72
changedNamespaces map [* namespaceEgress ]bool
73
+ updateEgressCIDRs bool
64
74
}
65
75
66
76
func NewEgressIPTracker (watcher EgressIPWatcher ) * EgressIPTracker {
@@ -84,7 +94,7 @@ func (eit *EgressIPTracker) Start(hostSubnetInformer networkinformers.HostSubnet
84
94
func (eit * EgressIPTracker ) ensureEgressIPInfo (egressIP string ) * egressIPInfo {
85
95
eg := eit .egressIPs [egressIP ]
86
96
if eg == nil {
87
- eg = & egressIPInfo {ip : egressIP }
97
+ eg = & egressIPInfo {ip : egressIP , parsed : net . ParseIP ( egressIP ) }
88
98
eit .egressIPs [egressIP ] = eg
89
99
}
90
100
return eg
@@ -177,10 +187,11 @@ func (eit *EgressIPTracker) UpdateHostSubnetEgress(hs *networkapi.HostSubnet) {
177
187
178
188
node := eit .nodesByNodeIP [hs .HostIP ]
179
189
if node == nil {
180
- if len (hs .EgressIPs ) == 0 {
190
+ if len (hs .EgressIPs ) == 0 && len ( hs . EgressCIDRs ) == 0 {
181
191
return
182
192
}
183
193
node = & nodeEgress {
194
+ nodeName : hs .Host ,
184
195
nodeIP : hs .HostIP ,
185
196
sdnIP : sdnIP ,
186
197
requestedIPs : sets .NewString (),
@@ -189,10 +200,27 @@ func (eit *EgressIPTracker) UpdateHostSubnetEgress(hs *networkapi.HostSubnet) {
189
200
} else if len (hs .EgressIPs ) == 0 {
190
201
delete (eit .nodesByNodeIP , hs .HostIP )
191
202
}
192
- oldRequestedIPs := node .requestedIPs
193
- node .requestedIPs = sets .NewString (hs .EgressIPs ... )
203
+
204
+ // Process EgressCIDRs
205
+ newRequestedCIDRs := sets .NewString (hs .EgressCIDRs ... )
206
+ if ! node .requestedCIDRs .Equal (newRequestedCIDRs ) {
207
+ if len (hs .EgressCIDRs ) == 0 {
208
+ eit .nodesWithCIDRs --
209
+ } else if node .requestedCIDRs .Len () == 0 {
210
+ eit .nodesWithCIDRs ++
211
+ }
212
+ node .requestedCIDRs = newRequestedCIDRs
213
+ node .parsedCIDRs = make (map [string ]* net.IPNet )
214
+ for _ , cidr := range hs .EgressCIDRs {
215
+ _ , parsed , _ := net .ParseCIDR (cidr )
216
+ node .parsedCIDRs [cidr ] = parsed
217
+ }
218
+ eit .updateEgressCIDRs = true
219
+ }
194
220
195
221
// Process new and removed EgressIPs
222
+ oldRequestedIPs := node .requestedIPs
223
+ node .requestedIPs = sets .NewString (hs .EgressIPs ... )
196
224
for _ , ip := range node .requestedIPs .Difference (oldRequestedIPs ).UnsortedList () {
197
225
eit .addNodeEgressIP (node , ip )
198
226
}
@@ -301,6 +329,13 @@ func (eit *EgressIPTracker) syncEgressIPs() {
301
329
for ns := range changedNamespaces {
302
330
eit .syncEgressNamespaceState (ns )
303
331
}
332
+
333
+ if eit .updateEgressCIDRs {
334
+ eit .updateEgressCIDRs = false
335
+ if eit .nodesWithCIDRs > 0 {
336
+ eit .watcher .UpdateEgressCIDRs ()
337
+ }
338
+ }
304
339
}
305
340
306
341
func (eit * EgressIPTracker ) syncEgressNodeState (eg * egressIPInfo , active bool ) {
@@ -313,6 +348,10 @@ func (eit *EgressIPTracker) syncEgressNodeState(eg *egressIPInfo, active bool) {
313
348
eit .watcher .ReleaseEgressIP (eg .ip , eg .assignedNodeIP )
314
349
eg .assignedNodeIP = ""
315
350
}
351
+
352
+ if eg .assignedNodeIP == "" {
353
+ eit .updateEgressCIDRs = true
354
+ }
316
355
}
317
356
318
357
func (eit * EgressIPTracker ) syncEgressNamespaceState (ns * namespaceEgress ) {
@@ -402,3 +441,100 @@ func (eit *EgressIPTracker) Ping(ip string, timeout time.Duration) bool {
402
441
return true
403
442
}
404
443
}
444
+
445
+ // Finds the best node to allocate the egress IP to, given the existing allocation. The
446
+ // boolean return value indicates whether multiple nodes could host the IP.
447
+ func (eit * EgressIPTracker ) findEgressIPAllocation (ip net.IP , allocation map [string ][]string ) (string , bool ) {
448
+ bestNode := ""
449
+ otherNodes := false
450
+
451
+ for _ , node := range eit .nodesByNodeIP {
452
+ egressIPs , exists := allocation [node .nodeName ]
453
+ if ! exists {
454
+ continue
455
+ }
456
+ for _ , parsed := range node .parsedCIDRs {
457
+ if parsed .Contains (ip ) {
458
+ if bestNode != "" {
459
+ otherNodes = true
460
+ if len (allocation [bestNode ]) < len (egressIPs ) {
461
+ continue
462
+ }
463
+ }
464
+ bestNode = node .nodeName
465
+ break
466
+ }
467
+ }
468
+ }
469
+
470
+ return bestNode , otherNodes
471
+ }
472
+
473
+ // ReallocateEgressIPs returns a map from Node name to array-of-Egress-IP. Unchanged nodes are not included.
474
+ func (eit * EgressIPTracker ) ReallocateEgressIPs () map [string ][]string {
475
+ eit .Lock ()
476
+ defer eit .Unlock ()
477
+
478
+ allocation := make (map [string ][]string )
479
+ changed := make (map [string ]bool )
480
+ for _ , node := range eit .nodesByNodeIP {
481
+ if len (node .parsedCIDRs ) > 0 {
482
+ allocation [node .nodeName ] = make ([]string , 0 , node .requestedIPs .Len ())
483
+ }
484
+ }
485
+ // For each active egress IP, if it still fits within some egress CIDR on its node,
486
+ // add it to that node's allocation. (Otherwise add the node to the "changed" map,
487
+ // since we'll be removing this egress IP from it.)
488
+ for egressIP , eip := range eit .egressIPs {
489
+ if eip .assignedNodeIP == "" {
490
+ continue
491
+ }
492
+ node := eip .nodes [0 ]
493
+ found := false
494
+ for _ , parsed := range node .parsedCIDRs {
495
+ if parsed .Contains (eip .parsed ) {
496
+ found = true
497
+ break
498
+ }
499
+ }
500
+ if found {
501
+ allocation [node .nodeName ] = append (allocation [node .nodeName ], egressIP )
502
+ } else {
503
+ changed [node .nodeName ] = true
504
+ }
505
+ }
506
+
507
+ // Allocate pending egress IPs that can only go to a single node
508
+ alreadyAllocated := make (map [string ]bool )
509
+ for egressIP , eip := range eit .egressIPs {
510
+ if eip .assignedNodeIP != "" {
511
+ alreadyAllocated [egressIP ] = true
512
+ continue
513
+ }
514
+ nodeName , otherNodes := eit .findEgressIPAllocation (eip .parsed , allocation )
515
+ if nodeName != "" && ! otherNodes {
516
+ allocation [nodeName ] = append (allocation [nodeName ], egressIP )
517
+ changed [nodeName ] = true
518
+ alreadyAllocated [egressIP ] = true
519
+ }
520
+ }
521
+ // Allocate any other pending egress IPs that we can
522
+ for egressIP , eip := range eit .egressIPs {
523
+ if alreadyAllocated [egressIP ] {
524
+ continue
525
+ }
526
+ nodeName , _ := eit .findEgressIPAllocation (eip .parsed , allocation )
527
+ if nodeName != "" {
528
+ allocation [nodeName ] = append (allocation [nodeName ], egressIP )
529
+ changed [nodeName ] = true
530
+ }
531
+ }
532
+
533
+ // Remove unchanged nodes from the return value
534
+ for _ , node := range eit .nodesByNodeIP {
535
+ if ! changed [node .nodeName ] {
536
+ delete (allocation , node .nodeName )
537
+ }
538
+ }
539
+ return allocation
540
+ }
0 commit comments