Я застрял на, надеюсь, простую задачу: я хочу получить следующий IP-адрес.
Вот некоторые из моих тестов:
//$binaryIp = inet_pton('192.168.1.1');
$binaryIp = inet_pton('2001:cdba::1');
$verySimple = inet_ntop(
$binaryIp++
);
var_dump($verySimple); //'2001:cdba::1'
$simpleMaths = inet_ntop(
$binaryIp + inet_pton('0.0.0.1')
);
var_dump($simpleMaths); //inet_ntop(): Invalid in_addr value
$aLittleBitOfSuccess = long2ip(
ip2long(inet_ntop($binaryIp)) + 1
);
var_dump($aLittleBitOfSuccess); //'0.0.0.1' but with IPv4 '192.168.1.2'
Хорошо, пока здесь не очевидно, что мои попытки — скорее глупость, чем реальный подход к моей проблеме, но что еще я могу попробовать? Я искал в Интернете и нашел несколько решений для подсетей и тому подобное, но ничего для простого сложения или вычитания.
Моей следующей попыткой было бы разделить строку из inet_ntop () и возиться с шестнадцатеричными значениями, но должно быть простое решение, чтобы добавить 1 в in6_addr!
За некоторыми исключениями, адрес IPv6 делится на две 64-битные части: сеть / подсеть и идентификатор интерфейса. Вы должны быть заинтересованы в 64 битах идентификатора интерфейса.
Проще всего разобрать адрес в два 64-битных целых числа без знака, увеличить идентификатор интерфейса, а затем объединить две части в 128-битный адрес.
Я выбрал шестнадцатеричный путь и сделал эту функцию:
protected function binaryIncrement($binaryIp, $increment = 1) {
//inet_pton creates values where each "character" is one ip-address-byte
//we are splitting the string so we can handle every byte for itselve.
$binaryIpArrayIn = str_split($binaryIp);
$binaryIpArrayOut = array();
$carry = 0 + $increment;
//reverse array because our following addition is done from right to left.
foreach (array_reverse($binaryIpArrayIn) as $binaryByte) {
//transforming on byte from our ip address to decimal
$decIp = hexdec(bin2hex($binaryByte));
$tempValue = $decIp + $carry;
$tempValueHex = dechex($tempValue);
//check if we have to deal with a carry
if (strlen($tempValueHex) > 2) {
//split $tempValueHex in carry and result
//str_pad because hex2bin only accepts even character counts
$carryHex = str_pad(substr($tempValueHex,0,1),2,'0',STR_PAD_LEFT);
$tempResultHex = str_pad(substr($tempValueHex,1,2),2,'0',STR_PAD_LEFT);
$carry = hexdec($carryHex);
} else {
$carry = 0;
$tempResultHex = str_pad($tempValueHex,2,'0',STR_PAD_LEFT);
}
//fill our result array
$binaryIpArrayOut[] = hex2bin($tempResultHex);
}
//we have to reverse our arry back to normal order and building a string
$binaryIpOut = implode(array_reverse($binaryIpArrayOut));
return $binaryIpOut;
}
$binaryIpV6In = inet_pton('2001:cdba::FFFF');
$binaryIpV6Out = $this->binaryIncrement($binaryIpV6In);
var_dump(inet_ntop($binaryIpV6Out));
$binaryIpV4In = inet_pton('192.168.1.1');
$binaryIpV4Out = $this->binaryIncrement($binaryIpV4In, 256);
var_dump(inet_ntop($binaryIpV4Out));
Таким образом, я могу использовать один и тот же метод для IPv4 и IPv6.