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

net.bytebuddy.agent.builder.AgentBuilder Maven / Gradle / Ivy

Go to download

Byte Buddy is a Java library for creating Java classes at run time. This artifact is a build of Byte Buddy with all ASM dependencies repackaged into its own name space.

There is a newer version: 1.15.11
Show newest version
package net.bytebuddy.agent.builder;

import net.bytebuddy.ByteBuddy;
import net.bytebuddy.description.method.MethodDescription;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.dynamic.ClassFileLocator;
import net.bytebuddy.dynamic.DynamicType;
import net.bytebuddy.dynamic.loading.ClassInjector;
import net.bytebuddy.dynamic.scaffold.InstrumentedType;
import net.bytebuddy.dynamic.scaffold.inline.MethodRebaseResolver;
import net.bytebuddy.implementation.Implementation;
import net.bytebuddy.implementation.LoadedTypeInitializer;
import net.bytebuddy.implementation.bytecode.ByteCodeAppender;
import net.bytebuddy.implementation.bytecode.Removal;
import net.bytebuddy.implementation.bytecode.StackManipulation;
import net.bytebuddy.implementation.bytecode.collection.ArrayFactory;
import net.bytebuddy.implementation.bytecode.constant.ClassConstant;
import net.bytebuddy.implementation.bytecode.constant.NullConstant;
import net.bytebuddy.implementation.bytecode.constant.TextConstant;
import net.bytebuddy.implementation.bytecode.member.MethodInvocation;
import net.bytebuddy.matcher.ElementMatcher;
import net.bytebuddy.pool.TypePool;
import net.bytebuddy.utility.StreamDrainer;
import net.bytebuddy.jar.asm.MethodVisitor;

import java.io.File;
import java.io.IOException;
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.Instrumentation;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.security.ProtectionDomain;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

import static net.bytebuddy.matcher.ElementMatchers.*;
import static net.bytebuddy.utility.ByteBuddyCommons.join;
import static net.bytebuddy.utility.ByteBuddyCommons.nonNull;

/**
 * 

* An agent builder provides a convenience API for defining a * Java agent using * Byte Buddy's * {@link net.bytebuddy.ByteBuddy#rebase(TypeDescription, ClassFileLocator)}. *

*

* When defining several {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s, the agent builder always * applies the transformers that were supplied with the last applicable matcher. Therefore, more general transfromers * should be defined first. *

*/ public interface AgentBuilder { /** * Matches a type being loaded in order to apply the supplied * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s before loading this type. * * @param matcher A matcher that decides if the entailed * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s should be applied for a type that * is being loaded. * @return A definable that represents this agent builder which allows for the definition of one or several * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s to be applied when the given {@code matcher} * indicates a match. */ Identified rebase(RawMatcher matcher); /** * Matches a type being loaded in order to apply the supplied * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s before loading this type. * * @param typeMatcher An {@link net.bytebuddy.matcher.ElementMatcher} that is applied on the type being loaded that * decides if the entailed * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s should be applied for that * type. * @return A definable that represents this agent builder which allows for the definition of one or several * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s to be applied when the given {@code typeMatcher} * indicates a match. */ Identified rebase(ElementMatcher typeMatcher); /** * Matches a type being loaded in order to apply the supplied * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s before loading this type. * * @param typeMatcher An {@link net.bytebuddy.matcher.ElementMatcher} that is applied on the type being * loaded that decides if the entailed * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s should be applied for * that type. * @param classLoaderMatcher An {@link net.bytebuddy.matcher.ElementMatcher} that is applied to the * {@link java.lang.ClassLoader} that is loading the type being loaded. This matcher * is always applied first where the type matcher is not applied in case that this * matcher does not indicate a match. * @return A definable that represents this agent builder which allows for the definition of one or several * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s to be applied when both the given * {@code typeMatcher} and {@code classLoaderMatcher} indicate a match. */ Identified rebase(ElementMatcher typeMatcher, ElementMatcher classLoaderMatcher); /** * Defines the given {@link net.bytebuddy.ByteBuddy} instance to be used by the created agent. * * @param byteBuddy The Byte Buddy instance to be used. * @return A new instance of this agent builder which makes use of the given {@code byteBuddy} instance. */ AgentBuilder withByteBuddy(ByteBuddy byteBuddy); /** * Defines the given {@link net.bytebuddy.agent.builder.AgentBuilder.Listener} to be notified by the created agent. * The given listener is notified after any other listener that is already registered. If a listener is registered * twice, it is also notified twice. * * @param listener The listener to be notified. * @return A new instance of this agent builder which creates an agent that informs the given listener about * events. */ AgentBuilder withListener(Listener listener); /** * Defines the use of the given binary locator for locating binary data to given class names. * * @param binaryLocator The binary locator to use. * @return A new instance of this agent builder which uses the given binary locator for looking up class files. */ AgentBuilder withBinaryLocator(BinaryLocator binaryLocator); /** * Enables the use of the given native method prefix for instrumented methods. Note that this prefix is also * applied when preserving non-native methods. The use of this prefix is also registered when installing the * final agent with an {@link java.lang.instrument.Instrumentation}. * * @param prefix The prefix to be used. * @return A new instance of this agent builder which uses the given native method prefix. */ AgentBuilder withNativeMethodPrefix(String prefix); /** *

* Disables the execution of any {@link net.bytebuddy.implementation.LoadedTypeInitializer}s that are registered * with a {@link net.bytebuddy.dynamic.DynamicType}. This might cause the dynamic type to malfunction if the * {@link net.bytebuddy.implementation.LoadedTypeInitializer} are not executed elsewhere before an instrumented * type is put in use for the first time. *

*

* In order to execute a self initialization, Byte Buddy adds a call back into any dynamic type's type initializer. * This call back requires the injection of a call back dispatcher into the system class loader what might not * be a feasible solution on distributed applications where classes are shared among different JVMs where a * different strategy for executing {@link net.bytebuddy.implementation.LoadedTypeInitializer}s might be * more appropriate. *

* * @return A new instance of this agent builder which does not apply self initialization. */ AgentBuilder disableSelfInitialization(); /** * Enables retransformation when this agent is installed. Note that retransformation on does not currently allow * for adding or removing fields or methods on the Hot Spot Virtual machine. * * @return A new instance of this agent builder which does not apply self initialization. */ AgentBuilder allowRetransformation(); /** * Enables class injection of auxiliary classes into the bootstrap class loader. * * @param folder The folder in which jar files of the injected classes are to be stored. * @param instrumentation The instrumentation instance that is used for appending jar files to the * bootstrap class path. * @return An agent builder with bootstrap class loader class injection enabled. */ AgentBuilder enableBootstrapInjection(File folder, Instrumentation instrumentation); /** * Creates a {@link java.lang.instrument.ClassFileTransformer} that implements the configuration of this * agent builder. * * @return A class file transformer that implements the configuration of this agent builder. */ ClassFileTransformer makeRaw(); /** * Creates and installs a {@link java.lang.instrument.ClassFileTransformer} that implements the configuration of * this agent builder with a given {@link java.lang.instrument.Instrumentation}. * * @param instrumentation The instrumentation on which this agent builder's configuration is to be installed. * @return The installed class file transformer. */ ClassFileTransformer installOn(Instrumentation instrumentation); /** * Creates and installs a {@link java.lang.instrument.ClassFileTransformer} that implements the configuration of * this agent builder with the Byte Buddy-agent which must be installed prior to calling this method. * * @return The installed class file transformer. */ ClassFileTransformer installOnByteBuddyAgent(); /** * Describes an {@link net.bytebuddy.agent.builder.AgentBuilder} which was handed a matcher for identifying * types to instrumented in order to supply one or several * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s. */ interface Identified { /** * Applies the given transformer for the already supplied matcher. * * @param transformer The transformer to apply. * @return This agent builder with the transformer being applied when the previously supplied matcher * identified a type for instrumentation which also allows for the registration of subsequent transformers. */ Extendable transform(Transformer transformer); /** * This interface is used to allow for optionally providing several * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer} to applied when a matcher identifies a type * to be instrumented. Any subsequent transformers are applied in the order they are registered. */ interface Extendable extends AgentBuilder, Identified { /* this is merely a unionizing interface that does not declare methods */ } } /** * A matcher that allows to determine if a {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer} * should be applied during the execution of a {@link java.lang.instrument.ClassFileTransformer} that was * generated by an {@link net.bytebuddy.agent.builder.AgentBuilder}. */ interface RawMatcher { /** * Decides if the given {@code typeDescription} should be instrumented with the entailed * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s. * * @param typeDescription A description of the type to be instrumented. * @param classLoader The class loader of the instrumented type. Might be {@code null} if this class * loader represents the bootstrap class loader. * @param classBeingRedefined The class being redefined which is only not {@code null} if a retransofmration * is applied. * @param protectionDomain The protection domain of the type being transformed. * @return {@code true} if the entailed {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s should * be applied for the given {@code typeDescription}. */ boolean matches(TypeDescription typeDescription, ClassLoader classLoader, Class classBeingRedefined, ProtectionDomain protectionDomain); /** * A raw matcher implementation that checks a {@link TypeDescription} * and its {@link java.lang.ClassLoader} against two suitable matchers in order to determine if the matched * type should be instrumented. */ class ForElementMatcherPair implements RawMatcher { /** * The type matcher to apply to a {@link TypeDescription}. */ private final ElementMatcher typeMatcher; /** * The class loader to apply to a {@link java.lang.ClassLoader}. */ private final ElementMatcher classLoaderMatcher; /** * Creates a new {@link net.bytebuddy.agent.builder.AgentBuilder.RawMatcher} that only matches the * supplied {@link TypeDescription} and its * {@link java.lang.ClassLoader} against two matcher in order to decied if an instrumentation should * be conducted. * * @param typeMatcher The type matcher to apply to a * {@link TypeDescription}. * @param classLoaderMatcher The class loader to apply to a {@link java.lang.ClassLoader}. */ public ForElementMatcherPair(ElementMatcher typeMatcher, ElementMatcher classLoaderMatcher) { this.typeMatcher = typeMatcher; this.classLoaderMatcher = classLoaderMatcher; } @Override public boolean matches(TypeDescription typeDescription, ClassLoader classLoader, Class classBeingRedefined, ProtectionDomain protectionDomain) { return classLoaderMatcher.matches(classLoader) && typeMatcher.matches(typeDescription); } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && classLoaderMatcher.equals(((ForElementMatcherPair) other).classLoaderMatcher) && typeMatcher.equals(((ForElementMatcherPair) other).typeMatcher); } @Override public int hashCode() { int result = typeMatcher.hashCode(); result = 31 * result + classLoaderMatcher.hashCode(); return result; } @Override public String toString() { return "AgentBuilder.RawMatcher.ForElementMatcherPair{" + "typeMatcher=" + typeMatcher + ", classLoaderMatcher=" + classLoaderMatcher + '}'; } } } /** * A transformer allows to apply modifications to a {@link net.bytebuddy.dynamic.DynamicType}. Such a modification * is then applied to any instrumented type that was matched by the preceding matcher. */ interface Transformer { /** * Allows for a transformation of a {@link net.bytebuddy.dynamic.DynamicType.Builder}. * * @param builder The dynamic builder to transform. * @param typeDescription The description of the type currently being instrumented. * @return A transformed version of the supplied {@code builder}. */ DynamicType.Builder transform(DynamicType.Builder builder, TypeDescription typeDescription); /** * A no-op implementation of a {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer} that does * not modify the supplied dynamic type. */ enum NoOp implements Transformer { /** * The singleton instance. */ INSTANCE; @Override public DynamicType.Builder transform(DynamicType.Builder builder, TypeDescription typeDescription) { return builder; } @Override public String toString() { return "AgentBuilder.Transformer.NoOp." + name(); } } /** * A compound transformer that allows to group several * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s as a single transformer. */ class Compound implements Transformer { /** * The transformers to apply in their application order. */ private final Transformer[] transformer; /** * Creates a new compound transformer. * * @param transformer The transformers to apply in their application order. */ public Compound(Transformer... transformer) { this.transformer = transformer; } @Override public DynamicType.Builder transform(DynamicType.Builder builder, TypeDescription typeDescription) { for (Transformer transformer : this.transformer) { builder = transformer.transform(builder, typeDescription); } return builder; } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && Arrays.equals(transformer, ((Compound) other).transformer); } @Override public int hashCode() { return Arrays.hashCode(transformer); } @Override public String toString() { return "AgentBuilder.Transformer.Compound{" + "transformer=" + Arrays.toString(transformer) + '}'; } } } /** * A binary locator allows to specify how binary data is located by an * {@link net.bytebuddy.agent.builder.AgentBuilder}. */ interface BinaryLocator { /** * Initializes this binary locator. * * @param typeName The binary name of the type that is being instrumented. * @param binaryRepresentation The binary representation of the instrumented type. * @param classLoader The class loader of the instrumented type. Might be {@code null} if this class * loader represents the bootstrap class loader. * @return This binary locator in its initialized form. */ Initialized initialize(String typeName, byte[] binaryRepresentation, ClassLoader classLoader); /** * A default implementation of a {@link net.bytebuddy.agent.builder.AgentBuilder.BinaryLocator} that * is using a {@link net.bytebuddy.pool.TypePool.Default} with a * {@link net.bytebuddy.pool.TypePool.CacheProvider.Simple} and a * {@link net.bytebuddy.dynamic.ClassFileLocator.ForClassLoader}. */ enum Default implements BinaryLocator { /** * The singleton instance. */ INSTANCE; @Override public BinaryLocator.Initialized initialize(String typeName, byte[] binaryRepresentation, ClassLoader classLoader) { return new Initialized(typeName, binaryRepresentation, new TypePool.CacheProvider.Simple(), ClassFileLocator.ForClassLoader.of(classLoader)); } @Override public String toString() { return "AgentBuilder.BinaryLocator.Default." + name(); } /** * The {@link net.bytebuddy.agent.builder.AgentBuilder.BinaryLocator.Default} in its initialized form. */ protected static class Initialized implements BinaryLocator.Initialized, ClassFileLocator { /** * The binary name of the instrumented type. */ private final String typeName; /** * The binary representation of the instrumented type. */ private final byte[] binaryRepresentation; /** * The class file locator to use. */ private final ClassFileLocator classFileLocator; /** * The type pool to use. */ private final TypePool typePool; /** * Creates a new initialized form of a default binary locator. * * @param typeName The binary name of the type that is being instrumented. * @param binaryRepresentation The binary representation of the instrumented type. * @param cacheProvider The cache provider to use. * @param classFileLocator The class file locator to use. */ public Initialized(String typeName, byte[] binaryRepresentation, TypePool.CacheProvider cacheProvider, ClassFileLocator classFileLocator) { this.typeName = typeName; this.binaryRepresentation = binaryRepresentation; typePool = new TypePool.Default(cacheProvider, classFileLocator); this.classFileLocator = classFileLocator; } @Override public TypePool getTypePool() { return typePool; } @Override public ClassFileLocator getClassFileLocator() { return this; } @Override public ClassFileLocator.Resolution locate(String typeName) throws IOException { return this.typeName.equals(typeName) ? new ClassFileLocator.Resolution.Explicit(binaryRepresentation) : classFileLocator.locate(typeName); } @Override public boolean equals(Object other) { if (this == other) return true; if (other == null || getClass() != other.getClass()) return false; Initialized that = (Initialized) other; return Arrays.equals(binaryRepresentation, that.binaryRepresentation) && classFileLocator.equals(that.classFileLocator) && typeName.equals(that.typeName) && typePool.equals(that.typePool); } @Override public int hashCode() { int result = typeName.hashCode(); result = 31 * result + Arrays.hashCode(binaryRepresentation); result = 31 * result + classFileLocator.hashCode(); result = 31 * result + typePool.hashCode(); return result; } @Override public String toString() { return "AgentBuilder.BinaryLocator.Default.Initialized{" + "typeName='" + typeName + '\'' + ", binaryRepresentation=<" + binaryRepresentation.length + " bytes>" + ", classFileLocator=" + classFileLocator + ", typePool=" + typePool + '}'; } } } /** * A {@link net.bytebuddy.agent.builder.AgentBuilder.BinaryLocator} in initialized state. */ interface Initialized { /** * Returns the type pool to be used of an {@link net.bytebuddy.agent.builder.AgentBuilder}. * * @return The type pool to use. */ TypePool getTypePool(); /** * Returns the class file locator to be used of an {@link net.bytebuddy.agent.builder.AgentBuilder}. * * @return The class file locator to use. */ ClassFileLocator getClassFileLocator(); } } /** * A listener that is informed about events that occur during an instrumentation process. */ interface Listener { /** * Invoked right before a successful transformation is applied. * * @param typeDescription The type that is being transformed. * @param dynamicType The dynamic type that was created. */ void onTransformation(TypeDescription typeDescription, DynamicType dynamicType); /** * Invoked when an error has occurred. * * @param typeName The binary name of the instrumented type. * @param throwable The occurred error. */ void onError(String typeName, Throwable throwable); /** * Invokes when a type is not transformed. * * @param typeName The binary name of the type. */ void onIgnored(String typeName); /** * Invoked after a class was attempted to be loaded, independently of its treatment. * * @param typeName The binary name of the instrumented type. */ void onComplete(String typeName); /** * A no-op implementation of a {@link net.bytebuddy.agent.builder.AgentBuilder.Listener}. */ enum NoOp implements Listener { /** * The singleton instance. */ INSTANCE; @Override public void onTransformation(TypeDescription typeDescription, DynamicType dynamicType) { /* do nothing */ } @Override public void onError(String typeName, Throwable throwable) { /* do nothing */ } @Override public void onIgnored(String typeName) { /* do nothing */ } @Override public void onComplete(String typeName) { /* do nothing */ } @Override public String toString() { return "AgentBuilder.Listener.NoOp." + name(); } } /** * A compound listener that allows to group several listeners in one instance. */ class Compound implements Listener { /** * The listeners that are represented by this compound listener in their application order. */ private final Listener[] listener; /** * Creates a new compound listener. * * @param listener The listeners to apply in their application order. */ public Compound(Listener... listener) { this.listener = listener; } @Override public void onTransformation(TypeDescription typeDescription, DynamicType dynamicType) { for (Listener listener : this.listener) { listener.onTransformation(typeDescription, dynamicType); } } @Override public void onError(String typeName, Throwable throwable) { for (Listener listener : this.listener) { listener.onError(typeName, throwable); } } @Override public void onIgnored(String typeName) { for (Listener listener : this.listener) { listener.onIgnored(typeName); } } @Override public void onComplete(String typeName) { for (Listener listener : this.listener) { listener.onComplete(typeName); } } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && Arrays.equals(listener, ((Compound) other).listener); } @Override public int hashCode() { return Arrays.hashCode(listener); } @Override public String toString() { return "AgentBuilder.Listener.Compound{" + "listener=" + Arrays.toString(listener) + '}'; } } } /** * The default implementation of an {@link net.bytebuddy.agent.builder.AgentBuilder}. */ class Default implements AgentBuilder { /** * The string value that is used to indicate that no name prefix for native methods should be used. * This value is not a valid prefix. */ protected static final String NO_NATIVE_PREFIX = ""; /** * 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 value that is to be returned from a {@link java.lang.instrument.ClassFileTransformer} to indicate * that no class file transformation is to be applied. */ private static final byte[] NO_TRANSFORMATION = null; /** * The {@link net.bytebuddy.ByteBuddy} instance to be used. */ private final ByteBuddy byteBuddy; /** * The binary locator to use. */ private final BinaryLocator binaryLocator; /** * The listener to notify on transformations. */ private final Listener listener; /** * The native method prefix to use which might also represent * {@link net.bytebuddy.agent.builder.AgentBuilder.Default#NO_NATIVE_PREFIX} to indicate that no * prefix should be added but rather a random suffix. */ private final String nativeMethodPrefix; /** * {@code true} if generated types should not create a callback inside their type initializer in order * to call their potential {@link net.bytebuddy.implementation.LoadedTypeInitializer}. */ private final boolean disableSelfInitialization; /** * {@code true} if the generated {@link java.lang.instrument.ClassFileTransformer} should also apply for * retransformations.. */ private final boolean retransformation; /** * The injection strategy for injecting classes into the bootstrap class loader. */ private final BootstrapInjectionStrategy bootstrapInjectionStrategy; /** * The list of transformation entries that are registered with this agent builder. */ private final List entries; /** * Creates a new default agent builder that uses a default {@link net.bytebuddy.ByteBuddy} instance for * creating classes. */ public Default() { this(new ByteBuddy()); } /** * Creates a new default agent builder. * * @param byteBuddy The Byte Buddy instance to be used. */ public Default(ByteBuddy byteBuddy) { this(nonNull(byteBuddy), BinaryLocator.Default.INSTANCE, Listener.NoOp.INSTANCE, NO_NATIVE_PREFIX, false, false, BootstrapInjectionStrategy.Disabled.INSTANCE, Collections.emptyList()); } /** * Creates a new default agent builder. * * @param byteBuddy The Byte Buddy instance to be used. * @param binaryLocator The binary locator to use. * @param listener The listener to notify on transformations. * @param nativeMethodPrefix The native method prefix to use which might also represent * {@link net.bytebuddy.agent.builder.AgentBuilder.Default#NO_NATIVE_PREFIX} * to indicate that no prefix should be added but rather a random suffix. * @param disableSelfInitialization {@code true} if generated types should not create a callback inside their * type initializer in order to call their potential * {@link net.bytebuddy.implementation.LoadedTypeInitializer}. * @param retransformation {@code true} if the generated * {@link java.lang.instrument.ClassFileTransformer} should also apply * for retransformations. * @param bootstrapInjectionStrategy The injection strategy for injecting classes into the bootstrap class loader. * @param entries The list of transformation entries that are registered with this * agent builder. */ protected Default(ByteBuddy byteBuddy, BinaryLocator binaryLocator, Listener listener, String nativeMethodPrefix, boolean disableSelfInitialization, boolean retransformation, BootstrapInjectionStrategy bootstrapInjectionStrategy, List entries) { this.byteBuddy = byteBuddy; this.binaryLocator = binaryLocator; this.listener = listener; this.nativeMethodPrefix = nativeMethodPrefix; this.disableSelfInitialization = disableSelfInitialization; this.retransformation = retransformation; this.bootstrapInjectionStrategy = bootstrapInjectionStrategy; this.entries = entries; } @Override public Identified rebase(RawMatcher matcher) { return new Matched(nonNull(matcher), Transformer.NoOp.INSTANCE); } @Override public Identified rebase(ElementMatcher typeMatcher) { return rebase(typeMatcher, any()); } @Override public Identified rebase(ElementMatcher typeMatcher, ElementMatcher classLoaderMatcher) { return rebase(new RawMatcher.ForElementMatcherPair(nonNull(typeMatcher), nonNull(classLoaderMatcher))); } @Override public AgentBuilder withByteBuddy(ByteBuddy byteBuddy) { return new Default(nonNull(byteBuddy), binaryLocator, listener, nativeMethodPrefix, disableSelfInitialization, retransformation, bootstrapInjectionStrategy, entries); } @Override public AgentBuilder withListener(Listener listener) { return new Default(byteBuddy, binaryLocator, new Listener.Compound(this.listener, nonNull(listener)), nativeMethodPrefix, disableSelfInitialization, retransformation, bootstrapInjectionStrategy, entries); } @Override public AgentBuilder withBinaryLocator(BinaryLocator binaryLocator) { return new Default(byteBuddy, nonNull(binaryLocator), listener, nativeMethodPrefix, disableSelfInitialization, retransformation, bootstrapInjectionStrategy, entries); } @Override public AgentBuilder withNativeMethodPrefix(String prefix) { if (nonNull(prefix).length() == 0) { throw new IllegalArgumentException("The empty string is not a legal value for a native method prefix"); } return new Default(byteBuddy, binaryLocator, listener, prefix, disableSelfInitialization, retransformation, bootstrapInjectionStrategy, entries); } @Override public AgentBuilder allowRetransformation() { return new Default(byteBuddy, binaryLocator, listener, nativeMethodPrefix, disableSelfInitialization, true, bootstrapInjectionStrategy, entries); } @Override public AgentBuilder disableSelfInitialization() { return new Default(byteBuddy, binaryLocator, listener, nativeMethodPrefix, true, retransformation, bootstrapInjectionStrategy, entries); } @Override public AgentBuilder enableBootstrapInjection(File folder, Instrumentation instrumentation) { return new Default(byteBuddy, binaryLocator, listener, nativeMethodPrefix, true, retransformation, new BootstrapInjectionStrategy.Enabled(nonNull(folder), nonNull(instrumentation)), entries); } @Override public ClassFileTransformer makeRaw() { return new ExecutingTransformer(); } @Override public ClassFileTransformer installOn(Instrumentation instrumentation) { ClassFileTransformer classFileTransformer = makeRaw(); instrumentation.addTransformer(classFileTransformer, retransformation); if (!NO_NATIVE_PREFIX.equals(nonNull(nativeMethodPrefix))) { instrumentation.setNativeMethodPrefix(classFileTransformer, nativeMethodPrefix); } return classFileTransformer; } @Override public ClassFileTransformer installOnByteBuddyAgent() { try { return installOn((Instrumentation) ClassLoader.getSystemClassLoader() .loadClass(BYTE_BUDDY_AGENT_TYPE) .getDeclaredMethod(GET_INSTRUMENTATION_METHOD) .invoke(STATIC_METHOD)); } catch (Exception e) { throw new IllegalStateException("The Byte Buddy agent is not installed or not accessible", e); } } @Override public boolean equals(Object other) { if (this == other) return true; if (other == null || getClass() != other.getClass()) return false; Default aDefault = (Default) other; return binaryLocator.equals(aDefault.binaryLocator) && byteBuddy.equals(aDefault.byteBuddy) && listener.equals(aDefault.listener) && nativeMethodPrefix.equals(aDefault.nativeMethodPrefix) && disableSelfInitialization == aDefault.disableSelfInitialization && retransformation == aDefault.retransformation && bootstrapInjectionStrategy.equals(aDefault.bootstrapInjectionStrategy) && entries.equals(aDefault.entries); } @Override public int hashCode() { int result = byteBuddy.hashCode(); result = 31 * result + binaryLocator.hashCode(); result = 31 * result + listener.hashCode(); result = 31 * result + nativeMethodPrefix.hashCode(); result = 31 * result + (disableSelfInitialization ? 1 : 0); result = 31 * result + (retransformation ? 1 : 0); result = 31 * result + bootstrapInjectionStrategy.hashCode(); result = 31 * result + entries.hashCode(); return result; } @Override public String toString() { return "AgentBuilder.Default{" + "byteBuddy=" + byteBuddy + ", binaryLocator=" + binaryLocator + ", listener=" + listener + ", nativeMethodPrefix=" + nativeMethodPrefix + ", disableSelfInitialization=" + disableSelfInitialization + ", retransformation=" + retransformation + ", bootstrapInjectionStrategy=" + bootstrapInjectionStrategy + ", entries=" + entries + '}'; } /** * An initialization strategy which determines the handling of * {@link net.bytebuddy.implementation.LoadedTypeInitializer}s. */ public interface InitializationStrategy { /** * Determines if and how a loaded type initializer is to be applied to a loaded type. * * @param type The loaded type. * @param loadedTypeInitializer The {@code type}'s loaded type initializer. */ void initialize(Class type, LoadedTypeInitializer loadedTypeInitializer); /** * Transforms the instrumented type to implement an appropriate initialization strategy. * * @param builder The builder which should implement the initialization strategy. * @return The given {@code builder} with the initialization strategy applied. */ DynamicType.Builder apply(DynamicType.Builder builder); /** * Registers a loaded type initializer for a type name and class loader pair. * * @param name The name of the type for which the loaded type initializer is to be * registered. * @param classLoader The class loader of the instrumented type. Might be {@code null} if * this class loader represents the bootstrap class loader. * @param loadedTypeInitializer The loaded type initializer that is being registered. */ void register(String name, ClassLoader classLoader, LoadedTypeInitializer loadedTypeInitializer); /** * A non-initializing initialization strategy. */ enum NoOp implements InitializationStrategy { /** * The singleton instance. */ INSTANCE; @Override public void initialize(Class type, LoadedTypeInitializer loadedTypeInitializer) { /* do nothing */ } @Override public DynamicType.Builder apply(DynamicType.Builder builder) { return builder; } @Override public void register(String name, ClassLoader classLoader, LoadedTypeInitializer loadedTypeInitializer) { /* do nothing */ } @Override public String toString() { return "AgentBuilder.Default.InitializationStrategy.NoOp." + name(); } } /** * An initialization strategy that adds a code block to an instrumented type's type initializer which * then calls a specific class that is responsible for the explicit initialization. */ class SelfInjection implements InitializationStrategy, Implementation, ByteCodeAppender { /** * An accessor for the initialization nexus that makes sure that the Nexus is loaded by the * system class loader is accessed. */ private final Nexus.Accessor accessor; /** * Creates a new self injection strategy. */ public SelfInjection() { accessor = Nexus.Accessor.INSTANCE; } @Override public DynamicType.Builder apply(DynamicType.Builder builder) { return builder.invokable(none()).intercept(this); } @Override public void initialize(Class type, LoadedTypeInitializer loadedTypeInitializer) { loadedTypeInitializer.onLoad(type); } @Override public InstrumentedType prepare(InstrumentedType instrumentedType) { return instrumentedType.withInitializer(accessor.initializerFor(instrumentedType)); } @Override public ByteCodeAppender appender(Target implementationTarget) { return this; } @Override public Size apply(MethodVisitor methodVisitor, Context implementationContext, MethodDescription instrumentedMethod) { throw new IllegalStateException("Initialization strategy illegally applied to " + instrumentedMethod); } @Override public void register(String name, ClassLoader classLoader, LoadedTypeInitializer loadedTypeInitializer) { if (loadedTypeInitializer.isAlive()) { accessor.register(name, classLoader, loadedTypeInitializer); } } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && accessor == ((SelfInjection) other).accessor; } @Override public int hashCode() { return accessor.hashCode(); } @Override public String toString() { return "AgentBuilder.Default.InitializationStrategy.SelfInjection{" + "accessor=" + accessor + '}'; } /** *

* This nexus is a global dispatcher for initializing classes with * {@link net.bytebuddy.implementation.LoadedTypeInitializer}s. To do so, this class is to be loaded * by the system class loader in an explicit manner. Any instrumented class is then injected a code * block into its static type initializer that makes a call to this very same nexus which had the * loaded type initializer registered before hand. *

*

* Important: The nexus must never be accessed directly but only by its * {@link net.bytebuddy.agent.builder.AgentBuilder.Default.InitializationStrategy.SelfInjection.Nexus.Accessor} * which makes sure that the nexus is loaded by the system class loader. Otherwise, a class might not * be able to initialize itself if it is loaded by different class loader that does not have the * system class loader in its hierarchy. *

*/ public static class Nexus { /** * A map of keys identifying a loaded type by its name and class loader mapping their * potential {@link net.bytebuddy.implementation.LoadedTypeInitializer} where the class * loader of these initializers is however irrelevant. */ private static final ConcurrentMap TYPE_INITIALIZERS = new ConcurrentHashMap(); /** * The name of a type for which a loaded type initializer is registered. */ private final String name; /** * The class loader for which a loaded type initializer is registered. */ private final ClassLoader classLoader; /** * Creates a key for identifying a loaded type initializer. * * @param type The loaded type for which a key is to be created. */ private Nexus(Class type) { name = type.getName(); classLoader = type.getClassLoader(); } /** * Creates a key for identifying a loaded type initializer. * * @param name The name of a type for which a loaded type initializer is registered. * @param classLoader The class loader for which a loaded type initializer is registered. */ private Nexus(String name, ClassLoader classLoader) { this.name = name; this.classLoader = classLoader; } /** * Initializes a loaded type. * * @param type The loaded type to initialize. * @throws Exception If an exception occurs. */ @SuppressWarnings("unused") public static void initialize(Class type) throws Exception { Object typeInitializer = TYPE_INITIALIZERS.remove(new Nexus(type)); if (typeInitializer != null) { typeInitializer.getClass().getMethod("onLoad", Class.class).invoke(typeInitializer, type); } } /** * @param name The name of the type for the loaded type initializer. * @param classLoader The class loader of the type for the loaded type initializer. * @param typeInitializer The type initializer to register. The initializer must be an instance * of {@link net.bytebuddy.implementation.LoadedTypeInitializer} where * it does however not matter which class loader loaded this latter type. */ @SuppressWarnings("unused") public static void register(String name, ClassLoader classLoader, Object typeInitializer) { TYPE_INITIALIZERS.put(new Nexus(name, classLoader), typeInitializer); } @Override public boolean equals(Object other) { if (this == other) return true; if (other == null || getClass() != other.getClass()) return false; Nexus nexus = (Nexus) other; return !(classLoader != null ? !classLoader.equals(nexus.classLoader) : nexus.classLoader != null) && name.equals(nexus.name); } @Override public int hashCode() { int result = name.hashCode(); result = 31 * result + (classLoader != null ? classLoader.hashCode() : 0); return result; } @Override public String toString() { return "AgentBuilder.Default.InitializationStrategy.SelfInjection.Nexus{" + "name='" + name + '\'' + ", classLoader=" + classLoader + '}'; } /** * An accessor for making sure that the accessed * {@link net.bytebuddy.agent.builder.AgentBuilder.Default.InitializationStrategy.SelfInjection.Nexus} * is loaded by the system class loader. */ protected enum Accessor { /** * The singleton instance. */ INSTANCE; /** * Indicates that a static method is invoked by reflection. */ private static final Object STATIC_METHOD = null; /** * The method for registering a type initializer in the system class loader's * {@link net.bytebuddy.agent.builder.AgentBuilder.Default.InitializationStrategy.SelfInjection.Nexus}. */ private final Method registration; /** * The {@link ClassLoader#getSystemClassLoader()} method. */ private final MethodDescription systemClassLoader; /** * The {@link java.lang.ClassLoader#loadClass(String)} method. */ private final MethodDescription loadClass; /** * The {@link java.lang.Class#getDeclaredMethod(String, Class[])} method. */ private final MethodDescription getDeclaredMethod; /** * The {@link java.lang.reflect.Method#invoke(Object, Object...)} method. */ private final MethodDescription invokeMethod; /** * Creates the singleton accessor. */ Accessor() { try { ClassLoader classLoader = ClassLoader.getSystemClassLoader(); TypeDescription nexusType = new TypeDescription.ForLoadedType(Nexus.class); Class nexus = new ClassInjector.UsingReflection(classLoader) .inject(Collections.singletonMap(nexusType, new StreamDrainer().drain(classLoader.getResourceAsStream(Nexus.class.getName().replace('.', '/') + ".class")))) .get(nexusType); registration = nexus.getDeclaredMethod("register", String.class, ClassLoader.class, Object.class); systemClassLoader = new TypeDescription.ForLoadedType(ClassLoader.class).getDeclaredMethods() .filter(named("getSystemClassLoader")).getOnly(); loadClass = new TypeDescription.ForLoadedType(ClassLoader.class).getDeclaredMethods() .filter(named("loadClass").and(takesArguments(String.class))).getOnly(); getDeclaredMethod = new TypeDescription.ForLoadedType(Class.class).getDeclaredMethods() .filter(named("getDeclaredMethod").and(takesArguments(String.class, Class[].class))).getOnly(); invokeMethod = new TypeDescription.ForLoadedType(Method.class).getDeclaredMethods() .filter(named("invoke").and(takesArguments(Object.class, Object[].class))).getOnly(); } catch (Exception e) { throw new IllegalStateException("Cannot create type initialization accessor", e); } } /** * Registers a type initializer with the class loader's nexus. * * @param name The name of a type for which a loaded type initializer is registered. * @param classLoader The class loader for which a loaded type initializer is registered. * @param typeInitializer The loaded type initializer to be registered. */ public void register(String name, ClassLoader classLoader, Object typeInitializer) { try { registration.invoke(STATIC_METHOD, name, classLoader, typeInitializer); } catch (IllegalAccessException e) { throw new IllegalStateException("Cannot register type initializer for " + name, e); } catch (InvocationTargetException e) { throw new IllegalStateException("Cannot register type initializer for " + name, e.getCause()); } } /** * Creates a stack manipulation for a given instrumented type that injects a code block for * calling the system class loader's nexus in order to apply a self-initialization. * * @param instrumentedType The instrumented type for which the code block is to be injected. * @return A byte code appender that implements the self-initialization. */ public ByteCodeAppender initializerFor(TypeDescription instrumentedType) { return new ByteCodeAppender.Simple(new StackManipulation.Compound( MethodInvocation.invoke(systemClassLoader), new TextConstant(Nexus.class.getName()), MethodInvocation.invoke(loadClass), new TextConstant("initialize"), ArrayFactory.forType(TypeDescription.CLASS) .withValues(Collections.singletonList(ClassConstant.of(TypeDescription.CLASS))), MethodInvocation.invoke(getDeclaredMethod), NullConstant.INSTANCE, ArrayFactory.forType(TypeDescription.OBJECT) .withValues(Collections.singletonList(ClassConstant.of(instrumentedType))), MethodInvocation.invoke(invokeMethod), Removal.SINGLE )); } @Override public String toString() { return "AgentBuilder.Default.InitializationStrategy.SelfInjection.Nexus.Accessor." + name(); } } } } } /** * An injection strategy for injecting classes into the bootstrap class loader. */ protected interface BootstrapInjectionStrategy { /** * Creates an injector for the bootstrap class loader. * * @param protectionDomain The protection domain to be used. * @return A class injector for the bootstrap class loader. */ ClassInjector make(ProtectionDomain protectionDomain); /** * A disabled bootstrap injection strategy. */ enum Disabled implements BootstrapInjectionStrategy { /** * The singleton instance. */ INSTANCE; @Override public ClassInjector make(ProtectionDomain protectionDomain) { throw new IllegalStateException("Injecting classes into the bootstrap class loader was not enabled"); } @Override public String toString() { return "AgentBuilder.Default.BootstrapInjectionStrategy.Disabled." + name(); } } /** * An enabled bootstrap injection strategy. */ class Enabled implements BootstrapInjectionStrategy { /** * The folder in which jar files are to be saved. */ private final File folder; /** * The instrumentation to use for appending jar files. */ private final Instrumentation instrumentation; /** * Creates a new enabled bootstrap class loader injection strategy. * * @param folder The folder in which jar files are to be saved. * @param instrumentation The instrumentation to use for appending jar files. */ public Enabled(File folder, Instrumentation instrumentation) { this.folder = folder; this.instrumentation = instrumentation; } @Override public ClassInjector make(ProtectionDomain protectionDomain) { return new ClassInjector.UsingInstrumentation(folder, ClassInjector.UsingInstrumentation.Target.BOOTSTRAP, instrumentation); } @Override public boolean equals(Object other) { if (this == other) return true; if (other == null || getClass() != other.getClass()) return false; Enabled enabled = (Enabled) other; return folder.equals(enabled.folder) && instrumentation.equals(enabled.instrumentation); } @Override public int hashCode() { int result = folder.hashCode(); result = 31 * result + instrumentation.hashCode(); return result; } @Override public String toString() { return "AgentBuilder.Default.BootstrapInjectionStrategy.Enabled{" + "folder=" + folder + ", instrumentation=" + instrumentation + '}'; } } } /** * A registered transformation as a combination of a * {@link net.bytebuddy.agent.builder.AgentBuilder.RawMatcher} and a * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}. */ protected static class Transformation implements RawMatcher, Transformer { /** * The raw matcher that is represented by this transformation. */ private final RawMatcher rawMatcher; /** * The transformer that is represented by this transformation. */ private final Transformer transformer; /** * Creates a new transformation. * * @param rawMatcher The raw matcher that is represented by this transformation. * @param transformer The transformer that is represented by this transformation. */ protected Transformation(RawMatcher rawMatcher, Transformer transformer) { this.rawMatcher = rawMatcher; this.transformer = transformer; } @Override public boolean matches(TypeDescription typeDescription, ClassLoader classLoader, Class classBeingRedefined, ProtectionDomain protectionDomain) { return rawMatcher.matches(typeDescription, classLoader, classBeingRedefined, protectionDomain); } @Override public DynamicType.Builder transform(DynamicType.Builder builder, TypeDescription typeDescription) { return transformer.transform(builder, typeDescription); } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && rawMatcher.equals(((Transformation) other).rawMatcher) && transformer.equals(((Transformation) other).transformer); } @Override public int hashCode() { int result = rawMatcher.hashCode(); result = 31 * result + transformer.hashCode(); return result; } @Override public String toString() { return "AgentBuilder.Default.Transformation{" + "rawMatcher=" + rawMatcher + ", transformer=" + transformer + '}'; } } /** * A {@link java.lang.instrument.ClassFileTransformer} that implements the enclosing agent builder's * configuration. */ protected class ExecutingTransformer implements ClassFileTransformer { /** * The method name transformer to be used for rebasing methods. */ private final MethodRebaseResolver.MethodNameTransformer methodNameTransformer; /** * The initialization strategy to use. */ private final InitializationStrategy initializationStrategy; /** * Creates a new executing transformer that reflects the enclosing agent builder's configuration. */ public ExecutingTransformer() { methodNameTransformer = NO_NATIVE_PREFIX.equals(nativeMethodPrefix) ? new MethodRebaseResolver.MethodNameTransformer.Suffixing() : new MethodRebaseResolver.MethodNameTransformer.Prefixing(nativeMethodPrefix); initializationStrategy = disableSelfInitialization ? InitializationStrategy.NoOp.INSTANCE : new InitializationStrategy.SelfInjection(); } @Override public byte[] transform(ClassLoader classLoader, String internalTypeName, Class classBeingRedefined, ProtectionDomain protectionDomain, byte[] binaryRepresentation) { String binaryTypeName = internalTypeName.replace('/', '.'); try { BinaryLocator.Initialized initialized = binaryLocator.initialize(binaryTypeName, binaryRepresentation, classLoader); TypeDescription typeDescription = initialized.getTypePool().describe(binaryTypeName).resolve(); for (Transformation transformation : entries) { if (transformation.matches(typeDescription, classLoader, classBeingRedefined, protectionDomain)) { DynamicType.Unloaded dynamicType = initializationStrategy.apply( transformation.transform(byteBuddy.rebase(typeDescription, initialized.getClassFileLocator(), methodNameTransformer), typeDescription)).make(); Map loadedTypeInitializers = dynamicType.getLoadedTypeInitializers(); if (loadedTypeInitializers.size() > 1) { ClassInjector classInjector = classLoader == null ? bootstrapInjectionStrategy.make(protectionDomain) : new ClassInjector.UsingReflection(classLoader, protectionDomain); for (Map.Entry> auxiliary : classInjector.inject(dynamicType.getRawAuxiliaryTypes()).entrySet()) { initializationStrategy.initialize(auxiliary.getValue(), loadedTypeInitializers.get(auxiliary.getKey())); } } initializationStrategy.register(binaryTypeName, classLoader, loadedTypeInitializers.get(dynamicType.getTypeDescription())); listener.onTransformation(typeDescription, dynamicType); return dynamicType.getBytes(); } } listener.onIgnored(binaryTypeName); return NO_TRANSFORMATION; } catch (Throwable throwable) { listener.onError(binaryTypeName, throwable); return NO_TRANSFORMATION; } finally { listener.onComplete(binaryTypeName); } } @Override public String toString() { return "AgentBuilder.Default.ExecutingTransformer{" + "agentBuilder=" + Default.this + ", methodNameTransformer=" + methodNameTransformer + ", initializationStrategy=" + initializationStrategy + '}'; } } /** * A helper class that describes a {@link net.bytebuddy.agent.builder.AgentBuilder.Default} after supplying * a {@link net.bytebuddy.agent.builder.AgentBuilder.RawMatcher} such that one or several * {@link net.bytebuddy.agent.builder.AgentBuilder.Transformer}s can be supplied. */ protected class Matched implements Identified.Extendable { /** * The supplied raw matcher. */ private final RawMatcher rawMatcher; /** * The supplied transformer. */ private final Transformer transformer; /** * Creates a new matched default agent builder. * * @param rawMatcher The supplied raw matcher. * @param transformer The supplied transformer. */ protected Matched(RawMatcher rawMatcher, Transformer transformer) { this.rawMatcher = rawMatcher; this.transformer = transformer; } @Override public Identified.Extendable transform(Transformer transformer) { return new Matched(rawMatcher, new Transformer.Compound(this.transformer, nonNull(transformer))); } @Override public Identified rebase(RawMatcher matcher) { return materialize().rebase(matcher); } @Override public Identified rebase(ElementMatcher typeMatcher) { return materialize().rebase(typeMatcher); } @Override public Identified rebase(ElementMatcher typeMatcher, ElementMatcher classLoaderMatcher) { return materialize().rebase(typeMatcher, classLoaderMatcher); } @Override public AgentBuilder withByteBuddy(ByteBuddy byteBuddy) { return materialize().withByteBuddy(byteBuddy); } @Override public AgentBuilder withListener(Listener listener) { return materialize().withListener(listener); } @Override public AgentBuilder withBinaryLocator(BinaryLocator binaryLocator) { return materialize().withBinaryLocator(binaryLocator); } @Override public AgentBuilder withNativeMethodPrefix(String prefix) { return materialize().withNativeMethodPrefix(prefix); } @Override public AgentBuilder disableSelfInitialization() { return materialize().disableSelfInitialization(); } @Override public AgentBuilder allowRetransformation() { return materialize().allowRetransformation(); } @Override public AgentBuilder enableBootstrapInjection(File folder, Instrumentation instrumentation) { return materialize().enableBootstrapInjection(folder, instrumentation); } @Override public ClassFileTransformer makeRaw() { return materialize().makeRaw(); } @Override public ClassFileTransformer installOn(Instrumentation instrumentation) { return materialize().installOn(instrumentation); } @Override public ClassFileTransformer installOnByteBuddyAgent() { return materialize().installOnByteBuddyAgent(); } /** * Materializes the currently described {@link net.bytebuddy.agent.builder.AgentBuilder.Default.Transformation}. * * @return An agent builder that represents the currently described entry of this instance. */ protected AgentBuilder materialize() { return new Default(byteBuddy, binaryLocator, listener, nativeMethodPrefix, disableSelfInitialization, retransformation, bootstrapInjectionStrategy, join(new Transformation(rawMatcher, transformer), entries)); } /** * Returns the outer instance. * * @return The outer instance. */ private Default getOuter() { return Default.this; } @Override public boolean equals(Object other) { return this == other || !(other == null || getClass() != other.getClass()) && rawMatcher.equals(((Matched) other).rawMatcher) && transformer.equals(((Matched) other).transformer) && Default.this.equals(((Matched) other).getOuter()); } @Override public int hashCode() { int result = rawMatcher.hashCode(); result = 31 * result + transformer.hashCode(); result = 31 * result + Default.this.hashCode(); return result; } @Override public String toString() { return "AgentBuilder.Default.Matched{" + "rawMatcher=" + rawMatcher + ", transformer=" + transformer + ", agentBuilder=" + Default.this + '}'; } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy