All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.bytebuddy.dynamic.loading.ClassReloadingStrategy Maven / Gradle / Ivy

package net.bytebuddy.dynamic.loading;

import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.utility.StreamDrainer;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.instrument.*;
import java.security.ProtectionDomain;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * 

* The class reloading strategy allows to redefine loaded {@link java.lang.Class}es. Note that this strategy * will always attempt to load an existing class prior to its redefinition, even if this class is not yet loaded. *

*

* Note: In order to redefine any type, neither its name or its modifiers must be changed. Furthermore, no * fields or methods must be removed or added. This makes this strategy generally incompatible to applying it to a * rebased class definition as by {@link net.bytebuddy.ByteBuddy#rebase(Class)} which copies the original method * implementations to additional methods. Furthermore, even the {@link net.bytebuddy.ByteBuddy#redefine(Class)} * adds a method if the original class contains an explicit class initializer. For these reasons, it is not * recommended to use this {@link ClassLoadingStrategy} with arbitrary classes. *

*/ public class ClassReloadingStrategy implements ClassLoadingStrategy { /** * The name of the Byte Buddy agent class. */ private static final String BYTE_BUDDY_AGENT_TYPE = "net.bytebuddy.agent.ByteBuddyAgent"; /** * The name of the {@code ByteBuddyAgent} class's method for obtaining an instrumentation. */ private static final String GET_INSTRUMENTATION_METHOD = "getInstrumentation"; /** * Base for access to a reflective member to make the code more readable. */ private static final Object STATIC_METHOD = null; /** * The class file extension. */ private static final String CLASS_FILE_EXTENSION = ".class"; /** * This instance's instrumentation. */ private final Instrumentation instrumentation; /** * An engine which performs the actual redefinition of a {@link java.lang.Class}. */ private final Engine engine; /** * The strategy to apply for injecting classes into the bootstrap class loader. */ private final BootstrapInjection bootstrapInjection; /** * Creates a class reloading strategy for the given instrumentation. The given instrumentation must either * support {@link java.lang.instrument.Instrumentation#isRedefineClassesSupported()} or * {@link java.lang.instrument.Instrumentation#isRetransformClassesSupported()}. If both modes are supported, * classes will be transformed using a class redefinition. * * @param instrumentation The instrumentation to be used by this reloading strategy. */ public ClassReloadingStrategy(Instrumentation instrumentation) { this.instrumentation = instrumentation; if (instrumentation.isRedefineClassesSupported()) { engine = Engine.REDEFINITION; } else if (instrumentation.isRetransformClassesSupported()) { engine = Engine.RETRANSFORMATION; } else { throw new IllegalArgumentException("Instrumentation does not support class redefinition: " + instrumentation); } bootstrapInjection = BootstrapInjection.Disabled.INSTANCE; } /** * Creates a class reloading strategy for the given instrumentation using an explicit transformation strategy which * is represented by an {@link net.bytebuddy.dynamic.loading.ClassReloadingStrategy.Engine}. * * @param instrumentation The instrumentation to be used by this reloading strategy. * @param engine An engine which performs the actual redefinition of a {@link java.lang.Class}. */ public ClassReloadingStrategy(Instrumentation instrumentation, Engine engine) { this.instrumentation = instrumentation; this.engine = engine; bootstrapInjection = BootstrapInjection.Disabled.INSTANCE; } /** * Creates a new class reloading strategy. * * @param instrumentation The instrumentation to be used by this reloading strategy. * @param engine An engine which performs the actual redefinition of a {@link java.lang.Class}. * @param bootstrapInjection The bootstrap class loader injection strategy to use. */ protected ClassReloadingStrategy(Instrumentation instrumentation, Engine engine, BootstrapInjection bootstrapInjection) { this.instrumentation = instrumentation; this.engine = engine; this.bootstrapInjection = bootstrapInjection; } /** *

* Obtains a {@link net.bytebuddy.dynamic.loading.ClassReloadingStrategy} from an installed Byte Buddy agent. This * agent must be installed either by adding the {@code byte-buddy-agent.jar} when starting up the JVM by *

*

* * java -javaagent:byte-buddy-agent.jar -jar app.jar * *

* or after the start up using the Attach API. A convenience installer for the OpenJDK is provided by the * {@code ByteBuddyAgent} within the {@code byte-buddy-agent} module. * * @return A class reloading strategy which uses the Byte Buddy agent's {@link java.lang.instrument.Instrumentation}. */ public static ClassReloadingStrategy fromInstalledAgent() { try { return new ClassReloadingStrategy((Instrumentation) ClassLoader.getSystemClassLoader() .loadClass(BYTE_BUDDY_AGENT_TYPE) .getDeclaredMethod(GET_INSTRUMENTATION_METHOD) .invoke(STATIC_METHOD)); } catch (RuntimeException exception) { throw exception; } catch (Exception exception) { throw new IllegalStateException("The Byte Buddy agent is not installed or not accessible", exception); } } @Override public Map> load(ClassLoader classLoader, Map types) { Map> loadedClasses = new HashMap>(types.size()); Map unloadedClasses = new HashMap(types.size()); Map, ClassDefinition> classDefinitions = new ConcurrentHashMap, ClassDefinition>(types.size()); for (Map.Entry entry : types.entrySet()) { try { Class type = classLoader.loadClass(entry.getKey().getName()); classDefinitions.put(type, new ClassDefinition(type, entry.getValue())); loadedClasses.put(entry.getKey(), type); } catch (ClassNotFoundException ignored) { unloadedClasses.put(entry.getKey(), entry.getValue()); } } try { engine.apply(instrumentation, classDefinitions); ClassInjector classInjector = classLoader == null ? bootstrapInjection.make(instrumentation) : new ClassInjector.UsingReflection(classLoader); loadedClasses.putAll(classInjector.inject(unloadedClasses)); } catch (ClassNotFoundException exception) { throw new IllegalArgumentException("Could not locate classes for redefinition", exception); } catch (UnmodifiableClassException exception) { throw new IllegalStateException("Cannot redefine specified class", exception); } return loadedClasses; } /** * Resets all classes to their original definition. * * @param type The types to reset. * @return This class reloading strategy. */ public ClassReloadingStrategy reset(Class... type) { Map, ClassDefinition> classDefinitions = new ConcurrentHashMap, ClassDefinition>(type.length); try { for (Class aType : type) { InputStream inputStream = aType.getClassLoader().getResourceAsStream(aType.getName().replace('.', '/') + CLASS_FILE_EXTENSION); try { classDefinitions.put(aType, new ClassDefinition(aType, new StreamDrainer().drain(inputStream))); } finally { inputStream.close(); } } } catch (IOException exception) { throw new IllegalStateException("Exception while resetting types " + Arrays.toString(type), exception); } try { engine.apply(instrumentation, classDefinitions); } catch (ClassNotFoundException exception) { throw new IllegalArgumentException("Cannot locate types " + Arrays.toString(type), exception); } catch (UnmodifiableClassException exception) { throw new IllegalStateException("Cannot reset types " + Arrays.toString(type), exception); } return this; } /** * Enables bootstrap injection for this class reloading strategy. * * @param folder The folder to save jar files in that are appended to the bootstrap class path. * @return A class reloading strategy with bootstrap injection enabled. */ public ClassReloadingStrategy enableBootstrapInjection(File folder) { return new ClassReloadingStrategy(instrumentation, engine, new BootstrapInjection.Enabled(folder)); } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && engine == ((ClassReloadingStrategy) other).engine && instrumentation.equals(((ClassReloadingStrategy) other).instrumentation) && bootstrapInjection.equals(((ClassReloadingStrategy) other).bootstrapInjection); } @Override public int hashCode() { return 31 * 31 * instrumentation.hashCode() + 31 * engine.hashCode() + bootstrapInjection.hashCode(); } @Override public String toString() { return "ClassReloadingStrategy{" + "instrumentation=" + instrumentation + ", engine=" + engine + ", bootstrapInjection=" + bootstrapInjection + '}'; } /** * An engine which performs the actual redefinition of a {@link java.lang.Class}. */ public enum Engine { /** * Redefines a class using * {@link java.lang.instrument.Instrumentation#redefineClasses(java.lang.instrument.ClassDefinition...)}. */ REDEFINITION(true) { @Override protected void apply(Instrumentation instrumentation, Map, ClassDefinition> classDefinitions) throws UnmodifiableClassException, ClassNotFoundException { instrumentation.redefineClasses(classDefinitions.values().toArray(new ClassDefinition[classDefinitions.size()])); } }, /** * Redefines a class using * {@link java.lang.instrument.Instrumentation#retransformClasses(Class[])}. This requires synchronization on * the {@link net.bytebuddy.dynamic.loading.ClassReloadingStrategy#instrumentation} object. */ RETRANSFORMATION(false) { @Override protected void apply(Instrumentation instrumentation, Map, ClassDefinition> classDefinitions) throws UnmodifiableClassException { ClassRedefinitionTransformer classRedefinitionTransformer = new ClassRedefinitionTransformer(classDefinitions); synchronized (instrumentation) { instrumentation.addTransformer(classRedefinitionTransformer, REDEFINE_CLASSES); try { instrumentation.retransformClasses(classDefinitions.keySet().toArray(new Class[classDefinitions.size()])); } finally { instrumentation.removeTransformer(classRedefinitionTransformer); } } classRedefinitionTransformer.assertTransformation(); } }; /** * Instructs a {@link java.lang.instrument.ClassFileTransformer} to redefine classes. */ private static final boolean REDEFINE_CLASSES = true; /** * {@code true} if the {@link net.bytebuddy.dynamic.loading.ClassReloadingStrategy.Engine#REDEFINITION} engine * is used. */ private final boolean redefinition; /** * Creates a new engine. * * @param redefinition {@code true} if the {@link net.bytebuddy.dynamic.loading.ClassReloadingStrategy.Engine#REDEFINITION} engine * is used. */ Engine(boolean redefinition) { this.redefinition = redefinition; } /** * Applies this engine for the given arguments. * * @param instrumentation The instrumentation to be used for applying the redefinition. * @param classDefinitions A mapping of the classes to be redefined to their redefinition. * @throws UnmodifiableClassException If a class is not modifiable. * @throws ClassNotFoundException If a class was not found. */ protected abstract void apply(Instrumentation instrumentation, Map, ClassDefinition> classDefinitions) throws UnmodifiableClassException, ClassNotFoundException; /** * Returns {@code true} if this engine represents {@link net.bytebuddy.dynamic.loading.ClassReloadingStrategy.Engine#REDEFINITION}. * * @return {@code true} if this engine represents {@link net.bytebuddy.dynamic.loading.ClassReloadingStrategy.Engine#REDEFINITION}. */ public boolean isRedefinition() { return redefinition; } @Override public String toString() { return "ClassReloadingStrategy.Engine." + name(); } /** * A class file transformer that applies a given {@link java.lang.instrument.ClassDefinition}. */ protected static class ClassRedefinitionTransformer implements ClassFileTransformer { /** * A mapping of classes to be redefined to their redefined class definitions. */ private final Map, ClassDefinition> redefinedClasses; /** * Creates a new class redefinition transformer. * * @param redefinedClasses A mapping of classes to be redefined to their redefined class definitions. */ protected ClassRedefinitionTransformer(Map, ClassDefinition> redefinedClasses) { this.redefinedClasses = redefinedClasses; } @Override public byte[] transform(ClassLoader loader, String className, Class classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException { ClassDefinition redefinedClass = redefinedClasses.remove(classBeingRedefined); if (redefinedClass == null) { throw new IllegalArgumentException("Encountered class without redefinition information"); } return redefinedClass.getDefinitionClassFile(); } /** * Validates that all given classes were redefined. */ public void assertTransformation() { if (!redefinedClasses.isEmpty()) { throw new IllegalStateException("Could not transform: " + redefinedClasses.keySet()); } } @Override public String toString() { return "ClassReloadingStrategy.Engine.ClassRedefinitionTransformer{redefinedClasses=" + redefinedClasses + '}'; } } } /** * A strategy to apply for injecting classes into the bootstrap class loader. */ protected interface BootstrapInjection { /** * Creates a class injector to use. * * @param instrumentation The instrumentation of this instance. * @return A class injector for the bootstrap class loader. */ ClassInjector make(Instrumentation instrumentation); /** * A disabled bootstrap injection strategy. */ enum Disabled implements BootstrapInjection { /** * The singleton instance. */ INSTANCE; @Override public ClassInjector make(Instrumentation instrumentation) { throw new IllegalStateException("Bootstrap injection is not enabled"); } @Override public String toString() { return "ClassReloadingStrategy.BootstrapInjection.Disabled." + name(); } } /** * An enabled bootstrap class loader injection strategy. */ class Enabled implements BootstrapInjection { /** * The folder to save jar files in. */ private final File folder; /** * Creates an enabled bootstrap class injection strategy. * * @param folder The folder to save jar files in. */ protected Enabled(File folder) { this.folder = folder; } @Override public ClassInjector make(Instrumentation instrumentation) { return ClassInjector.UsingInstrumentation.of(folder, ClassInjector.UsingInstrumentation.Target.BOOTSTRAP, instrumentation); } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && folder.equals(((Enabled) other).folder); } @Override public int hashCode() { return folder.hashCode(); } @Override public String toString() { return "ClassReloadingStrategy.BootstrapInjection.Enabled{" + "folder=" + folder + '}'; } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy