Skip to content

Commit 2b089ac

Browse files
committed
Track HostSubnet.EgressCIDR, assign Egress IPs from it on the master
1 parent 91e350e commit 2b089ac

File tree

5 files changed

+492
-9
lines changed

5 files changed

+492
-9
lines changed

pkg/network/common/egressip.go

+145-9
Original file line numberDiff line numberDiff line change
@@ -18,10 +18,15 @@ import (
1818
)
1919

2020
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
2530
}
2631

2732
type namespaceEgress struct {
@@ -32,7 +37,8 @@ type namespaceEgress struct {
3237
}
3338

3439
type egressIPInfo struct {
35-
ip string
40+
ip string
41+
parsed net.IP
3642

3743
nodes []*nodeEgress
3844
namespaces []*namespaceEgress
@@ -48,6 +54,8 @@ type EgressIPWatcher interface {
4854
SetNamespaceEgressNormal(vnid uint32)
4955
SetNamespaceEgressDropped(vnid uint32)
5056
SetNamespaceEgressViaEgressIP(vnid uint32, egressIP, nodeIP string)
57+
58+
UpdateEgressCIDRs()
5159
}
5260

5361
type EgressIPTracker struct {
@@ -58,9 +66,11 @@ type EgressIPTracker struct {
5866
nodesByNodeIP map[string]*nodeEgress
5967
namespacesByVNID map[uint32]*namespaceEgress
6068
egressIPs map[string]*egressIPInfo
69+
nodesWithCIDRs int
6170

6271
changedEgressIPs map[*egressIPInfo]bool
6372
changedNamespaces map[*namespaceEgress]bool
73+
updateEgressCIDRs bool
6474
}
6575

6676
func NewEgressIPTracker(watcher EgressIPWatcher) *EgressIPTracker {
@@ -84,7 +94,7 @@ func (eit *EgressIPTracker) Start(hostSubnetInformer networkinformers.HostSubnet
8494
func (eit *EgressIPTracker) ensureEgressIPInfo(egressIP string) *egressIPInfo {
8595
eg := eit.egressIPs[egressIP]
8696
if eg == nil {
87-
eg = &egressIPInfo{ip: egressIP}
97+
eg = &egressIPInfo{ip: egressIP, parsed: net.ParseIP(egressIP)}
8898
eit.egressIPs[egressIP] = eg
8999
}
90100
return eg
@@ -177,10 +187,11 @@ func (eit *EgressIPTracker) UpdateHostSubnetEgress(hs *networkapi.HostSubnet) {
177187

178188
node := eit.nodesByNodeIP[hs.HostIP]
179189
if node == nil {
180-
if len(hs.EgressIPs) == 0 {
190+
if len(hs.EgressIPs) == 0 && len(hs.EgressCIDRs) == 0 {
181191
return
182192
}
183193
node = &nodeEgress{
194+
nodeName: hs.Host,
184195
nodeIP: hs.HostIP,
185196
sdnIP: sdnIP,
186197
requestedIPs: sets.NewString(),
@@ -189,10 +200,27 @@ func (eit *EgressIPTracker) UpdateHostSubnetEgress(hs *networkapi.HostSubnet) {
189200
} else if len(hs.EgressIPs) == 0 {
190201
delete(eit.nodesByNodeIP, hs.HostIP)
191202
}
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+
}
194220

195221
// Process new and removed EgressIPs
222+
oldRequestedIPs := node.requestedIPs
223+
node.requestedIPs = sets.NewString(hs.EgressIPs...)
196224
for _, ip := range node.requestedIPs.Difference(oldRequestedIPs).UnsortedList() {
197225
eit.addNodeEgressIP(node, ip)
198226
}
@@ -301,6 +329,13 @@ func (eit *EgressIPTracker) syncEgressIPs() {
301329
for ns := range changedNamespaces {
302330
eit.syncEgressNamespaceState(ns)
303331
}
332+
333+
if eit.updateEgressCIDRs {
334+
eit.updateEgressCIDRs = false
335+
if eit.nodesWithCIDRs > 0 {
336+
eit.watcher.UpdateEgressCIDRs()
337+
}
338+
}
304339
}
305340

306341
func (eit *EgressIPTracker) syncEgressNodeState(eg *egressIPInfo, active bool) {
@@ -313,6 +348,10 @@ func (eit *EgressIPTracker) syncEgressNodeState(eg *egressIPInfo, active bool) {
313348
eit.watcher.ReleaseEgressIP(eg.ip, eg.assignedNodeIP)
314349
eg.assignedNodeIP = ""
315350
}
351+
352+
if eg.assignedNodeIP == "" {
353+
eit.updateEgressCIDRs = true
354+
}
316355
}
317356

318357
func (eit *EgressIPTracker) syncEgressNamespaceState(ns *namespaceEgress) {
@@ -402,3 +441,100 @@ func (eit *EgressIPTracker) Ping(ip string, timeout time.Duration) bool {
402441
return true
403442
}
404443
}
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

Comments
 (0)