作者:LoRexxar'@知道创宇404实验室 时间:2020年5月11日
周末看了一下这次空指针的第三次Web公开赛,稍微研究了下发现这是一份最新版DZ3.4几乎默认配置的环境,我们需要在这样一份几乎真实环境下的DZ中完成Get shell。这一下子提起了我的兴趣,接下来我们就一起梳理下这个渗透过程。
与默认环境的区别是,我们这次拥有两个额外的条件。
1、Web环境的后端为Windows 2、我们获得了一份config文件,里面有最重要的authkey
得到这两个条件之后,我们开始这次的渗透过程。
以下可能会多次提到的出题人写的DZ漏洞整理
· 这是一篇“不一样”的真实渗透测试案例分析文章[1]
//
authkey有什么用?
//
代码语言:javascript复制/ ------------------------- CONFIG SECURITY -------------------------- //
$_config['security']['authkey'] = '87042ce12d71b427eec3db2262db3765fQvehoxXi4yfNnjK5E';
authkey是DZ安全体系里最重要的主密钥,在DZ本体中,涉及到密钥相关的,基本都是用authkey
和cookie中的saltkey加密构造的。
当我们拥有了这个authkey之后,我们可以计算DZ本体各类操作相关的formhash(DZ所有POST相关的操作都需要计算formhash)
配合authkey,我们可以配合source/include/misc/misc_emailcheck.php
中的修改注册邮箱项来修改任意用户绑定的邮箱,但管理员不能使用修改找回密码的api。
可以用下面的脚本计算formhash
代码语言:javascript复制$username = "ddog";
$uid = 51;
$saltkey = "SuPq5mmP";
$config_authkey = "87042ce12d71b427eec3db2262db3765fQvehoxXi4yfNnjK5E";
$authkey = md5($config_authkey.$saltkey);
$formhash = substr(md5(substr($t, 0, -7).$username.$uid.$authkey."".""), 8, 8);
当我们发现光靠authkey没办法进一步渗透的时候,我们把目标转回到hint上。
1、Web环境的后端为Windows 2、dz有正常的备份数据,备份数据里有重要的key值
//
Windows短文件名安全问题
//
在2019年8月,dz曾爆出过这样一个问题。
• windows短文件名安全问题 数据库备份爆破[2]
在windows环境下,有许多特殊的有关通配符类型的文件名展示方法,其中不仅仅有 <>"
这类可以做通配符的符号,还有类似于~
的省略写法。这个问题由于问题的根在服务端,所以cms无法修复,所以这也就成了一个长久的问题存在。
具体的细节可以参考下面这篇文章:
• Windows下的"你画我猜" -- 告别效率低下的目录扫描方法[3]
配合这两篇文章,我们可以直接去读数据库的备份文件,这个备份文件存在
代码语言:javascript复制/data/backup_xxxxxx/200509_xxxxxx-1.sql
我们可以直接用
代码语言:javascript复制http://xxxxx/data/backup~1/200507~2.sql
拿到数据库文件
从数据库文件中,我们可以找到UC_KEY(dz)
在pre_ucenter_applications
的authkey字段找到UC_KEY(dz)
至此我们得到了两个信息:
代码语言:javascript复制uckey
x9L1efE1ff17a4O7i158xcSbUfo1U2V7Lebef3g974YdG4w0E2LfI4s5R1p2t4m5
authkey
87042ce12d71b427eec3db2262db3765fQvehoxXi4yfNnjK5E
当我们有了这两个key之后,我们可以直接调用uc_client的uc.php任意api。,后面的进一步利用也是建立在这个基础上。
//
uc.php api利用
//
这里我们主要关注/api/uc.php
通过UC_KEY
来计算code,然后通过authkey
计算formhash,我们就可以调用当前api下的任意函数,而在这个api下有几个比较重要的操作。
我们先把目光集中到updateapps
上来,这个函数的特殊之处在于由于DZ直接使用preg_replace
替换了UC_API
,可以导致后台的getshell。
具体详细分析可以看,这个漏洞最初来自于@dawu,我在CSS上的演讲中提到过这个后台getshell:
• https://paper.seebug.org/1144/#getwebshell
• https://lorexxar.cn/2020/01/14/css-mysql-chain/#任意文件读-with-配置文件泄露
根据这里的操作,我们可以构造$code = 'time='.time().'&action=updateapps';
来触发updateapps,可以修改配置中的UC_API
,但是在之前的某一个版本更新中,这里加入了条件限制。
if($post['UC_API']) {
$UC_API = str_replace(array(''', '"', '\', "