Я использую базу данных IP2Location, чтобы найти коды стран для адресов IPv6. У них есть метод преобразовать адрес IPv6 в (большое) число, которое можно использовать для запроса их базы данных.
$ipv6 = '2404:6800:4001:805::1006';
$int = inet_pton($ipv6);
$bits = 15;
$ipv6long = 0;
while($bits >= 0){
$bin = sprintf("%08b", (ord($int[$bits])));
if($ipv6long){
$ipv6long = $bin . $ipv6long;
}
else{
$ipv6long = $bin;
}
$bits--;
}
$ipv6long = gmp_strval(gmp_init($ipv6long, 2), 10);
В этом случае, $ipv6long
будет 47875086426098177934326549022813196294.
Теперь мне интересно, можно ли вернуть такое число в строковое представление адреса IPv6. И если да, то как?
inet_ntop()
Можно форматировать адреса IPv6, но сначала нужно преобразовать в упакованную строку (строка из 16 символов, где каждый символ равен одному байту числа).
function ipv6number2string($number) {
// convert to hex
$hex = gmp_strval(gmp_init($number, 10), 16);
// pad to 32 chars
$hex = str_pad($hex, 32, '0', STR_PAD_LEFT);
// convert to a binary string
$packed = hex2bin($hex);
// convert to IPv6 string
return inet_ntop($packed);
}
echo ipv6number2string(47875086426098177934326549022813196294);
Чтобы ответить на мой собственный вопрос: это будет работать:
function ipv6number2string($number) {
// thanks to joost @ http://php.net/manual/en/function.dechex.php
$hexvalues = array('0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f');
$hexval = '';
while($number != '0') {
$hexval = $hexvalues[bcmod($number,'16')].$hexval;
$number = bcdiv($number,'16',0);
}
// now format it with colons
$str = '';
preg_replace_callback('/([a-f0-9]{4})/', function($m) use (&$str) {
if (empty($str)) {
$str = is_numeric($m[0]) ? intval($m[0]) : $m[0];
} else {
$str .= ':' . (is_numeric($m[0]) ? intval($m[0]) : $m[0]);
}
}, $hexval);
return preg_replace(array('/:0/', '/:{3,}/'), '::', $str);
}
echo ipv6number2string(47875086426098177934326549022813196294);
Покажет 2404: 6800: 4001: 805 :: 1006.