128 || ($newmask > 32 && strpos($newip, ':') === FALSE)) { return FALSE; } $new_str = $newip . "/" . $mask; return TRUE; } return FALSE; } public static function calc_cidr_range($str, $as_hex = false) { if(self::validate_cidr($str, $str)) { list($ip, $prefix) = explode('/', $str); } elseif (filter_var($str, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4)) { $ip = $str; $prefix = 32; } elseif (filter_var($str, FILTER_VALIDATE_IP, FILTER_FLAG_IPV6)) { $ip = $str; $prefix = 128; } else { return FALSE; } $ip_bytes = unpack("C*", inet_pton($ip)); $total_bytes = count($ip_bytes); $num_diff_bits = 8 * $total_bytes - $prefix; if ($num_diff_bits >= 0) { $num_same_bytes = $prefix >> 3; $same_bytes = array_slice($ip_bytes, 0, $num_same_bytes); $diff_bytes_start = ($total_bytes === $num_same_bytes) ? array() : array_fill(0, $total_bytes - $num_same_bytes, 0); $diff_bytes_end = ($total_bytes === $num_same_bytes) ? array() : array_fill(0, $total_bytes - $num_same_bytes, 255); $start_same_bits = $prefix % 8; if ($start_same_bits !== 0) { $vary_byte = $ip_bytes[$num_same_bytes]; $diff_bytes_start[0] = $vary_byte & bindec(str_pad(str_repeat('1', $start_same_bits), 8, '0', STR_PAD_RIGHT)); $diff_bytes_end[0] = $diff_bytes_start[0] + bindec(str_repeat('1', 8 - $start_same_bits)); } $start_array = array_merge($same_bytes, $diff_bytes_start); $end_array = array_merge($same_bytes, $diff_bytes_end); if ($as_hex) { if ($total_bytes < 16) { $start_array = array_merge(array_fill(0, 16 - $total_bytes, 0), $start_array); $end_array = array_merge(array_fill(0, 16 - $total_bytes, 0), $end_array); } $start = unpack('H*hex', join(array_map('chr', $start_array)))['hex']; $end = unpack('H*hex', join(array_map('chr', $end_array)))['hex']; return array($start, $end); } else { $start = call_user_func_array('pack', array_merge(array("C*"), $start_array)); $end = call_user_func_array('pack', array_merge(array("C*"), $end_array)); return array($start, $end); } } return FALSE; } public static function ip_to_hex_str($ip) { if (!self::validate_ip($ip)) { return false; } $ip_bytes = unpack("C*", inet_pton($ip)); $total_bytes = count($ip_bytes); if ($total_bytes < 16) { $ip_bytes = array_merge(array_fill(0, 16 - $total_bytes, 0), $ip_bytes); } return unpack('H*hex', join(array_map('chr', $ip_bytes)))['hex']; } public static function check_ip($requestIp, $ips) { if (!self::validate_ip($requestIp)) { return false; } if (!\is_array($ips)) { $ips = [$ips]; } $method = substr_count($requestIp, ':') > 1 ? 'check_ip6' : 'check_ip4'; foreach ($ips as $ip) { if (self::$method($requestIp, $ip)) { return true; } } return false; } public static function check_ip6($requestIp, $ip) { if (false !== strpos($ip, '/')) { list($address, $netmask) = explode('/', $ip, 2); if ('0' === $netmask) { return (bool) unpack('n*', @inet_pton($address)); } if ($netmask < 1 || $netmask > 128) { return false; } } else { $address = $ip; $netmask = 128; } $bytesAddr = unpack('n*', @inet_pton($address)); $bytesTest = unpack('n*', @inet_pton($requestIp)); if (!$bytesAddr || !$bytesTest) { return false; } for ($i = 1, $ceil = ceil($netmask / 16); $i <= $ceil; ++$i) { $left = $netmask - 16 * ($i - 1); $left = ($left <= 16) ? $left : 16; $mask = ~(0xffff >> $left) & 0xffff; if (($bytesAddr[$i] & $mask) != ($bytesTest[$i] & $mask)) { return false; } } return true; } public static function check_ip4($requestIp, $ip) { if (false !== strpos($ip, '/')) { list($address, $netmask) = explode('/', $ip, 2); if ('0' === $netmask) { return false; } if ($netmask < 0 || $netmask > 32) { return false; } } else { $address = $ip; $netmask = 32; } if (false === ip2long($address)) { return false; } return 0 === substr_compare(sprintf('%032b', ip2long($requestIp)), sprintf('%032b', ip2long($address)), 0, $netmask); } public static function convert($ip) { global $_G; if (false !== strpos($ip, '/')) { list($ip, $netmask) = explode('/', $ip, 2); } if(!self::validate_ip($ip)) { return '- Invalid'; } if (!(filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE) !== false)) { return '- LAN'; } if (!(filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_RES_RANGE) !== false)) { return '- Reserved'; } if (array_key_exists('ipdb', $_G['config']) && array_key_exists('setting', $_G['config']['ipdb'])) { $s = $_G['config']['ipdb']['setting']; if (!empty($s['fullstack'])) { $c = 'ip_'.$s['fullstack']; } else if (!empty($s['ipv4']) && filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4) !== false) { $c = 'ip_'.$s['ipv4']; } else if (!empty($s['ipv6']) && filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_IPV6) !== false) { $c = 'ip_'.$s['ipv6']; } else if (!empty($s['default'])) { $c = 'ip_'.$s['default']; } else { $c = 'ip_tiny'; } } else { $c = 'ip_tiny'; } $ipobject = $c::getInstance(); return $ipobject === NULL ? '- Error' : $ipobject->convert($ip); } public static function checkaccess($ip, $accesslist) { return preg_match("/^(".str_replace(array("\r\n", ' '), array('|', ''), preg_quote($accesslist, '/')).")/", $ip); } public static function checkbanned($ip) { global $_G; if (array_key_exists('security', $_G['config']) && array_key_exists('useipban', $_G['config']['security']) && $_G['config']['security']['useipban'] == 0) { return false; } if($_G['setting']['ipaccess'] && !self::checkaccess($ip, $_G['setting']['ipaccess'])) { return true; } return C::t('common_banned')->check_banned(TIMESTAMP, $ip); } } ?>