Bypass JEP290

0nise  1703天前

关于JEP290

JEP290是Java底层为了缓解反序列化攻击提出的一种解决方案,主要做了以下几件事

  1. 提供一个限制反序列化类的机制,白名单或者黑名单。
  2. 限制反序列化的深度和复杂度。
  3. 为RMI远程调用对象提供了一个验证类的机制。
  4. 定义一个可配置的过滤机制,比如可以通过配置properties文件的形式来定义过滤器。

JEP290的实际限制

写一个RMIServer

RMIServer.java

package org.chabug.rmi.server;

import java.rmi.Naming;
import java.rmi.registry.LocateRegistry;

public class RMIServer {
    public static String HOST = "127.0.0.1";
    public static int PORT = 1099;
    public static String RMI_PATH = "/hello";
    public static final String RMI_NAME = "rmi://" + HOST + ":" + PORT + RMI_PATH;

    public static void main(String[] args) {
        try {
            // 注册RMI端口
            LocateRegistry.createRegistry(PORT);
            // 创建一个服务
            Hello hello = new HelloImpl();
            // 服务命名绑定
            Naming.rebind(RMI_NAME, hello);

            System.out.println("启动RMI服务在" + RMI_NAME);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

HelloImpl.java

package org.chabug.rmi.server;

import java.rmi.RemoteException;
import java.rmi.server.UnicastRemoteob ject;

public class HelloImpl extends UnicastRemoteob ject implements Hello {
    protected HelloImpl() throws RemoteException {
    }

    public String hello() throws RemoteException {
        return "hello world";
    }

    public String hello(String name) throws RemoteException {
        return "hello" + name;
    }

    public String hello(ob ject ob ject) throws RemoteException {
        System.out.println(ob ject);
        return "hello "+ob ject.toString();
    }
}

Hello.java

package org.chabug.rmi.server;

import java.rmi.Remote;
import java.rmi.RemoteException;

public interface Hello extends Remote {
    String hello() throws RemoteException;
    String hello(String name) throws RemoteException;
    String hello(ob ject ob ject) throws RemoteException;
}

使用JDK7u21打Commonscollections1,成功弹出calc。

1.png

使用JDK8u221启动RMIServer攻击失败

2.png

报错显示

ob jectInputFilter REJECTED: class sun.reflect.annotation.AnnotationInvocationHandler, array length: -1, nRefs: 8, depth: 2, bytes: 298, ex: n/a

JEP290的过滤机制

在上文的报错中可见sun.reflect.annotation.AnnotationInvocationHandler被拒绝,跟一下RMI的过程,看下在哪里过滤了,JEP290又是怎么实现的。以下使用JDK8U221调试

首先我们要清楚RMI的实现流程

3.png

在远程引用层中客户端服务端两个交互的类分别是RegistryImpl_StubRegistryImpl_Skel,在服务端的RegistryImpl_Skel类中,向注册中心进行bind、rebind操作时均进行了readob ject操作以此拿到Remote远程对象引用。

142f4a6a-d079-67d1-7589-2ec9b2c87bd9.png

跟进63行进入到java.io.ob jectInputStream#readob ject,然后进入readob ject0()

07e55638-7dc5-19b3-48ec-36ac1f55192f.png

在readob ject0()之中进入readOrdinaryob ject()

b22fab94-24cc-eca5-d3a5-f68ea77ee763.png

继续进入readClassDesc()

6820d57b-ebc5-d90d-61cc-1af5d21f740d.png

进入readProxyDesc()

b80e3ce1-ea18-dbce-c4db-b98460d02125.png

在readProxyDesc()中有filterCheck

ca3420f8-aa19-1d5e-7007-c90b759f0cbe.png

先检查其所有接口,然后检查对象自身。进入filterCheck()之后

0b94aa26-5a39-5aee-3881-996124443159.png

调用了serialFilter.checkInput(),最终来到sun.rmi.registry.RegistryImpl#registryFilter

6e8698a3-399a-d1c0-c105-59f20fadf905.png

return String.class != var2 && !Number.class.isAssignableFrom(var2) && !Remote.class.isAssignableFrom(var2) && !Proxy.class.isAssignableFrom(var2) && !UnicastRef.class.isAssignableFrom(var2) && !RMIClientSocketFactory.class.isAssignableFrom(var2) && !RMIServerSocketFactory.class.isAssignableFrom(var2) && !ActivationID.class.isAssignableFrom(var2) && !UID.class.isAssignableFrom(var2) ? Status.REJECTED : Status.ALLOWED;

9c3fb41b-ae66-6e1c-b8eb-743ef1b286a3.png

没有给AnnotationInvocationHandler白名单,所以返回 REJECTED。

绕过JEP290

在RMI远程方法调用过程中,方法参数需要先序列化,从本地JVM发送到远程JVM,然后在远程JVM上反序列化,执行完后,将结果序列化,发送回本地JVM,而在本地的参数是我们可以控制的,如果向参数中注入gadget会怎么样?

我在HelloImpl实现了三个hello()方法,分别是void、string、ob ject类型的参数

0014989d-f8b5-04e9-185c-b30549bf80bf.png

在客户端我向ob ject参数类型注入cc5的gadget

3491f2fc-6d09-0920-dcc1-0ea93e9a8ca2.png

运行成功弹出calc

b8e0cbed-c3d3-3686-50be-29e670538d4d.png

也就是说:如果目标的RMI服务暴漏了ob ject参数类型的方法,我们就可以注入payload进去。

那么别的参数类型呢?在sun.rmi.server.UnicastRef#unmarshalValue中判断了远程调用方法的参数类型

bf226af0-f060-0ecf-8c03-8c4d1ba0c353.png

如果不是基本类型,就进入readob ject,之后的流程也走了filterCheck过滤

4e2aa34b-3df1-4e38-38fc-75736af33456.png

不过在sun.rmi.transport.DGCImpl#checkInput这里ObjID是在白名单中的,所以可以被反序列化。

那这个只是ob ject类型的参数可以,其他的参数类型呢?

由于攻击者可以完全控制客户端,因此他可以用恶意对象替换从ob ject类派生的参数(例如String)有几种方法:

  1. 将java.rmi软件包的代码复制到新软件包,然后在其中更改代码
  2. 将调试器附加到正在运行的客户端,并在序列化对象之前替换对象
  3. 使用Javassist之类的工具更改字节码
  4. 通过实现代理来替换网络流上已经序列化的对象

afanti师傅用的是通过 RASP hook住java.rmi.server.Remoteob jectInvocationHandler类的InvokeRemoteMethod方法的第三个参数非ob ject的改为ob ject的gadget。他的项目地址在Remoteob jectInvocationHandler

修改src\main\java\afanti\rasp\visitor\Remoteob jectInvocationHandlerHookVisitor.java的dnslog地址,然后打包出来在RMIClient运行前加上-javaagent:e:/rasp-1.0-SNAPSHOT.jar

虽然报错参数类型不匹配

030103fb-3147-c2a0-c1df-1e1a09b6b6a1.png

但是dnslog已经收到请求了。

6ff3b113-581f-7443-05a7-5b5aa3b7eff5.png

参考

  1. https://mogwailabs.de/blog/2019/03/attacking-java-rmi-services-after-jep-290/
  2. https://www.anquanke.com/post/id/200860
  3. https://github.com/Afant1/Remoteob jectInvocationHandler
  4. https://paper.seebug.org/454/

◆来源:Bypass JEP290

◆本文版权归原作者所有,如有侵权请联系我们及时删除

最新评论

昵称
邮箱
提交评论