@@ -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,103 @@ 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
+ break
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
+ alreadyAllocated := make (map [string ]bool )
481
+ for _ , node := range eit .nodesByNodeIP {
482
+ if len (node .parsedCIDRs ) > 0 {
483
+ allocation [node .nodeName ] = make ([]string , 0 , node .requestedIPs .Len ())
484
+ }
485
+ }
486
+ // For each active egress IP, if it still fits within some egress CIDR on its node,
487
+ // add it to that node's allocation. (Otherwise add the node to the "changed" map,
488
+ // since we'll be removing this egress IP from it.)
489
+ for egressIP , eip := range eit .egressIPs {
490
+ if eip .assignedNodeIP == "" {
491
+ continue
492
+ }
493
+ node := eip .nodes [0 ]
494
+ found := false
495
+ for _ , parsed := range node .parsedCIDRs {
496
+ if parsed .Contains (eip .parsed ) {
497
+ found = true
498
+ break
499
+ }
500
+ }
501
+ if found {
502
+ allocation [node .nodeName ] = append (allocation [node .nodeName ], egressIP )
503
+ } else {
504
+ changed [node .nodeName ] = true
505
+ }
506
+ // (We set alreadyAllocated even if the egressIP will be removed from
507
+ // its current node; we can't assign it to a new node until the next
508
+ // reallocation.)
509
+ alreadyAllocated [egressIP ] = true
510
+ }
511
+
512
+ // Allocate pending egress IPs that can only go to a single node
513
+ for egressIP , eip := range eit .egressIPs {
514
+ if alreadyAllocated [egressIP ] {
515
+ continue
516
+ }
517
+ nodeName , otherNodes := eit .findEgressIPAllocation (eip .parsed , allocation )
518
+ if nodeName != "" && ! otherNodes {
519
+ allocation [nodeName ] = append (allocation [nodeName ], egressIP )
520
+ changed [nodeName ] = true
521
+ alreadyAllocated [egressIP ] = true
522
+ }
523
+ }
524
+ // Allocate any other pending egress IPs that we can
525
+ for egressIP , eip := range eit .egressIPs {
526
+ if alreadyAllocated [egressIP ] {
527
+ continue
528
+ }
529
+ nodeName , _ := eit .findEgressIPAllocation (eip .parsed , allocation )
530
+ if nodeName != "" {
531
+ allocation [nodeName ] = append (allocation [nodeName ], egressIP )
532
+ changed [nodeName ] = true
533
+ }
534
+ }
535
+
536
+ // Remove unchanged nodes from the return value
537
+ for _ , node := range eit .nodesByNodeIP {
538
+ if ! changed [node .nodeName ] {
539
+ delete (allocation , node .nodeName )
540
+ }
541
+ }
542
+ return allocation
543
+ }
0 commit comments