azure-container-networking/ipam/fileIpam.go

203 строки
4.5 KiB
Go

// Copyright 2017 Microsoft. All rights reserved.
// MIT License
package ipam
import (
"encoding/json"
"errors"
"net"
"os"
"runtime"
"strings"
"github.com/Azure/azure-container-networking/common"
"go.uber.org/zap"
)
const (
defaultLinuxFilePath = "/etc/kubernetes/interfaces.json"
defaultWindowsFilePath = `c:\k\interfaces.json`
windows = "windows"
)
// Microsoft Azure Stack IPAM configuration source.
type fileIpamSource struct {
name string
sink addressConfigSink
fileLoaded bool
filePath string
}
// MAS host agent JSON object format.
type NetworkInterfaces struct {
Interfaces []Interface
}
type Interface struct {
MacAddress string
IsPrimary bool
IPSubnets []IPSubnet
}
type IPSubnet struct {
Prefix string
IPAddresses []IPAddress
}
type IPAddress struct {
Address string
IsPrimary bool
}
// Creates the MAS/fileIpam source.
func newFileIpamSource(options map[string]interface{}) (*fileIpamSource, error) {
var filePath string
var name string
if runtime.GOOS == windows {
filePath = defaultWindowsFilePath
} else {
filePath = defaultLinuxFilePath
}
name = options[common.OptEnvironment].(string)
return &fileIpamSource{
name: name,
filePath: filePath,
}, nil
}
// Starts the MAS source.
func (source *fileIpamSource) start(sink addressConfigSink) error {
source.sink = sink
return nil
}
// Stops the MAS source.
func (source *fileIpamSource) stop() {
source.sink = nil
}
// Refreshes configuration.
func (source *fileIpamSource) refresh() error {
if source == nil {
return errors.New("fileIpamSource is nil")
}
if source.fileLoaded {
return nil
}
// Query the list of local interfaces.
localInterfaces, err := net.Interfaces()
if err != nil {
return err
}
// Query the list of Azure Network Interfaces
sdnInterfaces, err := getSDNInterfaces(source.filePath)
if err != nil {
return err
}
// Configure the local default address space.
local, err := source.sink.newAddressSpace(LocalDefaultAddressSpaceId, LocalScope)
if err != nil {
return err
}
if err = populateAddressSpace(local, sdnInterfaces, localInterfaces); err != nil {
return err
}
// Set the local address space as active.
if err = source.sink.setAddressSpace(local); err != nil {
return err
}
logger.Info("Address space successfully populated from config file")
source.fileLoaded = true
return nil
}
func getSDNInterfaces(fileLocation string) (*NetworkInterfaces, error) {
data, err := os.ReadFile(fileLocation)
if err != nil {
return nil, err
}
interfaces := &NetworkInterfaces{}
if err = json.Unmarshal(data, interfaces); err != nil {
return nil, err
}
return interfaces, nil
}
func populateAddressSpace(localAddressSpace *addressSpace, sdnInterfaces *NetworkInterfaces, localInterfaces []net.Interface) error {
// Find the interface with matching MacAddress or Name
for _, sdnIf := range sdnInterfaces.Interfaces {
ifName := ""
for _, localIf := range localInterfaces {
if macAddressesEqual(sdnIf.MacAddress, localIf.HardwareAddr.String()) {
ifName = localIf.Name
break
}
}
// Skip if interface is not found.
if ifName == "" {
logger.Info("Failed to find interface with", zap.String("MAC Address", sdnIf.MacAddress))
continue
}
// Prioritize secondary interfaces.
priority := 0
if !sdnIf.IsPrimary {
priority = 1
}
for _, subnet := range sdnIf.IPSubnets {
_, network, err := net.ParseCIDR(subnet.Prefix)
if err != nil {
logger.Error("Failed to parse subnet", zap.String("prefix", subnet.Prefix), zap.Error(err))
continue
}
addressPool, err := localAddressSpace.newAddressPool(ifName, priority, network)
if err != nil {
logger.Error("Failed to create pool", zap.Any("subnet", subnet), zap.String("ifName", ifName), zap.Error(err))
continue
}
// Add the IP addresses to the localAddressSpace address space.
for _, ipAddr := range subnet.IPAddresses {
// Primary addresses are reserved for the host.
if ipAddr.IsPrimary {
continue
}
address := net.ParseIP(ipAddr.Address)
_, err = addressPool.newAddressRecord(&address)
if err != nil {
logger.Error("Failed to create", zap.Any("address", address), zap.Error(err))
continue
}
}
}
}
return nil
}
func macAddressesEqual(macAddress1 string, macAddress2 string) bool {
macAddress1 = strings.ToLower(strings.Replace(macAddress1, ":", "", -1))
macAddress2 = strings.ToLower(strings.Replace(macAddress2, ":", "", -1))
return macAddress1 == macAddress2
}