web573
thinkphp 3.2.3sql注入漏洞 源代码
class IndexController extends Controller {
public function index(){
$a=M('xxx');
$id=I('GET.id');
$b=$a->find($id);
var_dump($b);
}
}
来调试一下看为什么普通的注入不行。本地sql表内数据如下。 data:image/s3,"s3://crabby-images/2f9e4/2f9e4b256d726118c1d7c280615da198b8745a60" alt="在这里插入图片描述" 传入?id=1' 首先在I函数里面有个过滤,调用think_filter函数 data:image/s3,"s3://crabby-images/702ba/702ba0383ae05dbc477655fd11d05bd79d51aace" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/40783/407838636f5cfe52f1102bb27f1e160b03955496" alt="在这里插入图片描述" 但是没什么影响,就算匹配到了最多加个空格。 跟进find函数,在find函数中_parseOptions 对$options 进行了处理,继续跟进 data:image/s3,"s3://crabby-images/37e11/37e11dfbbcac75db62ccc034e1141cbcbbae42b5" alt="在这里插入图片描述"
_parseType 对$options 进行了处理,跟进 data:image/s3,"s3://crabby-images/4c8e1/4c8e12788b0004a1e0bcbeaff8dd7681a8dccd9a" alt="在这里插入图片描述" 可以看到如果我们的数据库中的id是int型会进入intval函数,会直接去掉其他字符达到过滤的效果,如果id的类型是char则会接着往下走。 data:image/s3,"s3://crabby-images/170d0/170d04df56927d2ddaafbc556f2c08d3f7ed7b69" alt="在这里插入图片描述" 执行完_parseType 后回到find 中的select data:image/s3,"s3://crabby-images/0f875/0f875639c68d5ef8e84f7424893e9c596de2bd93" alt="在这里插入图片描述" 跟进buildSelectSql data:image/s3,"s3://crabby-images/4360d/4360dd73dbb754b216016ffdbd7beb9d14e3c74f" alt="在这里插入图片描述" 跟进parseSql data:image/s3,"s3://crabby-images/3155e/3155e635c0a8ff70f2e37775b6fd7c37dd6e33d3" alt="在这里插入图片描述"
我们知道我们的id的值在$options[‘where’]中,跟进parseWhere data:image/s3,"s3://crabby-images/ab5bc/ab5bcf58a7aeb8de9b491d907894d6bab1da43ea" alt="在这里插入图片描述" 最终是进到了parseValue data:image/s3,"s3://crabby-images/f6dd5/f6dd550da54e1628cc115ff57302dfc8f55a989f" alt="在这里插入图片描述" 在parseValue中中我们看到了具体的转义函数 data:image/s3,"s3://crabby-images/f7ecc/f7eccd5f9420c5bd74225176c3113a13f2b7a9f1" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/809c1/809c173c88ddfaa9afc3144bf2177c6f35b9d141" alt="在这里插入图片描述" 也就是通过addslashes()函数进行转义
addslashes() 函数会在预定义字符之前添加反斜杠的字符串。
预定义字符是:
单引号(')
双引号(")
反斜杠(\)
NULL
那么假设我们传入的内容是数组呢?id[where]=1' 当我们传入的是数组时,不会进入下面的if中,所以$options['where']="1'" ,否则的话就是 $options['where']['id']="1'"
data:image/s3,"s3://crabby-images/e084b/e084b9929fe2278db03dbd86e30a33ff658fadcb" alt="在这里插入图片描述" 按照刚才的方式往下走到_parseOptions ,如果我们的$options['where'] 是数组的话会进入_parseType ,我们前面说到,现在的$options['where']="1'" 是个字符串,也就不会进入这个if,所以当目标数据库中的id字段是int型时可以绕过intval过滤。 data:image/s3,"s3://crabby-images/7aa8f/7aa8f3852629b92572d6f7150378f82e42b26016" alt="在这里插入图片描述" 接着往后走,直接在parsewhere中进入if$whereStr = $where; 最终返回的内容是WHERE 1' data:image/s3,"s3://crabby-images/c4a0d/c4a0dd6437abdc972820e51cf7a65527dffae7db" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/9c2b0/9c2b07079947db3e52343a85c8d6c763b049b5cc" alt="在这里插入图片描述" 最终执行的sql语句是select * from xxx where 1' limit 1 这样我们的单引号就保留了下来。 data:image/s3,"s3://crabby-images/ee7ab/ee7ab3105c2a94fb2c0e72013e250bf290e61df6" alt="在这里插入图片描述" 剩下的就很简单了。 payload:?id[where]=id=0 union select 1,flag4s,3,4 from flags 前面的跑表名和列名就不具体写了。
|