mirror of
https://github.com/cloudnativelabs/kube-router.git
synced 2025-09-27 19:11:05 +02:00
169 lines
6.0 KiB
Go
169 lines
6.0 KiB
Go
package utils
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
|
|
"github.com/vishvananda/netlink"
|
|
|
|
apiv1 "k8s.io/api/core/v1"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/client-go/kubernetes"
|
|
netutils "k8s.io/utils/net"
|
|
)
|
|
|
|
// GetNodeObject returns the node API object for the node
|
|
func GetNodeObject(clientset kubernetes.Interface, hostnameOverride string) (*apiv1.Node, error) {
|
|
// assuming kube-router is running as pod, first check env NODE_NAME
|
|
nodeName := os.Getenv("NODE_NAME")
|
|
if nodeName != "" {
|
|
node, err := clientset.CoreV1().Nodes().Get(context.Background(), nodeName, metav1.GetOptions{})
|
|
if err == nil {
|
|
return node, nil
|
|
}
|
|
}
|
|
|
|
// if env NODE_NAME is not set then check if node is register with hostname
|
|
hostName, _ := os.Hostname()
|
|
node, err := clientset.CoreV1().Nodes().Get(context.Background(), hostName, metav1.GetOptions{})
|
|
if err == nil {
|
|
return node, nil
|
|
}
|
|
|
|
// if env NODE_NAME is not set and node is not registered with hostname, then use host name override
|
|
if hostnameOverride != "" {
|
|
node, err = clientset.CoreV1().Nodes().Get(context.Background(), hostnameOverride, metav1.GetOptions{})
|
|
if err == nil {
|
|
return node, nil
|
|
}
|
|
}
|
|
|
|
return nil, fmt.Errorf("failed to identify the node by NODE_NAME, hostname or --hostname-override")
|
|
}
|
|
|
|
// GetPrimaryNodeIP returns the most valid external facing IP address for a node.
|
|
// Order of preference:
|
|
// 1. NodeInternalIP
|
|
// 2. NodeExternalIP (Only set on cloud providers usually)
|
|
func GetPrimaryNodeIP(node *apiv1.Node) (net.IP, error) {
|
|
addresses := node.Status.Addresses
|
|
addressMap := make(map[apiv1.NodeAddressType][]apiv1.NodeAddress)
|
|
for i := range addresses {
|
|
addressMap[addresses[i].Type] = append(addressMap[addresses[i].Type], addresses[i])
|
|
}
|
|
if addresses, ok := addressMap[apiv1.NodeInternalIP]; ok {
|
|
return net.ParseIP(addresses[0].Address), nil
|
|
}
|
|
if addresses, ok := addressMap[apiv1.NodeExternalIP]; ok {
|
|
return net.ParseIP(addresses[0].Address), nil
|
|
}
|
|
return nil, errors.New("host IP unknown")
|
|
}
|
|
|
|
// addressMap is a mapping of address types to a list of addresses of that type.
|
|
// It preallocates the slices of addresses.
|
|
type addressMap map[apiv1.NodeAddressType][]apiv1.NodeAddress
|
|
|
|
// add adds an address of the given type to the address map. If the given type
|
|
// was not already in the map, it creates a new preallocated entry for it.
|
|
func (m addressMap) add(address apiv1.NodeAddress) {
|
|
if _, ok := m[address.Type]; ok {
|
|
m[address.Type] = append(m[address.Type], address)
|
|
} else {
|
|
// There can be at most 2 addresses of the same type.
|
|
m[address.Type] = make([]apiv1.NodeAddress, 2)
|
|
m[address.Type] = append(m[address.Type], address)
|
|
}
|
|
}
|
|
|
|
// GetAllNodeIPs returns all internal and external IP addresses grouped as IPv4 and IPv6
|
|
func GetAllNodeIPs(node *apiv1.Node) (map[apiv1.NodeAddressType][]net.IP, map[apiv1.NodeAddressType][]net.IP) {
|
|
ipAddrv4 := make(map[apiv1.NodeAddressType][]net.IP)
|
|
ipAddrv6 := make(map[apiv1.NodeAddressType][]net.IP)
|
|
addresses := node.Status.Addresses
|
|
addressesPerType := make(addressMap)
|
|
for _, address := range addresses {
|
|
addressesPerType.add(address)
|
|
}
|
|
if internalAddresses, ok := addressesPerType[apiv1.NodeInternalIP]; ok {
|
|
for _, address := range internalAddresses {
|
|
if netutils.IsIPv4String(address.Address) {
|
|
ipAddrv4[apiv1.NodeInternalIP] = append(ipAddrv4[apiv1.NodeInternalIP], net.ParseIP(address.Address))
|
|
}
|
|
if netutils.IsIPv6String(address.Address) {
|
|
ipAddrv6[apiv1.NodeInternalIP] = append(ipAddrv6[apiv1.NodeInternalIP], net.ParseIP(address.Address))
|
|
}
|
|
}
|
|
}
|
|
if externalAddresses, ok := addressesPerType[apiv1.NodeExternalIP]; ok {
|
|
for _, address := range externalAddresses {
|
|
if netutils.IsIPv4String(address.Address) {
|
|
ipAddrv4[apiv1.NodeExternalIP] = append(ipAddrv4[apiv1.NodeExternalIP], net.ParseIP(address.Address))
|
|
}
|
|
if netutils.IsIPv6String(address.Address) {
|
|
ipAddrv6[apiv1.NodeExternalIP] = append(ipAddrv6[apiv1.NodeExternalIP], net.ParseIP(address.Address))
|
|
}
|
|
}
|
|
}
|
|
|
|
return ipAddrv4, ipAddrv6
|
|
}
|
|
|
|
func FindBestIPv6NodeAddress(priIP net.IP, intExtIPv6Addresses map[apiv1.NodeAddressType][]net.IP) net.IP {
|
|
if priIP != nil && priIP.To4() == nil && priIP.To16() != nil {
|
|
// the NRC's primary IP is already an IPv6 address, so we'll use that
|
|
return priIP
|
|
}
|
|
// the NRC's primary IP is not an IPv6, let's try to find the best available IPv6 address out of our
|
|
// available node addresses to use as the nextHop for our route
|
|
if intExtIPv6Addresses != nil {
|
|
if len(intExtIPv6Addresses[apiv1.NodeInternalIP]) > 0 {
|
|
return intExtIPv6Addresses[apiv1.NodeInternalIP][0]
|
|
} else if len(intExtIPv6Addresses[apiv1.NodeExternalIP]) > 0 {
|
|
return intExtIPv6Addresses[apiv1.NodeExternalIP][0]
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func FindBestIPv4NodeAddress(priIP net.IP, intExtIPv4Addresses map[apiv1.NodeAddressType][]net.IP) net.IP {
|
|
if priIP != nil && priIP.To4() != nil {
|
|
// the NRC's primary IP is already an IPv6 address, so we'll use that
|
|
return priIP
|
|
}
|
|
// the NRC's primary IP is not an IPv6, let's try to find the best available IPv6 address out of our
|
|
// available node addresses to use as the nextHop for our route
|
|
if intExtIPv4Addresses != nil {
|
|
if len(intExtIPv4Addresses[apiv1.NodeInternalIP]) > 0 {
|
|
return intExtIPv4Addresses[apiv1.NodeInternalIP][0]
|
|
} else if len(intExtIPv4Addresses[apiv1.NodeExternalIP]) > 0 {
|
|
return intExtIPv4Addresses[apiv1.NodeExternalIP][0]
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// GetMTUFromNodeIP returns the MTU by detecting it from the IP on the node and figuring in tunneling configurations
|
|
func GetMTUFromNodeIP(nodeIP net.IP) (int, error) {
|
|
links, err := netlink.LinkList()
|
|
if err != nil {
|
|
return 0, errors.New("failed to get list of links")
|
|
}
|
|
for _, link := range links {
|
|
addresses, err := netlink.AddrList(link, netlink.FAMILY_ALL)
|
|
if err != nil {
|
|
return 0, errors.New("failed to get list of addr")
|
|
}
|
|
for _, addr := range addresses {
|
|
if addr.IPNet.IP.Equal(nodeIP) {
|
|
linkMTU := link.Attrs().MTU
|
|
return linkMTU, nil
|
|
}
|
|
}
|
|
}
|
|
return 0, errors.New("failed to find interface with specified node IP")
|
|
}
|