Discuz5.0中读取纯真IP数据库函数分析
2023-06-13 09:13:54 时间
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数据
?>
相关文章
- 自定义 WordPress 数据库错误页
- redis bind多个ip的方法详解数据库
- SqlServer时间格式化详解数据库
- 数据库Oracle IP数据库:提供无限可能性(oracle的ip)
- 数据库恢复oracle数据库的DP技术实践(dp恢复oracle)
- MySQL虚拟IP:有效管理数据库应用程序(mysql虚拟ip)
- MySQL远程连接:通过IP访问数据库(mysql通过ip访问)
- PHP操作MySQL数据库,实现强大的数据功能(php调用mysql数据库)
- 安全提高安全:立刻搭建Linux IP防火墙!(linux防火墙ip)
- Linux如何查看网卡IP(linux查看网卡ip)
- 改变Redis数据库IP地址: 一步步指南(redis修改ip地址)
- MySQL数据库主机地址简介(mysql数据库主机地址)
- Linux下如何修改IP网关(linux 修改ip网关)
- Linux下实现子IP设置指南(linux设置子ip)
- Linux 下端口活IP配置实现(linux 端口配置ip)
- DB2与Oracle数据库挑选一个最适合的方案(db2数据库oracle)
- 解决Oracle数据库IP连接超时问题(ip连接oracle超时)
- 指定IP访问Redis 解锁更多可能性(指定ip访问redis)
- 检测检测Oracle 会话IP,保证数据安全(oracle 会话ip)
- Redis确保网络安全的IP限制策略(redis限制来源ip)
- Redis维护IP访问次数上限(redis限制ip次数)
- Oracle两个IP端口完全相同(oracle两个ip相等)
- 解决Oracle数据库无法IP访问的问题(oracle不能ip访问)
- Oracle数据库安全性提升IP白名单的配置方法(oracle ip白名单)