-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsubnetmath.go
316 lines (291 loc) · 8.98 KB
/
subnetmath.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
package subnetmath
import (
"math"
"math/big"
"net"
)
// commonly used bigint values
var bigZero = big.NewInt(0)
var bigOne = big.NewInt(1)
var bigTwo = big.NewInt(2)
// ParseNetworkCIDR is a convienence function that will return either the *net.IPNet
// or nil if the supplied cidr is invalid
func ParseNetworkCIDR(cidr string) *net.IPNet {
addr, network, err := net.ParseCIDR(cidr)
if err == nil && network != nil && network.IP.Equal(addr) {
return network
}
return nil
}
// NetworksAreIdentical returns a bool with regards to the two networks being equal
func NetworksAreIdentical(first, second *net.IPNet) bool {
if first != second {
if first.IP.Equal(second.IP) {
firstSize, _ := first.Mask.Size()
secondSize, _ := second.Mask.Size()
if firstSize == secondSize {
return true
}
}
return false
}
return true
}
// NetworkComesBefore returns a bool with regards to numerical network order.
// Note that IPv4 networks come before IPv6 networks.
func NetworkComesBefore(first, second *net.IPNet) bool {
if first != nil && second != nil {
if first.IP.Equal(second.IP) {
firstMask, _ := first.Mask.Size()
secondMask, _ := second.Mask.Size()
if firstMask < secondMask {
return true
}
return false
}
return AddressComesBefore(first.IP, second.IP)
}
return false
}
// AddressComesBefore returns a bool with regards to numerical address order.
// Note that IPv4 addresses come before IPv6 addresses.
func AddressComesBefore(firstIP, secondIP net.IP) bool {
if firstIP.To4() == nil && secondIP.To4() != nil {
return false
} else if firstIP.To4() != nil && secondIP.To4() == nil {
return true
}
if AddrToInt(firstIP).Cmp(AddrToInt(secondIP)) < 0 {
return true
}
return false
}
// DuplicateNetwork returns a new copy of *net.IPNet
func DuplicateNetwork(network *net.IPNet) *net.IPNet {
if network != nil {
newIP := make(net.IP, len(network.IP))
newMask := make(net.IPMask, len(network.Mask))
copy(newIP, network.IP)
copy(newMask, network.Mask)
return &net.IPNet{
IP: newIP,
Mask: newMask,
}
}
return nil
}
// DuplicateAddr creates a new copy of net.IP
func DuplicateAddr(addr net.IP) net.IP {
newIP := make(net.IP, len(addr))
copy(newIP, addr)
return newIP
}
// SubnetZeroAddr returns the subnet zero address
func SubnetZeroAddr(address net.IP, network *net.IPNet) net.IP {
if network != nil {
return address.Mask(network.Mask)
}
return nil
}
// NextNetwork returns the next network of the same size
func NextNetwork(network *net.IPNet) *net.IPNet {
if network != nil {
nextNetwork := DuplicateNetwork(network)
networkInt := AddrToInt(network.IP)
networkInt.Add(networkInt, addressCount(network))
nextNetwork.IP = IntToAddr(networkInt)
return nextNetwork
}
return nil
}
// BroadcastAddr returns the broadcast address
func BroadcastAddr(network *net.IPNet) net.IP {
if network != nil {
networkInt := AddrToInt(network.IP)
networkInt.Add(networkInt, addressCount(network))
networkInt.Sub(networkInt, bigOne)
return IntToAddr(networkInt)
}
return nil
}
// NextAddr returns a new net.IP that is the next address
func NextAddr(addr net.IP) net.IP {
addrInt := AddrToInt(addr)
addrInt.Add(addrInt, bigOne)
return IntToAddr(addrInt)
}
func addressCount(network *net.IPNet) *big.Int {
if network != nil {
ones, bits := network.Mask.Size()
if bits <= 32 {
return new(big.Int).SetInt64(int64(math.Exp2(float64(bits - ones))))
}
return new(big.Int).Exp(bigTwo, big.NewInt(int64(bits-ones)), nil)
}
return nil
}
// ShrinkNetwork increases the mask size by one
func ShrinkNetwork(network *net.IPNet) *net.IPNet {
if network != nil {
ones, bits := network.Mask.Size()
if ones < bits {
network.Mask = net.CIDRMask(ones+1, bits)
return network
}
}
return nil
}
func sameAddrType(first, second net.IP) bool {
alpha, bravo := first.To4(), second.To4()
if alpha == nil && bravo == nil || alpha != nil && bravo != nil {
return true
}
return false
}
func maskBitLength(address net.IP) int {
if address.To4() != nil {
return 32
}
return 128
}
// FindInbetweenSubnets returns a slice of subnets given a range of IP addresses.
// Note that the delimiter 'stop' is inclusive. In other words, it will be included in the result.
func FindInbetweenSubnets(start, stop net.IP) []*net.IPNet {
if sameAddrType(start, stop) && AddressComesBefore(start, stop) {
var subnets []*net.IPNet
maskBits := maskBitLength(start)
current := DuplicateAddr(start)
stopInt := AddrToInt(stop)
for {
currentSubnet := &net.IPNet{IP: current}
for ones := 1; ones <= maskBits; ones++ {
currentSubnet.Mask = net.CIDRMask(ones, maskBits)
increment := addressCount(currentSubnet)
addressInt := AddrToInt(currentSubnet.IP)
addressInt.Add(addressInt, increment)
addressInt.Sub(addressInt, bigOne)
if addressInt.Cmp(stopInt) > 0 {
continue
}
if SubnetZeroAddr(currentSubnet.IP, currentSubnet).Equal(currentSubnet.IP) {
break
}
}
subnets = append(subnets, currentSubnet)
current = NextNetwork(currentSubnet).IP
if !current.Equal(stop) && AddressComesBefore(stop, current) ||
AddressComesBefore(current, start) {
break
}
}
return subnets
}
return nil
}
func isValidSubnet(network *net.IPNet) bool {
return SubnetZeroAddr(network.IP, network).Equal(network.IP)
}
func findNetworkIntersection(network *net.IPNet, otherNetworks ...*net.IPNet) *net.IPNet {
// OPTIMIZE: would having the subnets sorted allow for substantial shortcuts?
for _, otherNetwork := range otherNetworks {
if network.Contains(otherNetwork.IP) || otherNetwork.Contains(network.IP) {
return otherNetwork
}
}
return nil
}
func findNetworkWithoutIntersection(currentNetwork *net.IPNet, otherNetworks ...*net.IPNet) {
// BUG: the logic here needs major rework, lots of performance left on the table
allZeroMask := make(net.IPMask, len(currentNetwork.Mask))
copy(currentNetwork.Mask, allZeroMask)
var lastIntersect *net.IPNet
for {
// shrink the network cidr by one
nextNetwork := ShrinkNetwork(currentNetwork)
if nextNetwork == nil {
// We trying to shrink a network size of one. This address is completely unusable.
// Try again with a new address offset from the lastIntersection
lastIntersectAddr := AddrToInt(lastIntersect.IP)
lastIntersectAddr.Add(lastIntersectAddr, addressCount(lastIntersect))
currentNetwork.IP = IntToAddr(lastIntersectAddr)
currentNetwork.Mask = allZeroMask
} else {
// search through all otherNetworks trying to find an intersection
lastIntersect = findNetworkIntersection(currentNetwork, otherNetworks...)
// return this network if there is no intersection and our network is valid
if lastIntersect == nil && isValidSubnet(currentNetwork) {
return
}
}
}
}
// NetworkContainsSubnet validates that the network is a valid supernet
func NetworkContainsSubnet(network *net.IPNet, subnet *net.IPNet) bool {
if network != nil && subnet != nil {
supernetInt := AddrToInt(network.IP)
subnetInt := AddrToInt(subnet.IP)
if supernetInt.Cmp(subnetInt) <= 0 {
supernetInt.Add(supernetInt, addressCount(network))
subnetInt.Add(subnetInt, addressCount(subnet))
if supernetInt.Cmp(subnetInt) >= 0 {
return true
}
}
}
return false
}
// FindUnusedSubnets returns a slice of unused subnets given the aggregate and sibling subnets
func FindUnusedSubnets(aggregate *net.IPNet, subnets ...*net.IPNet) (unused []*net.IPNet) {
nextSubnet := DuplicateNetwork(aggregate)
if len(subnets) > 0 && findNetworkIntersection(aggregate, subnets...) != nil {
for {
findNetworkWithoutIntersection(nextSubnet, subnets...)
if NetworkContainsSubnet(aggregate, nextSubnet) {
unused = append(unused, nextSubnet)
nextSubnet = NextNetwork(nextSubnet)
continue
}
return unused
}
}
return append(unused, nextSubnet)
}
// IntToAddr will return the net.IP of the big.Int represented address
func IntToAddr(intAddress *big.Int) net.IP {
intBytes := intAddress.Bytes()
if len(intBytes) == 4 {
return net.IPv4(intBytes[0], intBytes[1], intBytes[2], intBytes[3])
}
return intBytes
}
// AddrToInt will return the *bit.Int of a given IPv6 address
func AddrToInt(address net.IP) *big.Int {
v4addr := address.To4()
if v4addr != nil {
return new(big.Int).SetBytes(v4addr)
}
return new(big.Int).SetBytes(address.To16())
}
// IPv4ClassfulNetwork eithers return the classful network given an IPv4 address or
// returns nil if given a multicast address or IPv6 address
func IPv4ClassfulNetwork(address net.IP) *net.IPNet {
if address.To4() != nil {
var newIP net.IP
var newMask net.IPMask
switch {
case uint8(address[0]) < 128:
newIP = net.IPv4(uint8(address[0]), 0, 0, 0)
newMask = net.IPv4Mask(255, 0, 0, 0)
case uint8(address[0]) < 192:
newIP = net.IPv4(uint8(address[0]), uint8(address[1]), 0, 0)
newMask = net.IPv4Mask(255, 255, 0, 0)
case uint8(address[0]) < 224:
newIP = net.IPv4(uint8(address[0]), uint8(address[1]), uint8(address[2]), 0)
newMask = net.IPv4Mask(255, 255, 255, 0)
default:
return nil
}
return &net.IPNet{IP: newIP, Mask: newMask}
}
return nil
}