Discuz 5.0 中读取纯真IP数据库函数分析
程序员文章站
2022-05-26 23:38:28
discuz 5.0 不在使用自己的ip数据,而是使用纯真ip的数据格式, 存取纯真ip数据库稍微有点麻烦,它的存储格式比较特殊也...
discuz 5.0 不在使用自己的ip数据,而是使用纯真ip的数据格式, 存取纯真ip数据库稍微有点麻烦,它的存储格式比较特殊也很有趣,具体的格式分析参考下面两个链接,其他语言实现参考文章末的链接。
《纯真ip数据库格式详解》
链接一:http://blog.csdn.net/heiyeshuwu/archive/2006/05/12/725675.aspx
链接二:http://lumaqq.linuxsir.org/article/qqwry_format_detail.html
纯真ip数据库官网:http://www.cz88.net/ip/
纯真ip数据库下载:http://update.cz88.net/soft/qqwry.rar
以下函数conrvertip()位于 discuz!5_gbk/upload/include/misc.func.php 路径中,有兴趣可以具体去阅读分析。(下面代码我做了简单的修改,更便于阅读,核心没有修改)
<?
//===================================
//
// 功能:ip地址获取真实地址函数
// 参数:$ip - ip地址
// 作者:[discuz!] (c) comsenz inc.
//
//===================================
function convertip($ip) {
//ip数据文件路径
$dat_path = 'qqwry.dat';
//检查ip地址
if(!preg_match("/^d{1,3}.d{1,3}.d{1,3}.d{1,3}$/", $ip)) {
return 'ip address error';
}
//打开ip数据文件
if(!$fd = @fopen($dat_path, 'rb')){
return 'ip date file not exists or access denied';
}
//分解ip进行运算,得出整形数
$ip = explode('.', $ip);
$ipnum = $ip[0] * 16777216 + $ip[1] * 65536 + $ip[2] * 256 + $ip[3];
//获取ip数据索引开始和结束位置
$databegin = fread($fd, 4);
$dataend = fread($fd, 4);
$ipbegin = implode('', unpack('l', $databegin));
if($ipbegin < 0) $ipbegin += pow(2, 32);
$ipend = implode('', unpack('l', $dataend));
if($ipend < 0) $ipend += pow(2, 32);
$ipallnum = ($ipend - $ipbegin) / 7 + 1;
$beginnum = 0;
$endnum = $ipallnum;
//使用二分查找法从索引记录中搜索匹配的ip记录
while($ip1num>$ipnum || $ip2num<$ipnum) {
$middle= intval(($endnum + $beginnum) / 2);
//偏移指针到索引位置读取4个字节
fseek($fd, $ipbegin + 7 * $middle);
$ipdata1 = fread($fd, 4);
if(strlen($ipdata1) < 4) {
fclose($fd);
return 'system error';
}
//提取出来的数据转换成长整形,如果数据是负数则加上2的32次幂
$ip1num = implode('', unpack('l', $ipdata1));
if($ip1num < 0) $ip1num += pow(2, 32);
//提取的长整型数大于我们ip地址则修改结束位置进行下一次循环
if($ip1num > $ipnum) {
$endnum = $middle;
continue;
}
//取完上一个索引后取下一个索引
$dataseek = fread($fd, 3);
if(strlen($dataseek) < 3) {
fclose($fd);
return 'system error';
}
$dataseek = implode('', unpack('l', $dataseek.chr(0)));
fseek($fd, $dataseek);
$ipdata2 = fread($fd, 4);
if(strlen($ipdata2) < 4) {
fclose($fd);
return 'system error';
}
$ip2num = implode('', unpack('l', $ipdata2));
if($ip2num < 0) $ip2num += pow(2, 32);
//没找到提示未知
if($ip2num < $ipnum) {
if($middle == $beginnum) {
fclose($fd);
return 'unknown';
}
$beginnum = $middle;
}
}
//下面的代码读晕了,没读明白,有兴趣的慢慢读
$ipflag = fread($fd, 1);
if($ipflag == chr(1)) {
$ipseek = fread($fd, 3);
if(strlen($ipseek) < 3) {
fclose($fd);
return 'system error';
}
$ipseek = implode('', unpack('l', $ipseek.chr(0)));
fseek($fd, $ipseek);
$ipflag = fread($fd, 1);
}
if($ipflag == chr(2)) {
$addrseek = fread($fd, 3);
if(strlen($addrseek) < 3) {
fclose($fd);
return 'system error';
}
$ipflag = fread($fd, 1);
if($ipflag == chr(2)) {
$addrseek2 = fread($fd, 3);
if(strlen($addrseek2) < 3) {
fclose($fd);
return 'system error';
}
$addrseek2 = implode('', unpack('l', $addrseek2.chr(0)));
fseek($fd, $addrseek2);
} else {
fseek($fd, -1, seek_cur);
}
while(($char = fread($fd, 1)) != chr(0))
$ipaddr2 .= $char;
$addrseek = implode('', unpack('l', $addrseek.chr(0)));
fseek($fd, $addrseek);
while(($char = fread($fd, 1)) != chr(0))
$ipaddr1 .= $char;
} else {
fseek($fd, -1, seek_cur);
while(($char = fread($fd, 1)) != chr(0))
$ipaddr1 .= $char;
$ipflag = fread($fd, 1);
if($ipflag == chr(2)) {
$addrseek2 = fread($fd, 3);
if(strlen($addrseek2) < 3) {
fclose($fd);
return 'system error';
}
$addrseek2 = implode('', unpack('l', $addrseek2.chr(0)));
fseek($fd, $addrseek2);
} else {
fseek($fd, -1, seek_cur);
}
while(($char = fread($fd, 1)) != chr(0)){
$ipaddr2 .= $char;
}
}
fclose($fd);
//最后做相应的替换操作后返回结果
if(preg_match('/http/i', $ipaddr2)) {
$ipaddr2 = '';
}
$ipaddr = "$ipaddr1 $ipaddr2";
$ipaddr = preg_replace('/cz88.net/is', '', $ipaddr);
$ipaddr = preg_replace('/^s*/is', '', $ipaddr);
$ipaddr = preg_replace('/s*$/is', '', $ipaddr);
if(preg_match('/http/i', $ipaddr) || $ipaddr == '') {
$ipaddr = 'unknown';
}
return $ipaddr;
}
//========================
//
// 调用举例(速度很快)
//
//========================
echo convertip('219.238.235.10');
//输出: 北京市 电信通
echo convertip('23.56.82.12');
//输出:iana
echo convertip('250.69.52.0');
//输出:iana保留地址
echo convertip('238.69.52.0');
//输出:iana保留地址 用于多点传送
echo convertip('192.168.0.1');
//输出:局域网 对方和您在同一内部网
echo convertip('255.255.255.255');
//输出:纯真网络 2006年11月20日ip数据
?>
《纯真ip数据库格式详解》
链接一:http://blog.csdn.net/heiyeshuwu/archive/2006/05/12/725675.aspx
链接二:http://lumaqq.linuxsir.org/article/qqwry_format_detail.html
纯真ip数据库官网:http://www.cz88.net/ip/
纯真ip数据库下载:http://update.cz88.net/soft/qqwry.rar
以下函数conrvertip()位于 discuz!5_gbk/upload/include/misc.func.php 路径中,有兴趣可以具体去阅读分析。(下面代码我做了简单的修改,更便于阅读,核心没有修改)
<?
//===================================
//
// 功能:ip地址获取真实地址函数
// 参数:$ip - ip地址
// 作者:[discuz!] (c) comsenz inc.
//
//===================================
function convertip($ip) {
//ip数据文件路径
$dat_path = 'qqwry.dat';
//检查ip地址
if(!preg_match("/^d{1,3}.d{1,3}.d{1,3}.d{1,3}$/", $ip)) {
return 'ip address error';
}
//打开ip数据文件
if(!$fd = @fopen($dat_path, 'rb')){
return 'ip date file not exists or access denied';
}
//分解ip进行运算,得出整形数
$ip = explode('.', $ip);
$ipnum = $ip[0] * 16777216 + $ip[1] * 65536 + $ip[2] * 256 + $ip[3];
//获取ip数据索引开始和结束位置
$databegin = fread($fd, 4);
$dataend = fread($fd, 4);
$ipbegin = implode('', unpack('l', $databegin));
if($ipbegin < 0) $ipbegin += pow(2, 32);
$ipend = implode('', unpack('l', $dataend));
if($ipend < 0) $ipend += pow(2, 32);
$ipallnum = ($ipend - $ipbegin) / 7 + 1;
$beginnum = 0;
$endnum = $ipallnum;
//使用二分查找法从索引记录中搜索匹配的ip记录
while($ip1num>$ipnum || $ip2num<$ipnum) {
$middle= intval(($endnum + $beginnum) / 2);
//偏移指针到索引位置读取4个字节
fseek($fd, $ipbegin + 7 * $middle);
$ipdata1 = fread($fd, 4);
if(strlen($ipdata1) < 4) {
fclose($fd);
return 'system error';
}
//提取出来的数据转换成长整形,如果数据是负数则加上2的32次幂
$ip1num = implode('', unpack('l', $ipdata1));
if($ip1num < 0) $ip1num += pow(2, 32);
//提取的长整型数大于我们ip地址则修改结束位置进行下一次循环
if($ip1num > $ipnum) {
$endnum = $middle;
continue;
}
//取完上一个索引后取下一个索引
$dataseek = fread($fd, 3);
if(strlen($dataseek) < 3) {
fclose($fd);
return 'system error';
}
$dataseek = implode('', unpack('l', $dataseek.chr(0)));
fseek($fd, $dataseek);
$ipdata2 = fread($fd, 4);
if(strlen($ipdata2) < 4) {
fclose($fd);
return 'system error';
}
$ip2num = implode('', unpack('l', $ipdata2));
if($ip2num < 0) $ip2num += pow(2, 32);
//没找到提示未知
if($ip2num < $ipnum) {
if($middle == $beginnum) {
fclose($fd);
return 'unknown';
}
$beginnum = $middle;
}
}
//下面的代码读晕了,没读明白,有兴趣的慢慢读
$ipflag = fread($fd, 1);
if($ipflag == chr(1)) {
$ipseek = fread($fd, 3);
if(strlen($ipseek) < 3) {
fclose($fd);
return 'system error';
}
$ipseek = implode('', unpack('l', $ipseek.chr(0)));
fseek($fd, $ipseek);
$ipflag = fread($fd, 1);
}
if($ipflag == chr(2)) {
$addrseek = fread($fd, 3);
if(strlen($addrseek) < 3) {
fclose($fd);
return 'system error';
}
$ipflag = fread($fd, 1);
if($ipflag == chr(2)) {
$addrseek2 = fread($fd, 3);
if(strlen($addrseek2) < 3) {
fclose($fd);
return 'system error';
}
$addrseek2 = implode('', unpack('l', $addrseek2.chr(0)));
fseek($fd, $addrseek2);
} else {
fseek($fd, -1, seek_cur);
}
while(($char = fread($fd, 1)) != chr(0))
$ipaddr2 .= $char;
$addrseek = implode('', unpack('l', $addrseek.chr(0)));
fseek($fd, $addrseek);
while(($char = fread($fd, 1)) != chr(0))
$ipaddr1 .= $char;
} else {
fseek($fd, -1, seek_cur);
while(($char = fread($fd, 1)) != chr(0))
$ipaddr1 .= $char;
$ipflag = fread($fd, 1);
if($ipflag == chr(2)) {
$addrseek2 = fread($fd, 3);
if(strlen($addrseek2) < 3) {
fclose($fd);
return 'system error';
}
$addrseek2 = implode('', unpack('l', $addrseek2.chr(0)));
fseek($fd, $addrseek2);
} else {
fseek($fd, -1, seek_cur);
}
while(($char = fread($fd, 1)) != chr(0)){
$ipaddr2 .= $char;
}
}
fclose($fd);
//最后做相应的替换操作后返回结果
if(preg_match('/http/i', $ipaddr2)) {
$ipaddr2 = '';
}
$ipaddr = "$ipaddr1 $ipaddr2";
$ipaddr = preg_replace('/cz88.net/is', '', $ipaddr);
$ipaddr = preg_replace('/^s*/is', '', $ipaddr);
$ipaddr = preg_replace('/s*$/is', '', $ipaddr);
if(preg_match('/http/i', $ipaddr) || $ipaddr == '') {
$ipaddr = 'unknown';
}
return $ipaddr;
}
//========================
//
// 调用举例(速度很快)
//
//========================
echo convertip('219.238.235.10');
//输出: 北京市 电信通
echo convertip('23.56.82.12');
//输出:iana
echo convertip('250.69.52.0');
//输出:iana保留地址
echo convertip('238.69.52.0');
//输出:iana保留地址 用于多点传送
echo convertip('192.168.0.1');
//输出:局域网 对方和您在同一内部网
echo convertip('255.255.255.255');
//输出:纯真网络 2006年11月20日ip数据
?>
上一篇: PHP中cookies使用指南