前言
红名谷线下半决赛遇到的一个PHP论坛,据说这个论坛有几个很精彩的漏洞,这次来玩一玩。
因为找不到什么文章以及漏洞适用环境不符,所以基本上的思路是利用任意文件删除来删除install.lock,然后再重装论坛配置密码,到后台使用命令执行漏洞。
配置环境
首先到GitHub下载源码,我用的是(https://github.com/mektap/Discuz-3.4-Uyghurche)。
data:image/s3,"s3://crabby-images/85727/857274a81c28d0d014b5cf26ab59a288030068d1" alt=""
任意文件删除漏洞
先在首页注册一个账号,然后到设置页面源码里面找到formhash保存起来,一会还要用到。
data:image/s3,"s3://crabby-images/87703/87703643c4142ab59da074187fb7334734baf23a" alt=""
对保存这个按钮事件触发的请求包抓包,把数据包中的出生地改成要删除文件的相对路径。
data:image/s3,"s3://crabby-images/53c7c/53c7c3ac86b4a5f3f03b1445973ed172182658bd" alt=""
可以看到页面出生地的地方回显成刚刚修改的相对路径。
data:image/s3,"s3://crabby-images/76142/7614283457a3a0b3d4c91ff4a938f2ceb7eb6ec5" alt=""
构造一个payload来上传文件,这里的formhash记得改成一开始从前端获取到的那个。
data:image/s3,"s3://crabby-images/1649c/1649ce7925ebd1d73d66891c4c2922991b6d764e" alt=""
这里贴一下payload的源码。
<body>
<form action="http://[网站域名]/home.php?mod=spacecp&profilesubmit=1&formhash=[刚刚得到的formhash]" method="post" enctype="multipart/form-data">
<input type="file" name="birthprovince" />
<input type="submit" value="upload" />
</form>
</body>
接着,在本地表单上传一张喜欢的照片。(不喜欢的其实也行)
data:image/s3,"s3://crabby-images/0954b/0954b50d1f46a54fbd1f96dc1d7b0dc52243f909" alt=""
可以看到data目录下的install.lock已经被删除啦!
data:image/s3,"s3://crabby-images/6b61f/6b61f598b5a4950910c0d12997088cfb5a5abd0e" alt=""
来看一下在源码里面到底发生了什么。首先来到source/include/spacecp/spacecp_profile.php,这里有个submitcheck('profilesubit')需要绕过。
data:image/s3,"s3://crabby-images/01dc2/01dc2f1393a4b73fbe15ccbabb56f50aa19cdcae" alt=""
进来submitcheck()函数之后会进入getgpc()函数,这里需要通过get对profilesubmit传参,var才能获取到其它的get参数。 这就解释了为什么上传文件那个payload中的url带有profilesubmit参数。
data:image/s3,"s3://crabby-images/9ff70/9ff70b63eacf29cc92dd477660ec100b78ca35a8" alt=""
接着,回调submitcheck()函数,可以看到要满足if的条件才能返回一个TRUE。 可以看到这里对formhash进行了校验,所以在刚刚上传文件那个payload中的url带上从前端获取的formhash。
data:image/s3,"s3://crabby-images/c4071/c4071c9fc15786c0b29f8874880c679633fad689" alt=""
这个地方的unlink是补丁前的利用点,补丁添加了formtype的判断, 导致这个unlink不可以利用。不过,下面还有一个unlink可以利用。
data:image/s3,"s3://crabby-images/2446a/2446a27ca1ae275415bd60d1a1c4417ee879d40e" alt=""
上传文件后会进入这个if语句中,先是if判断是否上传失败,成功的话往if里面走。
data:image/s3,"s3://crabby-images/c5e7e/c5e7ec4921d6702226b2f5e6f7722cc08abb13c9" alt=""
接着,这里对更新前的个人资料做unlink处理,这就解释了为什么要在个人设置中找到一个可控的出生地(birthprovine), 并且把它赋值为要删除文件的相对路径。所以,这里删除的就是我们一开始在burpsuite中构造的出生地(birthprovince)。
data:image/s3,"s3://crabby-images/5fe2b/5fe2bf4cfe48f30f60a2fe9c846401469f3a3e4d" alt=""
然后在这里开始对数据库内容进行update处理,构造update的sql语句。
data:image/s3,"s3://crabby-images/d4ba7/d4ba73e019cae658932aba46b12fb10c71ada59a" alt=""
执行update的sql语句,将个人资料出生地变为照片的相对路径。
data:image/s3,"s3://crabby-images/2563e/2563e26d274424769b66c1ebd5b98364a97959f0" alt=""
重装getshell漏洞
访问/install/index.php,进入重装页面,将数据表前缀改为x');phpinfo();('。不过这里不知道数据库的账号密码,实战中可以尝试下root或者123456这类弱密码。 实在不行的话,牺牲一下自己的vps,远程连接自己的数据库(毕竟拿shell血赚不亏)。
不过这里有一个前提条件需要注意,目标网站需要没登录过后台。因为登录后台后会把install/index.php删除导致不能重装, 所以拿到站,最好先访问/install/index.php,看看能不能通过重装getshell再决定要不要打payload。
data:image/s3,"s3://crabby-images/946d2/946d2dcedf2cebbaf961267959805de3a783aecb" alt=""
访问config/config_ucenter.php就可以执行插入的payload了。
data:image/s3,"s3://crabby-images/a0ba4/a0ba427bd8991026d396a1a65774385091479f11" alt=""
审计一下源码,可以看到进入了203行的elseif分支。
data:image/s3,"s3://crabby-images/a8564/a856429cb5bc360d24d9ebb2b8e3d5f68db3216c" alt=""
接着,进入install_uc_server()函数。
data:image/s3,"s3://crabby-images/f4dea/f4dea67bdede34a270b705e9e8b37121e59b371d" alt=""
调用save_uc_config()函数保存配置。
data:image/s3,"s3://crabby-images/9b73e/9b73eb03050868168c43069a6dae8c7a548e8c7a" alt=""
可以看到这里没有任何的过滤就直接写入php配置文件,所以只要闭合')即可代码注入。
data:image/s3,"s3://crabby-images/33376/333761600804a8dd8da221944410b8b9faf52b9b" alt=""
后台SQL注入
注入点在后台的UCenter应用id处。
data:image/s3,"s3://crabby-images/76b90/76b909392adac660c150c117fa34b660a718625b" alt=""
构造报错注入payload,页面成功回显。
data:image/s3,"s3://crabby-images/5c763/5c763f92ea3d38180083fc3805ce41be0b26c949" alt=""
来到后台设置的代码段。
data:image/s3,"s3://crabby-images/18ae8/18ae81c8ce090adffb7883b19b3eb15439f129ec" alt=""
可以看到这里对输入的id进行了addslashes()转义处理。
data:image/s3,"s3://crabby-images/19c64/19c646b9c3bfc1738ae2a411358497525f1f7258" alt=""
这里读取配置文件,对配置缓存进行了更新。 而问题在于此处读取时没有再次转义,转义符号在读取的时候已被解析,造成了二次注入。
data:image/s3,"s3://crabby-images/a472b/a472bf8fd1ed39188fde0b4dfcc2b4f887293a53" alt=""
note_exists()函数通过result_first()函数将配置内容select出来。
data:image/s3,"s3://crabby-images/09925/0992530635fa6a71e3690d4e89be0622e0b7ae8f" alt=""
从result_first()函数执行的sql语句,也可以看到转义符号已经被解析,构造了SQL注入的payload。
data:image/s3,"s3://crabby-images/660fb/660fbbb07b7527fc28a4af4a244e814d4c7542c4" alt=""
There Is Nothing Below
data:image/s3,"s3://crabby-images/e1204/e1204db1770ccffe2a43222720440f2a99c687ad" alt=""