data:image/s3,"s3://crabby-images/380ef/380ef8ee9a75072a15e6cf4dabadf78f768dd7fc" alt="image-20211117223456251"
漏洞概要
漏洞存在于 ThinkPHP模板引擎中,在加载模版解析变量时存在变量覆盖问题,而且程序没有对数据进行很好的过滤,最终导致文件包含漏洞 的产生。
影响版本
5.0.0 <= Thinkphp <= 5.0.18
5.1.0 <= ThinkPHP <= 5.1.10
环境搭建
Phpstudy:
- OS:
Windows - PHP:
7.3.4 - ThinkPHP:
5.0.18
创建测试环境:
composer create-project topthink/think=5.0.18 thinkphp5.0.18
将 composer.json 文件的 require 字段设置成如下:
"require": {
"php": ">=5.4.0",
"topthink/framework": "5.0.18"
},
执行composer update
控制器:application/index/controller/Index.php
<?php
namespace app\index\controller;
use think\Controller;
class Index extends Controller
{
public function index()
{
$this->assign(request()->get());
return $this->fetch();
}
}
创建 application/index/view/index/index.html 文件,内容任意,作为这个index控制器的index方法的模板。
在public目录放置一个图片马,模拟文件上传操作
漏洞复现
http://localhost:8090/public/?cacheFile=1.jpg
访问即可文件包含
data:image/s3,"s3://crabby-images/6cb7c/6cb7cec4199a9b2156e0e3742df4d547a7353a9f" alt="image-20211130235551702"
漏洞分析
public function index()
{
$this->assign(request()->get());
return $this->fetch();
}
跟进assign() 方法
data:image/s3,"s3://crabby-images/523ea/523eaed4c213f65a4c5484abe243dca38eabe802" alt="image-20211201002846035"
跟进$this->view->assign
在:thinkphp/library/think/View.php
data:image/s3,"s3://crabby-images/b640e/b640ee81a9350f0050b16cb6850a179b4c6ec5fa" alt="image-20211201093144107"
将参数赋值后返回,跟进fetch
data:image/s3,"s3://crabby-images/e3878/e3878170685d1d8261cbdff43a89b40b4e4065c6" alt="image-20211201093317181"
继续跟进
data:image/s3,"s3://crabby-images/0b095/0b0955bbc7a92335a8a813f9573df2060fa17b71" alt="image-20211201093932731"
变量$vars 为我们get传入的参数并赋值给$this->data
跟进$this->engine->$method($template, $vars, $config);
在think/view/driver/Think.php
data:image/s3,"s3://crabby-images/381fa/381fa5da8425971862a5ee952056a6c0a0924c1d" alt="image-20211201095558659"
可以看到,$data 是get传参。parseTemplate() 函数获取模板文件名,得到默认文件名
继续跟进fetch 方法
public function fetch($template, $vars = [], $config = [])
{
if ($vars) {
$this->data = $vars;
}
...
if ($template) {
...
$this->storage->read($cacheFile, $this->data);
...
echo $content;
}
}
跟进$this->storage->read($cacheFile, $this->data);
在:think/template/driver/File.php
data:image/s3,"s3://crabby-images/dfa7c/dfa7cc48aaa0c0c9c5c84e59b33c17939372fb88" alt="image-20211201115020846"
这里有extract($vars, EXTR_OVERWRITE); 。在EXTR_OVERWRITE标记的情况下,如果有冲突,覆盖已有的变量。那么我们可以通过变量覆盖$cacheFile 变量,从而实现文件包含。
方法调用栈:
data:image/s3,"s3://crabby-images/92dcc/92dcc63d1c82d8fabba39967df74ce27227d5745" alt="image-20211201115312700"
漏洞修复
判断get传参里是否有cacheFile 键,有的话就删掉,也删掉_think_cacheFile 键,然后再进行文件包含。
data:image/s3,"s3://crabby-images/75cef/75cef251a79c9c5ac6de5533141c872f9b93cc33" alt="image-20211201120146191"
|