前言
这是Struts系列第十六篇,继续加油!
Struts简介
Struts2是用Java语言编写的一个基于MVC设计模式的Web应用框架
漏洞复现
漏洞简介
S2-059漏洞,又名CVE-2019-0230漏洞
Struts2框架会对某些标签属性(比如id)的属性值进行二次表达式解析,因此当这些标签属性中使用了%{x}且x的值用户可控时,传入一个将在呈现标签属性时再次解析的OGNL表达式,就可以造成OGNL表达式注入,造成任意代码执行.
漏洞成因
Struts2框架会对某些标签属性(比如id)的属性值进行二次表达式解析,因此当这些标签属性中使用了%{x}且x的值用户可控时,传入一个将在呈现标签属性时再次解析的OGNL表达式,就可以造成OGNL表达式注入,造成任意代码执行.
漏洞影响范围
Struts 2.0.0 - Struts 2.5.20
环境搭建
使用IDEA直接打开源码地址中的对应文件,然后修改/web/index.jsp文件(因为有很多干扰字段,不方便调试),然后配置好Tomcat就可以运行了.
Payload
无
POC
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
37
38
39
40
41
42
43
44
45
46
47
48
49
#!/usr/bin/env python3
import re
import requests
class S2_059_BaseVerify:
def __init__(self, url):
self.info = {
'name': 'S2-059漏洞,又名CVE-2019-0230漏洞',
'description': 'Struts2 Remote Code Execution Vulnerability, Struts 2.0.0 - Struts 2.5.20',
'date': '2020-08-11',
'type': 'RCE'
}
self.url = url
if not self.url.startswith("http") and not self.url.startswith("https"):
self.url = "http://" + self.url
if '.action' not in self.url:
self.url = self.url + '/index.action'
self.headers = {
'User-Agent': "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/45.0.2454.85 Safari/537.36 115Browser/6.0.3",
'Content-Type': "application/x-www-form-urlencoded"
}
self.payload = {
'skillName': '%{11*11}',
'url': '/s2_059_war_exploded/index.action'
}
def run(self):
"""
检测是否存在漏洞
:param:
:return str True or False
"""
check_req = requests.post(self.url, data = self.payload, headers = self.headers)
check_str = re.sub('\n', '', check_req.text)
result = re.findall('label id="(.*?)">', check_str)
if '121' in result[0]:
return True
else:
return False
if __name__ == "__main__":
S2_059 = S2_059_BaseVerify('http://localhost:8080/s2_059_war_exploded/index.action')
print(S2_059.run())
漏洞分析
首先Struts2的运行流程是
1.HTTP请求经过一系列的标准过滤器(Filter)组件链(这些拦截器可以是Struts2 自带的,也可以是用户自定义的,本环境中struts.xml中的package继承自struts-default,struts-default就使用了Struts2自带的拦截器.ActionContextCleanUp主要是清理当前线程的ActionContext、Dispatcher,FilterDispatcher主要是通过ActionMapper来决定需要调用那个Action,FilterDispatcher是控制器的核心,也是MVC中控制层的核心组件),最后到达FilterDispatcher过滤器.
2.核心控制器组件FilterDispatcher根据ActionMapper中的设置确定是否需要调用某个Action组件来处理这个HttpServletRequest请求,如果ActionMapper决定调用某个Action组件,FilterDispatcher核心控制器组件就会把请求的处理权委托给ActionProxy组件.
3.ActionProxy组件通过Configuration Manager组件获取Struts2框架的配置文件struts.xml,最后找到需要调用的目标Action组件类,然后ActionProxy组件就创建出一个实现了命令模式的ActionInvocation类的对象实例类的对象实例(这个过程包括调用Anction组件本身之前调用多个的拦截器组件的before()方法)同时ActionInvocation组件通过代理模式调用目标Action组件.但是在调用之前ActionInvocation组件会根据配置文件中的设置项目加载与目标Action组件相关的所有拦截器组件(Interceptor)
4.一旦Action组件执行完毕,ActionInvocation组件将根据开发人员在Struts2.xml配置文件中定义的各个配置项目获得对象的返回结果,这个返回结果是这个Action组件的结果码(比如SUCCESS、INPUT),然后根据返回的该结果调用目标JSP页面以实现显示输出.
5.最后各个拦截器组件会被再次执行(但是顺序和开始时相反,并调用after()方法),然后请求最终被返回给系统的部署文件中配置的其他过滤器,如果已经设置了ActionContextCleanUp过滤器,那么FilterDispatcher就不会清理在ThreadLocal对象中保存的ActionContext信息.如果没有设置ActionContextCleanUp过滤器,FilterDispatcher就会清除掉所有的ThreadLocal对象.
具体分析过程:
1.首先来看下jsp文件,标签属性中使用了%{skillName},就可以触发漏洞.首先在index.jsp第16行打断点,这时候开始解析标签<>,会调用doStartTag方法
2.在lib/struts2-core-2.3.24.jar/org.apache.struts2.views.jsp.ComponentTagSupport.doStartTag第34行打断点.然后在index.jsp按F8进入到doStartTag方法内,F7跟入populateParams方法
3.F7继续进入父方法populateParams,然后F7继续进入父方法populateParams
4.就可以看到了具体设置参数的函数,找到设置id的地方,F7进入setId方法
5.见到了熟悉方法,findString,F7进入findString方法
6.F7继续进入findValue方法
7.findValue方法会判断是否开启了altSyntax,开启后才会继续下去,所以触发这个漏洞需要开启altSyntax,默认altSyntax是开启的.又见到了老朋友translateVariables.继续F7跟下去translateVariables,直到具体实现方法.
8.F7进入evalvate方法,在第56行Object o = evaluator.evaluate(var);进行了第一次解析,解析后%{skillName}变成了%{11*11},因为在S2-001漏洞的修复中用maxLoopCount限制了解析次数,所以只能解析一次.
9.回到doStartTag方法,继续往下执行.F7进入start方法.
10.F7进入父方法start,然后F7进入evalvateParams方法
11.evalvateParams方法添加了一些参数,然后解析参数.F7进入populateComponentHtmlId方法.
12.F7进入findStringIfAltSyntax方法.发现还是判断是否开启了altSyntax,开启后才会继续下去,默认altSyntax是开启的.F7进入findString方法.
13.会发现接下来的流程和第一次解析的流程一样了.F7进入findValue方法.再次见到了老朋友translateVariables.继续F7跟下去translateVariables,直到具体实现方法.
14.F7进入evalvate方法,在第56行Object o = evaluator.evaluate(var);进行了第二次解析,解析后%{11*11}变成了121,解析后通过值栈查找并将其返回.
漏洞修复
升级到Struts 2.5.22或更高版本
总结
只要是ongl表达式注入漏洞就最终就是通过translateVariables方法执行的.
参考
https://cwiki.apache.org/confluence/display/WW/S2-059
https://github.com/vulhub/vulhub/blob/master/struts2/s2-059/README.zh-cn.md
https://mp.weixin.qq.com/s/VyLiLrUV0yakh_lzTBYGyQ
https://xz.aliyun.com/t/8653