// Package ipamutils provides utility functions for ipam management package ipamutils import ( "fmt" "net" "sync" "github.com/sirupsen/logrus" ) var ( // PredefinedBroadNetworks contains a list of 31 IPv4 private networks with host size 16 and 12 // (172.17-31.x.x/16, 192.168.x.x/20) which do not overlap with the networks in `PredefinedGranularNetworks` PredefinedBroadNetworks []*net.IPNet // PredefinedGranularNetworks contains a list of 64K IPv4 private networks with host size 8 // (10.x.x.x/24) which do not overlap with the networks in `PredefinedBroadNetworks` PredefinedGranularNetworks []*net.IPNet initNetworksOnce sync.Once defaultBroadNetwork = []*NetworkToSplit{{"172.17.0.0/16", 16}, {"172.18.0.0/16", 16}, {"172.19.0.0/16", 16}, {"172.20.0.0/14", 16}, {"172.24.0.0/14", 16}, {"172.28.0.0/14", 16}, {"192.168.0.0/16", 20}} defaultGranularNetwork = []*NetworkToSplit{{"10.0.0.0/8", 24}} ) // NetworkToSplit represent a network that has to be split in chunks with mask length Size. // Each subnet in the set is derived from the Base pool. Base is to be passed // in CIDR format. // Example: a Base "10.10.0.0/16 with Size 24 will define the set of 256 // 10.10.[0-255].0/24 address pools type NetworkToSplit struct { Base string `json:"base"` Size int `json:"size"` } // InitNetworks initializes the broad network pool and the granular network pool func InitNetworks(defaultAddressPool []*NetworkToSplit) { initNetworksOnce.Do(func() { // error ingnored should never fail PredefinedGranularNetworks, _ = splitNetworks(defaultGranularNetwork) if defaultAddressPool == nil { defaultAddressPool = defaultBroadNetwork } var err error if PredefinedBroadNetworks, err = splitNetworks(defaultAddressPool); err != nil { logrus.WithError(err).Error("InitAddressPools failed to initialize the default address pool") } }) } // splitNetworks takes a slice of networks, split them accordingly and returns them func splitNetworks(list []*NetworkToSplit) ([]*net.IPNet, error) { localPools := make([]*net.IPNet, 0, len(list)) for _, p := range list { _, b, err := net.ParseCIDR(p.Base) if err != nil { return nil, fmt.Errorf("invalid base pool %q: %v", p.Base, err) } ones, _ := b.Mask.Size() if p.Size <= 0 || p.Size < ones { return nil, fmt.Errorf("invalid pools size: %d", p.Size) } localPools = append(localPools, splitNetwork(p.Size, b)...) } return localPools, nil } func splitNetwork(size int, base *net.IPNet) []*net.IPNet { one, bits := base.Mask.Size() mask := net.CIDRMask(size, bits) n := 1 << uint(size-one) s := uint(bits - size) list := make([]*net.IPNet, 0, n) for i := 0; i < n; i++ { ip := copyIP(base.IP) addIntToIP(ip, uint(i<= 0; i-- { array[i] |= (byte)(ordinal & 0xff) ordinal >>= 8 } }