1
|
|
|
<?php |
2
|
|
|
|
3
|
|
|
namespace IpNetCalc; |
4
|
|
|
|
5
|
|
|
/** |
6
|
|
|
* Class IpNetCalc |
7
|
|
|
* |
8
|
|
|
* Compute the common mask from multiple IP addresses |
9
|
|
|
* |
10
|
|
|
* @author Alexander Over <[email protected]> |
11
|
|
|
*/ |
12
|
|
|
class IpNetCalc |
13
|
|
|
{ |
14
|
|
|
/** |
15
|
|
|
* @param array $ips |
16
|
|
|
* @return string |
17
|
|
|
*/ |
18
|
4 |
|
public function calcNetSum(array $ips) |
19
|
|
|
{ |
20
|
4 |
|
if (count($ips) < 2) { |
21
|
|
|
throw new \InvalidArgumentException('too few arguments'); |
22
|
|
|
} |
23
|
|
|
|
24
|
4 |
|
$validated = $n = $s = []; |
25
|
4 |
|
$isV4 = $isV6 = false; |
26
|
|
|
|
27
|
4 |
|
foreach ($ips as $ip) { |
28
|
4 |
|
if (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4) && !$isV6) { |
29
|
2 |
|
$isV4 = true; |
30
|
2 |
|
$validated[] = $ip; |
31
|
4 |
|
} elseif (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_IPV6) && !$isV4) { |
32
|
1 |
|
$isV6 = true; |
33
|
1 |
|
$validated[] = $ip; |
34
|
1 |
|
} else { |
|
|
|
|
35
|
1 |
|
throw new \InvalidArgumentException('mixing of IPs not allowed'); |
36
|
|
|
} |
37
|
3 |
|
} |
38
|
|
|
|
39
|
3 |
|
$mask = (($isV4) ? 32 : 128); |
40
|
3 |
|
$ipType = (($isV4) ? 4 : 6); |
41
|
|
|
|
42
|
3 |
|
asort($validated); |
43
|
|
|
|
44
|
|
|
// we need only smalest and biggest ip |
45
|
|
|
$compare = [ |
46
|
3 |
|
array_shift($validated), |
47
|
3 |
|
array_pop($validated) |
48
|
3 |
|
]; |
49
|
|
|
|
50
|
3 |
|
foreach ($compare as $key => $ip) { |
51
|
3 |
|
$s[$key] = implode('', $this->bitCalcIP($ip, $ipType)); |
52
|
3 |
|
} |
53
|
|
|
|
54
|
3 |
|
$t = ''; |
55
|
|
|
|
56
|
3 |
|
if ($s[0] === $s[1]) { |
57
|
1 |
|
$t = $s[0]; |
58
|
1 |
|
$i = $mask; |
59
|
1 |
|
} else { |
|
|
|
|
60
|
2 |
|
$o = ''; |
61
|
2 |
|
for ($i = 0, $len = strlen($s[0]); $i < $len; $i++) { |
62
|
2 |
|
if (substr($s[0], $i, 1) == substr($s[1], $i, 1)) { |
63
|
2 |
|
$o .= substr($s[0], $i, 1); |
64
|
2 |
|
} else { |
|
|
|
|
65
|
2 |
|
$t = str_pad($o, $mask, 0, STR_PAD_RIGHT); |
66
|
2 |
|
break; |
67
|
|
|
} |
68
|
2 |
|
} |
69
|
|
|
} |
70
|
|
|
|
71
|
3 |
|
$q = str_split($t, 8); |
72
|
|
|
|
73
|
3 |
|
if ($isV4) { |
74
|
2 |
|
foreach ($q as $b) { |
75
|
2 |
|
$n[] = bindec($b); |
76
|
2 |
|
} |
77
|
2 |
|
$n = implode('.', $n); |
78
|
2 |
|
} else { |
|
|
|
|
79
|
1 |
|
for ($j = 0, $len = count($q); $j < $len; $j += 2) { |
80
|
1 |
|
$n[$j] = dechex(bindec($q[$j])) . |
81
|
1 |
|
str_pad(dechex(bindec($q[$j + 1])), 2, 0, STR_PAD_LEFT); |
82
|
1 |
|
} |
83
|
1 |
|
$n = inet_ntop(inet_pton(implode(':', $n))); |
84
|
|
|
} |
85
|
|
|
|
86
|
3 |
|
return $n . '/' . $i; |
87
|
|
|
} |
88
|
|
|
|
89
|
|
|
/** |
90
|
|
|
* @param string $ip |
91
|
|
|
* @param integer $ipType |
92
|
|
|
* |
93
|
|
|
* @return array |
94
|
|
|
*/ |
95
|
3 |
|
private function bitCalcIP($ip, $ipType) |
96
|
|
|
{ |
97
|
3 |
|
$r = []; |
98
|
|
|
|
99
|
3 |
|
if (6 == $ipType) { |
100
|
1 |
|
$e = $this->handleV6($ip); |
101
|
1 |
|
} else { |
|
|
|
|
102
|
2 |
|
$e = explode('.', $ip); |
103
|
|
|
} |
104
|
|
|
|
105
|
3 |
|
foreach ($e as $b) { |
106
|
3 |
|
$r[] = str_pad(decbin($b), 8, 0, STR_PAD_LEFT); |
107
|
3 |
|
} |
108
|
|
|
|
109
|
3 |
|
return $r; |
110
|
|
|
} |
111
|
|
|
|
112
|
|
|
/** |
113
|
|
|
* @param string $ip |
114
|
|
|
* @return array |
115
|
|
|
*/ |
116
|
1 |
|
private function handleV6($ip) |
117
|
|
|
{ |
118
|
1 |
|
$n = []; |
119
|
1 |
|
$unpack = unpack('H*', inet_pton($ip)); |
120
|
1 |
|
$e = str_split($unpack[1], 4); |
121
|
1 |
|
for ($i = 0; $i < 8; $i++) { |
122
|
1 |
|
$n[] = hexdec(substr($e[$i], 0, 2)); |
123
|
1 |
|
$n[] = hexdec(substr($e[$i], 2, 2)); |
124
|
1 |
|
} |
125
|
1 |
|
return $n; |
126
|
|
|
} |
127
|
|
|
} |
128
|
|
|
|