Java Agent

JDK1.5开始,Java新增了Instrumentation(Java Agent API)JVMTI(JVM Tool Interface)功能,允许JVM在加载某个class文件之前对其字节码进行修改,同时也支持对已加载的class(类字节码)进行重新加载(Retransform)。

利用Java Agent这一特性衍生出了APM(Application Performance Management,应用性能管理)RASP(Runtime application self-protection,运行时应用自我保护)IAST(Interactive Application Security Testing,交互式应用程序安全测试)等相关产品,它们都无一例外的使用了Instrumentation/JVMTIAPI来实现动态修改Java类字节码并插入监控或检测代码。

Java Agent有两种运行模式:

  1. 启动Java程序时添加-javaagent(Instrumentation API实现方式)-agentpath/-agentlib(JVMTI的实现方式)参数,如java -javaagent:/data/XXX.jar LingXeTest
  2. JDK1.6新增了attach(附加方式)方式,可以对运行中的Java进程附加Agent

这两种运行方式的最大区别在于第一种方式只能在程序启动时指定Agent文件,而attach方式可以在Java程序运行后根据进程ID动态注入AgentJVM

Java Agent Hello World

让我们来运行一个JavaHelloWorld程序。

HelloWorld示例代码:

  1. package com.anbai.sec.agent;
  2. /**
  3. * Creator: yz
  4. * Date: 2020/1/2
  5. */
  6. public class HelloWorld {
  7. public static void main(String[] args) {
  8. System.out.println("Hello World...");
  9. }
  10. }

程序运行结果:

  1. Hello World...

假设我们现在有一个需求:必须在不重新编译某个类的情况下(甚至有可能是不重启应用服务的情况下)动态的改变类方法的执行逻辑是非常困难的,但如果使用AgentInstrumentation API就可以非常容易的实现了,例如将下列程序(HelloWorld.java)的输出变成Hello Agent...

首先我们需要修改:javaweb-sec/javaweb-sec-source/javasec-agent/src/main/resources/MANIFEST.MF文件中的Premain-Class配置为com.anbai.sec.agent.JavaSecHelloWorldAgent,然后再执行如下命令使用Maven构建Agent Jar包:

  1. cd javaweb-sec/javaweb-sec-source/javasec-agent
  2. mvn clean install

Maven构建完成后在javaweb-sec/javaweb-sec-source/javasec-agent/target目录会自动生成一个javasec-agent.jar文件,这个文件也就是我们写好的用于处理HelloWorld程序输出结果的Java Agent程序。

JavaSecHelloWorldAgent动态替换HelloWorld字符串示例代码:

  1. /*
  2. * 灵蜥Java Agent版 [Web应用安全智能防护系统]
  3. * ----------------------------------------------------------------------
  4. * Copyright © 安百科技(北京)有限公司
  5. */
  6. package com.anbai.sec.agent;
  7. import java.lang.instrument.ClassFileTransformer;
  8. import java.lang.instrument.Instrumentation;
  9. import java.security.ProtectionDomain;
  10. import java.util.Arrays;
  11. /**
  12. * Creator: yz
  13. * Date: 2020/1/2
  14. */
  15. public class JavaSecHelloWorldAgent {
  16. /**
  17. * 替换HelloWorld的输出字符串为"Hello Agent...",将二进制转换成字符串数组,替换字符串数组并生成新的二进制
  18. *
  19. * @param className 类名
  20. * @param classBuffer 类字节码
  21. * @return 替换后的类字节码
  22. */
  23. private static byte[] replaceBytes(String className, byte[] classBuffer) {
  24. // 将类字节码转换成byte字符串
  25. String bufferStr = Arrays.toString(classBuffer);
  26. System.out.println(className + "类替换前的字节码:" + bufferStr);
  27. bufferStr = bufferStr.replace("[", "").replace("]", "");
  28. // 查找需要替换的Java二进制内容
  29. byte[] findBytes = "Hello World...".getBytes();
  30. // 把搜索的字符串byte转换成byte字符串
  31. String findStr = Arrays.toString(findBytes).replace("[", "").replace("]", "");
  32. // 二进制替换后的byte值,注意这个值需要和替换的字符串长度一致,不然会破坏常量池
  33. byte[] replaceBytes = "Hello Agent...".getBytes();
  34. // 把替换的字符串byte转换成byte字符串
  35. String replaceStr = Arrays.toString(replaceBytes).replace("[", "").replace("]", "");
  36. bufferStr = bufferStr.replace(findStr, replaceStr);
  37. // 切割替换后的byte字符串
  38. String[] byteArray = bufferStr.split("\\s*,\\s*");
  39. // 创建新的byte数组,存储替换后的二进制
  40. byte[] bytes = new byte[byteArray.length];
  41. // 将byte字符串转换成byte
  42. for (int i = 0; i < byteArray.length; i++) {
  43. bytes[i] = Byte.parseByte(byteArray[i]);
  44. }
  45. System.out.println(className + "类替换后的字节码:" + Arrays.toString(bytes));
  46. // 返回修改后的二进制
  47. return bytes;
  48. }
  49. /**
  50. * Java Agent模式入口
  51. *
  52. * @param args 命令参数
  53. * @param inst Agent Instrumentation 实例
  54. */
  55. public static void premain(String args, final Instrumentation inst) {
  56. // 添加自定义的Transformer
  57. inst.addTransformer(new ClassFileTransformer() {
  58. /**
  59. * 类文件转换方法,重写transform方法可获取到待加载的类相关信息
  60. *
  61. * @param loader 定义要转换的类加载器;如果是引导加载器,则为 null
  62. * @param className 类名,如:java/lang/Runtime
  63. * @param classBeingRedefined 如果是被重定义或重转换触发,则为重定义或重转换的类;如果是类加载,则为 null
  64. * @param protectionDomain 要定义或重定义的类的保护域
  65. * @param classfileBuffer 类文件格式的输入字节缓冲区(不得修改)
  66. * @return 返回一个通过ASM修改后添加了防御代码的字节码byte数组。
  67. */
  68. @Override
  69. public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
  70. ProtectionDomain protectionDomain, byte[] classfileBuffer) {
  71. // 将目录路径替换成Java类名
  72. className = className.replace("/", ".");
  73. // 只处理com.anbai.sec.agent.HelloWorld类的字节码
  74. if (className.equals("com.anbai.sec.agent.HelloWorld")) {
  75. // 替换HelloWorld的输出字符串
  76. return replaceBytes(className, classfileBuffer);
  77. }
  78. return classfileBuffer;
  79. }
  80. }, true);// 第二个参数true表示是否允许Agent Retransform,需配合MANIFEST.MF中的Can-Retransform-Classes: true配置
  81. }
  82. }

我们需要在运行HelloWorld的时候添加-javaagent:jar路径参数,例如:

  1. java -jar -javaagent:/Users/yz/IdeaProjects/javaweb-sec/javaweb-sec-source/javasec-agent/target/javasec-agent.jar com.anbai.sec.agent.HelloWorld

程序执行结果:

  1. com.anbai.sec.agent.HelloWorld类替换前的字节码:[....省去两则一致的数据, 87, 111, 114, 108, 100, ...]
  2. com.anbai.sec.agent.HelloWorld类替换后的字节码:[....省去两则一致的数据, 65, 103, 101, 110, 116, ...]
  3. Hello Agent...

由上可以看到程序的最终执行结果已经被我们动态的修改为了:Hello Agent...,这种方式是最为简单暴力的修改二进制中的字符串值的方式在实际的业务场景下很显然是不可行的,因为只要修改后的字符串长度不一致就会破坏常量池导致程序无法执行,为了能够精准有效的修改类字节码我们通常会使用ASM库。

Instrumentation

java.lang.instrument.InstrumentationJava提供的监测运行在JVM程序的API。利用Instrumentation我们可以实现如下功能:

  1. 动态添加自定义的Transformer(addTransformer)
  2. 动态修改classpath(appendToBootstrapClassLoaderSearch、appendToSystemClassLoaderSearch)
  3. 动态获取所有JVM已加载的类(getAllLoadedClasses)。
  4. 动态获取某个类加载器已实例化的所有类(getInitiatedClasses)。
  5. 直接修改已加载的类的字节码(redefineClasses)。
  6. 动态设置JNI前缀(setNativeMethodPrefix)。
  7. 重加载指定类字节码(retransformClasses)。

Instrumentation类方法如下:

07EC4F97-CD49-41E6-95CE-FEB000325E33

ClassFileTransformer

java.lang.instrument.ClassFileTransformer是一个转换类文件的代理接口,我们可以在获取到Instrumentation对象后通过addTransformer方法添加自定义类文件转换器。

示例中我们使用了addTransformer注册了一个我们自定义的TransformerJava Agent,当有新的类被JVM加载时JVM会自动回调用我们自定义的Transformer类的transform方法,传入该类的transform信息(类名、类加载器、类字节码等),我们可以根据传入的类信息决定是否需要修改类字节码,修改完字节码后我们将新的类字节码返回给JVMJVM会验证类和相应的修改是否合法,如果符合类加载要求JVM会加载我们修改后的类字节码。

ClassFileTransformer类代码:

  1. package java.lang.instrument;
  2. public interface ClassFileTransformer {
  3. /**
  4. * 类文件转换方法,重写transform方法可获取到待加载的类相关信息
  5. *
  6. * @param loader 定义要转换的类加载器;如果是引导加载器,则为 null
  7. * @param className 类名,如:java/lang/Runtime
  8. * @param classBeingRedefined 如果是被重定义或重转换触发,则为重定义或重转换的类;如果是类加载,则为 null
  9. * @param protectionDomain 要定义或重定义的类的保护域
  10. * @param classfileBuffer 类文件格式的输入字节缓冲区(不得修改)
  11. * @return 返回一个通过ASM修改后添加了防御代码的字节码byte数组。
  12. */
  13. byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
  14. ProtectionDomain protectionDomain, byte[] classfileBuffer);
  15. }

重写transform方法需要注意以下事项:

  1. ClassLoader如果是被Bootstrap ClassLoader(引导类加载器)所加载那么loader参数的值是空。
  2. 修改类字节码时需要特别注意插入的代码在对应的ClassLoader中可以正确的获取到,否则会报ClassNotFoundException,比如修改java.io.FileInputStream(该类由Bootstrap ClassLoader加载)时插入了我们检测代码,那么我们将必须保证FileInputStream能够获取到我们的检测代码类。
  3. JVM类名的书写方式路径方式:java/lang/String而不是我们常用的类名方式:java.lang.String
  4. 类字节必须符合JVM校验要求,如果无法验证类字节码会导致JVM崩溃或者VerifyError(类验证错误)
  5. 如果修改的是retransform类(修改已被JVM加载的类),修改后的类字节码不得新增方法修改方法参数类成员变量
  6. addTransformer时如果没有传入retransform参数(默认是false)就算MANIFEST.MF中配置了Can-Redefine-Classes: true而且手动调用了retransformClasses方法也一样无法retransform
  7. 卸载transform时需要使用创建时的Instrumentation实例。