Соответствие адреса IPv6 подсети CIDR - PullRequest
7 голосов
/ 31 октября 2011

Есть ли хороший способ сопоставить адрес IPv6 с подсетью IPv6, используя нотацию CIDR?То, что я ищу, это эквивалент IPv6: Сопоставление IP с маской CIDR в PHP 5?

Приведенный выше пример не может быть использован, поскольку адрес IPv6 равен 128биты, предотвращающие правильную работу побитового сдвига влево.Можете ли вы придумать другой способ?

РЕДАКТИРОВАТЬ: добавил мое собственное решение в список ответов.

Ответы [ 7 ]

13 голосов
/ 31 октября 2011

Поскольку вы не можете преобразовать адреса IPv6 в целое число, вы должны использовать биты, например:

$ip='21DA:00D3:0000:2F3B:02AC:00FF:FE28:9C5A';
$cidrnet='21DA:00D3:0000:2F3B::/64';

// converts inet_pton output to string with bits
function inet_to_bits($inet) 
{
   $unpacked = unpack('A16', $inet);
   $unpacked = str_split($unpacked[1]);
   $binaryip = '';
   foreach ($unpacked as $char) {
             $binaryip .= str_pad(decbin(ord($char)), 8, '0', STR_PAD_LEFT);
   }
   return $binaryip;
}    

$ip = inet_pton($ip);
$binaryip=inet_to_bits($ip);

list($net,$maskbits)=explode('/',$cidrnet);
$net=inet_pton($net);
$binarynet=inet_to_bits($net);

$ip_net_bits=substr($binaryip,0,$maskbits);
$net_bits   =substr($binarynet,0,$maskbits);

if($ip_net_bits!==$net_bits) echo 'Not in subnet';
else echo 'In subnet';

Кроме того, если вы используете некоторую базу данных для хранения IP-адресов, она может уже иметь все функции для их сравнения,Например, Postgres имеет тип inet и может определять, содержится ли IP в подсети, например:

SELECT 
   '21DA:00D3:0000:2F3B:02AC:00FF:FE28:9C5A'::inet << 
   '21DA:00D3:0000:2F3B::/64'::inet;

9.11.Функции и операторы сетевых адресов в PostgreSQL

6 голосов
/ 22 марта 2016

Вы также можете использовать класс IpUtils из пакета symfony / http-foundation :

IpUtils::checkIp6('2a01:8760:2:3001::1', '2a01:8760:2:3001::1/64')

Это позволит проверить правильность IPv6 и соответствие диапазона,Вернет false, если это не так.

3 голосов
/ 31 октября 2011

Я создал собственное решение, используя следующий код:

function iPv6MaskToByteArray($subnetMask) {
  $addr = str_repeat("f", $subnetMask / 4);
  switch ($subnetMask % 4) {
    case 0:
      break;
    case 1:
      $addr .= "8";
      break;
    case 2:
      $addr .= "c";
      break;
    case 3:
      $addr .= "e";
      break;
  }
  $addr = str_pad($addr, 32, '0');
  $addr = pack("H*" , $addr);
  return $addr;
}

function iPv6CidrMatch($address, $subnetAddress, $subnetMask) {
  $binMask = iPv6MaskToByteArray($subnetMask);
  return ($address & $binMask) == $subnetAddress;
}

Обратите внимание, что $ address и $ subnetAddress были получены путем запуска строкового адреса через inet_pton.Вызовите функцию следующим образом:

$subnet = inet_pton("2001:06b8::");
$mask = 32;
$addr = inet_pton("2001:06b8:0000:0000:0000:0000:1428:07ab");
$match = iPv6CidrMatch($addr, $subnet, $mask); // TRUE
1 голос
/ 09 мая 2019

PHP может выполнять побитовые операции со строками!

  • IPv4 или IPv6
  • Нет базового преобразования
  • Нет битовых строк ASCII
  • Довольно быстро
<?php

/**
 * Does the given IP match the CIDR prefix?
 */
function matchIp(string $ip, string $cidr): bool
{
  // Get mask bits
  list($net, $maskBits) = explode('/', $cidr);

  // Size
  $size = (strpos($ip, ':') === false) ? 4 : 16;

  // Convert to binary
  $ip = inet_pton($ip);
  $net = inet_pton($net);
  if (!$ip || !$net) {
    throw new InvalidArgumentException('Invalid IP address');
  }

  // Build mask
  $solid = floor($maskBits / 8);
  $solidBits = $solid * 8;
  $mask = str_repeat(chr(255), $solid);
  for ($i = $solidBits; $i < $maskBits; $i += 8) {
    $bits = max(0, min(8, $maskBits - $i));
    $mask .= chr((pow(2, $bits) - 1) << (8 - $bits));
  }
  $mask = str_pad($mask, $size, chr(0));

  // Compare the mask
  return ($ip & $mask) === ($net & $mask);
}

1 голос
/ 21 февраля 2017

Вот пример, который работает путем проверки IP-адреса по списку отдельных IP-адресов или CIDR, как IPv4, так и IPv6:

https://gist.github.com/lyquix-owner/2620da22d927c99d57555530aab3279b

<?php
// IP to check
$ip_check = $_SERVER['REMOTE_ADDR'];

// Array of allowed IPs and subnets, both IPv4 and IPv6
$ips_allowed = array(
    '192.30.252.0/22'
    '2620:112:3000::/44'
    '192.168.16.104'
);

// Flag for IP match allowed list
$ip_match = false;

foreach($ips_allowed as $ip_allow) {
    // If IP has / means CIDR notation
    if(strpos($ip_allow, '/') === false) {
        // Check Single IP
        if(inet_pton($ip_check) == inet_pton($ip_allow)) {
            $allow = true;
            break;
        }
    }
    else {
        // Check IP range
        list($subnet, $bits) = explode('/', $ip_allow);

        // Convert subnet to binary string of $bits length
        $subnet = unpack('H*', inet_pton($subnet)); // Subnet in Hex
        foreach($subnet as $i => $h) $subnet[$i] = base_convert($h, 16, 2); // Array of Binary
        $subnet = substr(implode('', $subnet), 0, $bits); // Subnet in Binary, only network bits

        // Convert remote IP to binary string of $bits length
        $ip = unpack('H*', inet_pton($ip_check)); // IP in Hex
        foreach($ip as $i => $h) $ip[$i] = base_convert($h, 16, 2); // Array of Binary
        $ip = substr(implode('', $ip), 0, $bits); // IP in Binary, only network bits

        // Check network bits match
        if($subnet == $ip) {
            $allow = true;
            break;
        }
    }
}
if(!$allow) {
    die('IP not allowed');
}
1 голос
/ 30 мая 2014

Если ваши маски всегда делятся на четыре (что довольно часто встречается в ipv6). Вы можете использовать:

function checkIPv6WithinRange($ipv6, $range) {
    list ($net, $mask) = preg_split("/\//", $range);

    if ($mask % 4)
        throw new NotImplementedException("Only masks divisible by 4 are supported");
    $stripChars = (128-$mask)/4;

    $hexNet = bin2hex(inet_pton($net));
    $reducedNet = substr($hexNet, 0, 0 - $stripChars);

    $hexIp = bin2hex(inet_pton($ipv6));
    $reducedIp = substr($hexIp, 0, 0 - $stripChars);

    return $reducedIp === $reducedNet;
}
0 голосов
/ 01 августа 2016

В основном http://www.phpclasses.org/browse/file/70429.html

Чтобы использовать это, просто позвоните

$cidr = new CIDR();
$cidr->match($ipv6, $ipv6_in_cidr);

Результат - «Хорошо».

...