Thomas Ferrandiz d6dcf76759 rename Adresses
2023-10-07 08:52:31 -05:00

143 lines
4.7 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")
}
// GetNodeIP 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 GetNodeIP(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)
}
}
// GetNodeIP returns the most valid external facing IP address for a node (IPv4 and IPv6).
// Order of preference:
// 1. NodeInternalIP
// 2. NodeExternalIP (Only set on cloud providers usually)
func GetNodeIPDualStack(node *apiv1.Node, enableIPv4, enableIPv6 bool) (net.IP, net.IP, error) {
var ipAddrv4, ipAddrv6 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 ipAddrv4 == nil && enableIPv4 && netutils.IsIPv4String(address.Address) {
ipAddrv4 = net.ParseIP(address.Address)
}
if ipAddrv6 == nil && enableIPv6 && netutils.IsIPv6String(address.Address) {
ipAddrv6 = net.ParseIP(address.Address)
}
}
}
if externalAddresses, ok := addressesPerType[apiv1.NodeExternalIP]; ok {
for _, address := range externalAddresses {
if ipAddrv4 == nil && enableIPv4 && netutils.IsIPv4String(address.Address) {
ipAddrv4 = net.ParseIP(address.Address)
}
if ipAddrv6 == nil && enableIPv6 && netutils.IsIPv6String(address.Address) {
ipAddrv6 = net.ParseIP(address.Address)
}
}
}
if enableIPv4 && ipAddrv4 == nil {
return nil, nil, errors.New("host IPv4 unknown")
}
if enableIPv6 && ipAddrv6 == nil {
return nil, nil, errors.New("host IPv6 unknown")
}
return ipAddrv4, ipAddrv6, 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")
}