漏洞公告
https://cwiki.apache.org/confluence/display/WW/S2-057
问题:
It is possible to perform a RCE attack when namespace value isn’t set for a result defined in underlying xml configurations and in same time, its upper action(s) configurations have no or wildcard namespace. Same possibility when using url tag which doesn’t have value and action set and in same time, its upper action(s) configurations have no or wildcard namespace.
漏洞发现者的博客: https://lgtm.com/blog/apache_struts_CVE-2018-11776
环境搭建
下载 https://archive.apache.org/dist/struts/2.5.16/struts-2.5.16-all.zip
IDEA中打开,修改apps/showcase/src/main/resources/struts-actionchaining.xml 为:
同时查看 org/apache/struts2/default.properties:201 ,其值为true
访问: http://localhost:8081/${(111+111)}/actionChain1.action
url变为: http://localhost:8081/222/register2.action
111+111=222 即产生了OGNL注入。
漏洞分析
这次的漏洞可以有多种攻击向量,根据漏洞作者blog有:
以上提及的三种都属于Struts2的跳转方式。在 struts-default.xml:190(截取部分)
为清楚起见,这里解释一下strut2中对默认result对象
的处理过程。这些默认result type
都要经过 com/opensymphony/xwork2/DefaultActionInvocation.java:367 处理
首先通过result = createResult()
获取到相应的result对象。如果result不为null则执行result.execute(this);
。这个execute
方法则由具体result对象实现。
有一些具体的result对象比如下面提到的Redirect action和Postback result,会产生一个跳转地址location,并传入org/apache/struts2/result/StrutsResultSupport.java:194:
而conditionalParse
定义如下,将会执行OGNL表达式。
所以可以看到重点是StrutsResultSupport中conditionalParse(location, invocation)
的location变量。
接下来部分就关注三种result-type的具体实现和具体攻击点。
攻击点一:Redirect action
apps/showcase/src/main/resources/struts-actionchaining.xml 中注意<result>
标签中<type>
为redirectAction
:
redirectAction
对应的处理类为org.apache.struts2.result.ServletActionRedirectResult
在 com/opensymphony/xwork2/DefaultActionInvocation.java:368
跟入redirectAction
的execute
方法即 org/apache/struts2/result/ServletActionRedirectResult.java:160
由于在配置xml时没有指定naPmespace,所以这里的namespace为null,将会执行invocation.getProxy().getNamespace();
所以执行后对于result对象的namespace即为/${(111+111)}
。
同一函数中继续执行 172行
ActionMapping
生成如下,this.namespace
值赋为/${(111+111)}
:
跟入getUriFromActionMapping
:
handleNamespace
处理结果如下:
当函数返回,tmpLocation
值为/${(111+111)}/register2.action
,然后通过setLocation(tmpLocation)
使得location变量值为/${(111+111)}/register2.action
,从而最终造成OGNL注入。
攻击点二: Action chaining
apps/showcase/src/main/resources/struts-actionchaining.xml 中注意<result>
标签中<type>
为chain
:
同样会先经过result = createResult()
,然后调用result.execute(this);
。这会进入到 com/opensymphony/xwork2/ActionChainResult.java:203
由于没有设定namespace
,所以通过invocation.getProxy().getNamespace()
使得this.namespace
值为/${(111+111)}
。然后调用了String finalNamespace = TextParseUtil.translateVariables(namespace, stack);
对namespace进行OGNL解析。如下
攻击点三:Postback result
apps/showcase/src/main/resources/struts-actionchaining.xml 中注意<result>
标签中<type>
为postback
:
经过result = createResult()
,跟入定位到postback
这个result对象的处理方法,在 org/apache/struts2/result/PostbackResult.java:113
跟入makePostbackUri1
,在org/apache/struts2/result/PostbackResult.java:129
获取到namespace
值为/${(111+111)}
。跟入actionMapper.getUriFromActionMapping(new ActionMapping(actionName, namespace, method, null))
,其具体执行过程如攻击点一[Redirect action]提到的那样,设置namespace等参数,然后从getUriFromActionMapping
中返回uri。最后组装的postbackUri为/${(111+111)}/register2.action
回到前面的execute
中通过setLocation(postbackUri)
设置了location变量:
此后location变量传入,造成OGNL表达式注入