如何利用CodeQL挖掘CVE-2020-10199

admin 2023年3月8日22:22:12评论35 views字数 2719阅读9分3秒阅读模式
如何利用CodeQL挖掘CVE-2020-10199

前言

挖掘Java漏洞时候存在一个痛点在已知漏洞点的情况下,如何找到可控点,以及确定一条具体的数据流.
前几天看到@Alvaro Muñoz 利用codeql挖掘到[CVE-2020-10199]Nexus Repository Manager远程代码执行漏洞.于是就对codeql产生强烈的好奇心,怼了几天codeql语法和CVE案例。抠出挖掘该CVE的Ql代码

Codeql

对Codeql具体介绍可以参考:https://help.semmle.com/
使用codeql能够很明确搜索出方法调用的数据流, QL代码编写思路相当于在一个代码数据库中利用编写的逻辑搜索出可能的调用点。重点是确定Souce和Sink搜索逻辑.

漏洞原理简要分析

该漏洞是由于未做过滤造成表达式注入,作者确定漏洞点执行函数buildConstraintViolationWithTemplate,其该方法中bean.getMessage()返回结果message并未使用stripJavaEl方法过滤造成表达式注入
如何利用CodeQL挖掘CVE-2020-10199
最后调用org.hibernate.validator.internal.engine.messageinterpolation#interpolate,执行插入操作并解析恶意字符串,造成表达式注入.
如何利用CodeQL挖掘CVE-2020-10199

HelperBean bean#message

编写Codeql最重要的是先确定数据流的Sink点即污染点字段.这里跟进org.sonatype.nexus.validation.HelperBean#getMessage方法.返回message值.该值经由HelperBean方法传入
如何利用CodeQL挖掘CVE-2020-10199

Ql代码

这里直接给出代码并做简单讲解:搜索出流经HelperBean方法中message字段的数据流
isSink方法搜索逻辑: 定义参数p,该参数p由被调用的HelperBean方法使用且该p参数名为message
isSouce方法搜索逻辑:由于不清楚哪里数据源最终经过该sink,所以直接使用source.asExpr()调用所有可能的表达式作为数据流.
import javaimport semmle.code.java.dataflow.FlowSources

class Config extends DataFlow::Configuration{ Config(){ this = "1" } override predicate isSource(DataFlow::Node source){ exists(source.asExpr()) } override predicate isSink(DataFlow::Node sink){ exists( Parameter p | p = sink.asParameter() and p.getCallable().getName() = "HelperBean" and p.getName() = "message" ) }

}
from Config config ,DataFlow::PathNode source, DataFlow::PathNode sinkwhere config.hasFlowPath(source, sink)select source.getNode().getLocation(),source,sink

搜索结果

Run Query
如何利用CodeQL挖掘CVE-2020-10199

AbstractGroupRepositoriesApiResource#validateGroupMembers

根据搜索结果,这里最可疑的souce就是AbstractGroupRepositoriesApiResource:92.点击该结果跟进,Codeql很友好对污染点respositoryName进行标记。向上溯源得到respositoryName值最终由request中获取.
如何利用CodeQL挖掘CVE-2020-10199

Ql代码

现在就是需要确定AbstractGroupRepositoriesApiResource#validateGroupMembers()方法由哪里调用.
isSink方法搜索逻辑:定义参数p,该参数p由被调用的validateGroupMembers方法使用且该p参数名为request
isSouce方法搜索逻辑:这里souce原本是用Codeql自带RemoteFlowSource获取,但是发现并没有结果。所以这里直接用souce.asExpr()搜索全局表达式。
import javaimport semmle.code.java.dataflow.FlowSources
class Config extends DataFlow::Configuration{ Config(){ this = "1" } override predicate isSource(DataFlow::Node source){ exists(source.asExpr()) } override predicate isSink(DataFlow::Node sink){ exists( Parameter p | p = sink.asParameter() and p.getCallable().getName() = "validateGroupMembers" and p.getName() = "request"
) }

}// 得到可能可控的函数from Config config ,DataFlow::PathNode source, DataFlow::PathNode sinkwhere config.hasFlowPath(source, sink)select source, sink

搜索结果

Run Query
如何利用CodeQL挖掘CVE-2020-10199
其中GolangGroupRepositoriesApiResource就是该漏洞调用的可控点.
如何利用CodeQL挖掘CVE-2020-10199

结语

原文有误,还请师傅们斧正。希望师傅们喜欢

参考链接
https://help.semmle.com/wiki/display/CBJAVA/Override+of+method
https://help.semmle.com/QL/ql-explore-queries.html
https://help.semmle.com/qldoc/java/
https://www.youtube.com/watch?v=irrYp3wdtsw
https://securitylab.github.com/advisories/GHSL-2020-011-nxrm-sonatype
如何利用CodeQL挖掘CVE-2020-10199

- 结尾 -
精彩推荐
【技术分享】前尘——内存中无处可寻的木马
【技术分享】mruby字节码逆向入门
【技术分享】Intel AES-NI使用入门

如何利用CodeQL挖掘CVE-2020-10199
戳“阅读原文”查看更多内

原文始发于微信公众号(安全客):如何利用CodeQL挖掘CVE-2020-10199

  • 左青龙
  • 微信扫一扫
  • weinxin
  • 右白虎
  • 微信扫一扫
  • weinxin
admin
  • 本文由 发表于 2023年3月8日22:22:12
  • 转载请保留本文链接(CN-SEC中文网:感谢原作者辛苦付出):
                   如何利用CodeQL挖掘CVE-2020-10199https://cn-sec.com/archives/1241918.html

发表评论

匿名网友 填写信息