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

net.openhft.chronicle.wire.GenerateMethodReader Maven / Gradle / Ivy

There is a newer version: 2.27ea1
Show newest version
/*
 * Copyright 2016-2020 chronicle.software
 *
 *       https://chronicle.software
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *       http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package net.openhft.chronicle.wire;

import net.openhft.chronicle.bytes.*;
import net.openhft.chronicle.core.Jvm;
import net.openhft.chronicle.core.Maths;
import net.openhft.chronicle.core.io.Closeable;
import net.openhft.chronicle.core.util.GenericReflection;
import net.openhft.chronicle.core.util.IgnoresEverything;
import net.openhft.chronicle.wire.utils.JavaSourceCodeFormatter;
import net.openhft.chronicle.wire.utils.SourceCodeFormatter;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy;
import java.lang.reflect.Type;
import java.util.*;
import java.util.function.BiConsumer;
import java.util.stream.Collectors;

import static java.lang.String.format;
import static net.openhft.chronicle.core.util.GenericReflection.*;
import static net.openhft.chronicle.wire.GenerateMethodWriter.isSynthetic;

/**
 * Responsible for code generation and its runtime compilation of custom {@link MethodReader}s.
 * The class dynamically generates Java source code based on the provided configurations and compiles them at runtime.
 * It offers the flexibility to create custom MethodReaders tailored to specific needs without manual coding.
 */
public class GenerateMethodReader {

     // Configuration flag for dumping the generated code.
    private static final boolean DUMP_CODE = Jvm.getBoolean("dumpCode");
    // Set of interfaces that are not meant to be processed.
    private static final Set> IGNORED_INTERFACES = new LinkedHashSet<>();

    static {
        // Initialization: ensuring Wires classes are loaded and classpath is set up
        Wires.init();
        // Populate IGNORED_INTERFACES with predefined interfaces
        Collections.addAll(IGNORED_INTERFACES,
                BytesMarshallable.class,
                DocumentContext.class,
                ReadDocumentContext.class,
                WriteDocumentContext.class,
                ExcerptListener.class,
                FieldInfo.class,
                FieldNumberParselet.class,
                SelfDescribingMarshallable.class,
                BytesMarshallable.class,
                Marshallable.class,
                MarshallableIn.class,
                MarshallableOut.class,
                MethodWriter.class,
                SourceContext.class
        );
    }

    // Configuration for the type of wire to use for serialization/deserialization.
    private final WireType wireType;

     // Handlers for metadata during the method reader generation.
    private final Object[] metaDataHandler;

    // Instances of the classes/interfaces for which method readers are to be generated.
    private final Object[] instances;

    // Interceptor for handling method returns during the method reader generation.
    private final MethodReaderInterceptorReturns interceptor;
    // A mapping to ensure unique method names for the generated code.
    private final Map handledMethodNames = new HashMap<>();
    // A set to store method signatures that have been processed.
    private final Set handledMethodSignatures = new HashSet<>();
    // A set to store interfaces that have been processed.
    private final Set> handledInterfaces = new HashSet<>();
    // Buffers for holding generated Java source code.
    private final SourceCodeFormatter sourceCode = new JavaSourceCodeFormatter();
    private final SourceCodeFormatter fields = new JavaSourceCodeFormatter();
    private final SourceCodeFormatter eventNameSwitchBlock = new JavaSourceCodeFormatter();
    private final SourceCodeFormatter eventNameSwitchBlockMeta = new JavaSourceCodeFormatter();
    private final SourceCodeFormatter eventIdSwitchBlock = new JavaSourceCodeFormatter();
    private final SourceCodeFormatter eventIdSwitchBlockMeta = new JavaSourceCodeFormatter();
    private final SourceCodeFormatter numericConverters = new JavaSourceCodeFormatter();
    // Name of the class that will be generated.
    private final String generatedClassName;
    // Set of field names to ensure uniqueness in the generated code.
    private final Set fieldNames = new LinkedHashSet<>();

    // Flag indicating the presence of a method filter in the generated code.
    private boolean methodFilterPresent;

    // Flag indicating whether the source code has been generated.
    private boolean isSourceCodeGenerated;

    // Flag indicating if there are chained method calls in the generated code.
    private boolean hasChainedCalls;

    /**
     * Constructs a new instance of GenerateMethodReader.
     * Initializes the required configurations, metadata handlers, and instances which are essential for code generation.
     *
     * @param wireType Configuration for serialization/deserialization
     * @param interceptor An instance of MethodReaderInterceptorReturns
     * @param metaDataHandler Array of meta-data handlers
     * @param instances Instances that dictate the structure of the generated MethodReader
     */
    public GenerateMethodReader(WireType wireType, MethodReaderInterceptorReturns interceptor, Object[] metaDataHandler, Object... instances) {
        this.wireType = wireType;
        this.interceptor = interceptor;
        this.metaDataHandler = metaDataHandler;
        this.instances = instances;
        this.generatedClassName = generatedClassName0();
    }

    /**
     * Computes the signature of a given method.
     * The signature comprises the return type, method name, and parameter types.
     *
     * @param m The method for which the signature is to be computed
     * @param type The type under consideration
     * @return A string representing the method's signature
     */
    private static String signature(Method m, Class type) {
        return GenericReflection.getReturnType(m, type) + " " + m.getName() + " " + Arrays.toString(GenericReflection.getParameterTypes(m, type));
    }

    /**
     * Checks if the given class has an "INSTANCE" field.
     * Useful to verify if a class adheres to certain patterns or conventions.
     *
     * @param aClass The class to be checked
     * @return true if the class has an "INSTANCE" field, false otherwise
     */
    static boolean hasInstance(Class aClass) {
        try {
            aClass.getField("INSTANCE");
            return true;
        } catch (NoSuchFieldException e) {
            return false;
        }
    }

    /**
     * Generates and compiles the source code of a custom {@link MethodReader} at runtime.
     * It uses the configurations and instances provided during initialization.
     * If there are issues during compilation, it provides detailed error messages for easier debugging.
     *
     * @return A new class representing the custom MethodReader
     */
    public Class createClass() {
        // If source code isn't already generated, generate it.
        if (!isSourceCodeGenerated)
            generateSourceCode();

        final ClassLoader classLoader = instances[0].getClass().getClassLoader();
        final String fullClassName = packageName() + "." + generatedClassName();

        try {
            return Wires.loadFromJava(classLoader, fullClassName, sourceCode.toString());
        } catch (AssertionError e) {
            if (e.getCause() instanceof LinkageError) {
                try {
                    return Class.forName(fullClassName, true, classLoader);
                } catch (ClassNotFoundException x) {
                    throw Jvm.rethrow(x);
                }
            }
            throw Jvm.rethrow(e);
        } catch (Throwable e) {
            throw Jvm.rethrow(new ClassNotFoundException(e.getMessage() + '\n' + sourceCode, e));
        }
    }

    /**
     * Responsible for generating the source code of a {@link MethodReader} based on specified {@link #instances}.
     * This method encapsulates the logic for building the source code dynamically, by inspecting provided interfaces,
     * filtering ignored interfaces, and appending necessary components to the code.
     */
    private void generateSourceCode() {
        // Clear previously handled interfaces and method signatures for clean generation.
        handledInterfaces.clear();
        handledMethodNames.clear();
        handledMethodSignatures.clear();

        // Handle meta data handlers and their associated interfaces.
        for (int i = 0; metaDataHandler != null && i < metaDataHandler.length; i++) {
            final Class aClass = metaDataHandler[i].getClass();

            // Process each interface of the meta data handler.
            for (Class anInterface : ReflectionUtil.interfaces(aClass)) {
                if (Jvm.dontChain(anInterface))
                    continue;
                handleInterface(anInterface, "metaInstance" + i, false, eventNameSwitchBlockMeta, eventIdSwitchBlockMeta);
            }
        }

        // Clear previously handled interfaces and method signatures again to process instance interfaces.
        handledInterfaces.clear();
        handledMethodNames.clear();
        handledMethodSignatures.clear();

        // Handle instances and their associated interfaces.
        for (int i = 0; i < instances.length; i++) {
            final Class aClass = instances[i].getClass();

            // Check if the instance has method filters.
            boolean methodFilter = instances[i] instanceof MethodFilterOnFirstArg;
            methodFilterPresent |= methodFilter;

            // Process each interface of the instance.
            for (Class anInterface : ReflectionUtil.interfaces(aClass)) {
                if (IGNORED_INTERFACES.contains(anInterface))
                    continue;
                handleInterface(anInterface, "instance" + i, methodFilter, eventNameSwitchBlock, eventIdSwitchBlock);
            }
        }

        // Start constructing the source code.
        if (!packageName().isEmpty())
            sourceCode.append(format("package %s;\n", packageName()));

        // Import statements required for the generated code.
        sourceCode.append("" +
                "import net.openhft.chronicle.core.Jvm;\n" +
                "import net.openhft.chronicle.core.util.InvocationTargetRuntimeException;\n" +
                "import net.openhft.chronicle.core.util.ObjectUtils;\n" +
                "import net.openhft.chronicle.bytes.*;\n" +
                "import net.openhft.chronicle.wire.*;\n" +
                "import net.openhft.chronicle.wire.utils.*;\n" +
                "import net.openhft.chronicle.wire.BinaryWireCode;\n" +
                "\n" +
                "import java.util.Map;\n" +
                "import java.lang.reflect.Method;\n" +
                "\n");

        // Declare the generated class extending AbstractGeneratedMethodReader.
        sourceCode.append(format("public class %s extends AbstractGeneratedMethodReader {\n", generatedClassName()));

        // Inline comment for instances section.
        sourceCode.append("// instances on which parsed calls are invoked\n");

        for (int i = 0; metaDataHandler != null && i < metaDataHandler.length; i++) {
            sourceCode.append(format("private final Object metaInstance%d;\n", i));
        }
        for (int i = 0; i < instances.length; i++) {
            sourceCode.append(format("private final Object instance%d;\n", i));
        }
        sourceCode.append("private final WireParselet defaultParselet;\n");
        sourceCode.append("\n");

        // Check and declare an interceptor if one is present.
        if (hasRealInterceptorReturns()) {
            sourceCode.append("// method reader interceptor\n");
            sourceCode.append("private final MethodReaderInterceptorReturns interceptor;\n");
            sourceCode.append("\n");
        }

        // Appending fields to the source code.
        sourceCode.append(fields);

        // If a method filter is present, add an ignored flag.
        if (methodFilterPresent) {
            sourceCode.append("// flag for handling ignoreMethodBasedOnFirstArg\n");
            sourceCode.append("private boolean ignored;\n\n");
        }

        // Add numeric converters if any are present.
        if (numericConverters.length() > 0) {
            sourceCode.append("// numeric converters\n");
            sourceCode.append(numericConverters);
            sourceCode.append("\n");
        }

        // Add result for chained calls if any are present.
        if (hasChainedCalls) {
            sourceCode.append("// chained call result\n");
            sourceCode.append("private Object chainedCallReturnResult;");
            sourceCode.append("\n");
        }

        // Define the constructor of the generated class.
        sourceCode.append(format("public %s(MarshallableIn in, " +
                "WireParselet defaultParselet, " +
                "WireParselet debugLoggingParselet, " +
                "MethodReaderInterceptorReturns interceptor, " +
                "Object[] metaInstances, " +
                "Object[] instances) {\n" +
                "super(in, debugLoggingParselet);\n" +
                "this.defaultParselet = defaultParselet;\n", generatedClassName()));

        // Set interceptor if one is present.
        if (hasRealInterceptorReturns())
            sourceCode.append("this.interceptor = interceptor;\n");

        // Initialize metaInstance objects.
        for (int i = 0; metaDataHandler != null && i < metaDataHandler.length; i++)
            sourceCode.append(format("metaInstance%d = metaInstances[%d];\n", i, i));

        // Initialize instance objects.
        for (int i = 0; i < instances.length - 1; i++)
            sourceCode.append(format("instance%d = instances[%d];\n", i, i));

        sourceCode.append(format("instance%d = instances[%d];\n}\n\n", instances.length - 1, instances.length - 1));

        // Override the restIgnored method if chained calls are present.
        if (hasChainedCalls) {
            sourceCode.append("@Override\n" +
                            "public boolean restIgnored() {\n" +
                            "  return chainedCallReturnResult instanceof ")
                    .append(IgnoresEverything.class.getName())
                    .append(";\n" +
                            "}\n");
        }

        // Define the readOneGenerated method for the MethodReader.
        sourceCode.append("@Override\n" +
                "protected MethodReaderStatus readOneGenerated(WireIn wireIn) {\n" +
                "ValueIn valueIn = wireIn.getValueIn();\n" +
                "String lastEventName = \"\";\n" +
                "if (wireIn.bytes().peekUnsignedByte() == BinaryWireCode.FIELD_NUMBER) {\n" +
                "int methodId = (int) wireIn.readEventNumber();\n" +
                "switch (methodId) {\n");

        // Append method ID switch logic.
        addMethodIdSwitch(MethodReader.HISTORY, MethodReader.MESSAGE_HISTORY_METHOD_ID, eventIdSwitchBlock);
        sourceCode.append(eventIdSwitchBlock);

        // Handle default case and read the event name.
        sourceCode.append("default:\n" +
                // Comment explaining the garbage-free nature for low methodIds.
                "lastEventName = Integer.toString(methodId);\n" +
                "break;\n" +
                "}\n" +
                "}\n" +
                "else {\n" +
                "lastEventName = wireIn.readEvent(String.class);\n" +
                "}\n" +
                // Try-catch block for reading method names.
                "try {\n" +
                "if (Jvm.isDebug())\n" +
                "debugLoggingParselet.accept(lastEventName, valueIn);\n" +
                "if (lastEventName == null)\n" +
                "throw new IllegalStateException(\"Failed to read method name or ID\");\n" +
                "switch (lastEventName) {\n");
        // History case logic.
        if (!eventNameSwitchBlock.contains("case \"history\":"))
            sourceCode.append("case MethodReader.HISTORY:\n" +
                    "valueIn.marshallable(messageHistory);\n" +
                    "return MethodReaderStatus.HISTORY;\n\n");

        // Append to the source code based on event name.
        sourceCode.append(eventNameSwitchBlock);

        // Handle default case if the event name isn't recognized.
        sourceCode.append("default:\n" +
                "defaultParselet.accept(lastEventName, valueIn);\n" +
                "return MethodReaderStatus.UNKNOWN;\n" +
                "}\n");

        // Check if we've handled the event successfully.
        if (eventNameSwitchBlock.contains("break;"))
            sourceCode.append("return MethodReaderStatus.KNOWN;\n");
        sourceCode.append("} \n" +
                "catch (InvocationTargetRuntimeException e) {\n" +
                "throw e;\n" +
                "}\n" +
                "}\n");

        // Handle meta-generated events using a similar logic.
        sourceCode.append("@Override\n" +
                "protected MethodReaderStatus readOneMetaGenerated(WireIn wireIn) {\n" +
                "ValueIn valueIn = wireIn.getValueIn();\n" +
                "String lastEventName = \"\";\n" +
                "if (wireIn.bytes().peekUnsignedByte() == BinaryWireCode.FIELD_NUMBER) {\n" +
                "int methodId = (int) wireIn.readEventNumber();\n" +
                "switch (methodId) {\n");

        sourceCode.append(eventIdSwitchBlockMeta);

        sourceCode.append("default:\n" +
                "valueIn.skipValue();\n" +
                "return MethodReaderStatus.UNKNOWN;\n" +
                "}\n" +
                "}\n" +
                "else {\n" +
                "lastEventName = wireIn.readEvent(String.class);\n" +
                "}\n" +
                "try {\n" +
                "if (Jvm.isDebug())\n" +
                "debugLoggingParselet.accept(lastEventName, valueIn);\n" +
                "if (lastEventName == null)\n" +
                "throw new IllegalStateException(\"Failed to read method name or ID\");\n" +
                "switch (lastEventName) {\n" +
                "case MethodReader.HISTORY:\n" +
                "valueIn.marshallable(messageHistory);\n" +
                "return MethodReaderStatus.HISTORY;\n\n");

        // Append to the source code based on meta event name.
        sourceCode.append(eventNameSwitchBlockMeta);

        // Handle default case if the meta event name isn't recognized.
        sourceCode.append("default:\n" +
                "defaultParselet.accept(lastEventName, valueIn);\n" +
                "return MethodReaderStatus.UNKNOWN;\n" +
                "}\n");
        // Check if we've handled the meta event successfully.
        if (eventNameSwitchBlockMeta.contains("break;"))
            sourceCode.append("return MethodReaderStatus.KNOWN;\n");
        sourceCode.append("} \n" +
                "catch (InvocationTargetRuntimeException e) {\n" +
                "throw e;\n" +
                "}\n" +
                "}\n}\n");

        // Set flag indicating source code has been generated.
        isSourceCodeGenerated = true;

        // Dump the generated source code to console if the flag is set.
        if (DUMP_CODE)
            System.out.println(sourceCode);
    }

    /**
     * This method is used to generate code for handling all method calls of a given interface.
     * It processes the methods recursively in case of chained methods.
     *
     * 

It first checks if the given interface should be chained using {@code Jvm.dontChain()} method. * If not, it immediately returns without executing further. It also checks whether the interface * has already been processed. If yes, it immediately returns. * *

Then it proceeds to process all non-static, non-synthetic methods declared in the given * interface but not in {@code java.lang.Object}. * If a method has already been processed, it's skipped. * *

It also validates that the method isn't one of those defined in {@code java.lang.Object}, * if it is, it's skipped. * *

If a method name has already been processed before, it throws an {@code IllegalStateException}. * This is because MethodReader does not support overloaded methods. * *

Finally, it calls {@code handleMethod()} on the current method if it passed all the above checks. * * @param anInterface The interface being processed. * @param instanceFieldName In the generated code, methods are executed on a field with this name. * @param methodFilter Indicates if the passed interface is marked with {@link MethodFilterOnFirstArg}. If true, only certain methods are processed. * @ blocks based on method event IDs. * @param eventNameSwitchBlock The block of code that handles the switching of event names. * @param eventIdSwitchBlock The block of code that handles the switching of event IDs. */ private void handleInterface(Class anInterface, String instanceFieldName, boolean methodFilter, SourceCodeFormatter eventNameSwitchBlock, SourceCodeFormatter eventIdSwitchBlock) { if (Jvm.dontChain(anInterface)) return; if (!handledInterfaces.add(anInterface)) return; for (@NotNull Method m : anInterface.getMethods()) { Class declaringClass = m.getDeclaringClass(); if (declaringClass == Object.class) continue; final int modifiers = m.getModifiers(); if (Modifier.isStatic(modifiers) || isSynthetic(modifiers)) continue; final String signature = signature(m, anInterface); if (!handledMethodSignatures.add(signature)) continue; final String methodName = m.getName(); try { // skip Object defined methods. Object.class.getMethod(methodName, m.getParameterTypes()); continue; } catch (NoSuchMethodException e) { // not an Object method. } if (handledMethodNames.containsKey(methodName)) { throw new IllegalStateException("MethodReader does not support overloaded methods. " + "Method: " + handledMethodNames.get(methodName) + ", and: " + signature); } handledMethodNames.put(methodName, signature); handleMethod(m, anInterface, instanceFieldName, methodFilter, eventNameSwitchBlock, eventIdSwitchBlock); } } /** * This method generates code for handling the call of a specific method. It sets up necessary fields and structures, * prepares parameters, and constructs a switch block for method calls. * *

Initially, it ensures that the method is accessible and obtains its parameter types and return type. * It processes the method parameters and creates fields for storing them. It also checks if the return type of the method * is chainable and updates the state accordingly. * *

If a real interceptor is returned by the method, it creates an array field to store the interceptor's arguments * and also adds a static field to hold a reference to the method itself. * *

Furthermore, if the method is annotated with {@code MethodId}, it extracts the method ID from the annotation * and adds a switch case for this ID to the {@code eventIdSwitchBlock}. * *

Then, it builds a case for the method in the {@code eventNameSwitchBlock}. The structure of this case varies * depending on the number of parameters the method has and if it's marked with {@code MethodFilterOnFirstArg}. * *

If the method's return type is {@code DocumentContext}, it also adds code to copy the method's result to the wire * and close it. * *

Finally, if the method's return type is chainable, it calls {@code handleInterface()} on it. * * @param m The method for which code is generated. * @param anInterface The interface containing the method. * @param instanceFieldName In the generated code, this method is executed on a field with this name. * @param methodFilter Indicates if the passed interface is marked with {@link MethodFilterOnFirstArg}. If true, only certain methods are processed. * @param eventIdSwitchBlock The block of code that handles the switching of event IDs. * @param eventNameSwitchBlock The block of code that handles the switching of event names. */ private void handleMethod(Method m, Class anInterface, String instanceFieldName, boolean methodFilter, SourceCodeFormatter eventNameSwitchBlock, SourceCodeFormatter eventIdSwitchBlock) { Jvm.setAccessible(m); // Retrieving parameter and return type information of the method Type[] parameterTypes = getParameterTypes(m, anInterface); Class chainReturnType = erase(getReturnType(m, anInterface)); // Initial setup for chainable return types if (chainReturnType != DocumentContext.class && (!chainReturnType.isInterface() || Jvm.dontChain(chainReturnType))) chainReturnType = null; // Field setup based on method parameters and interceptor if (parameterTypes.length > 0 || hasRealInterceptorReturns()) fields.append(format("// %s\n", m.getName())); // Iterating through method parameters to setup fields for (int i = 0; i < parameterTypes.length; i++) { Class parameterType = erase(parameterTypes[i]); final String typeName = parameterType.getCanonicalName(); String fieldName = m.getName() + "arg" + i; if (fieldNames.add(fieldName)) { if (parameterType == Bytes.class) fields.append(format("private Bytes %s = Bytes.allocateElasticOnHeap();\n", fieldName)); else fields.append(format("private %s %s;\n", typeName, fieldName)); } } // Handling chainable return types and interceptor returns if (chainReturnType != null) hasChainedCalls = true; if (hasRealInterceptorReturns()) { fields.append(format("private final Object[] interceptor%sArgs = new Object[%d];\n", m.getName(), parameterTypes.length)); String parameterTypesArg = parameterTypes.length == 0 ? "" : ", " + Arrays.stream(parameterTypes) .map(t -> erase(t).getCanonicalName()) .map(s -> s + ".class") .collect(Collectors.joining(", ")); fields.append(format("private static final Method %smethod = lookupMethod(%s.class, \"%s\"%s);\n", m.getName(), anInterface.getCanonicalName(), m.getName(), parameterTypesArg)); } if (parameterTypes.length > 0 || hasRealInterceptorReturns()) fields.append("\n"); // Checking and handling @MethodId annotation final MethodId methodIdAnnotation = Jvm.findAnnotation(m, MethodId.class); if (methodIdAnnotation != null) { int methodId = Maths.toInt32(methodIdAnnotation.value()); addMethodIdSwitch(m.getName(), methodId, eventIdSwitchBlock); } String chainedCallPrefix = chainReturnType != null ? "chainedCallReturnResult = " : ""; // Handling code generation for event name switch block eventNameSwitchBlock.append(format("case \"%s\":\n", m.getName())); if (parameterTypes.length == 0) { eventNameSwitchBlock.append("valueIn.skipValue();\n"); eventNameSwitchBlock.append(methodCall(m, instanceFieldName, chainedCallPrefix, chainReturnType)); } else if (parameterTypes.length == 1) { eventNameSwitchBlock.append(argumentRead(m, 0, false, parameterTypes)); eventNameSwitchBlock.append(methodCall(m, instanceFieldName, chainedCallPrefix, chainReturnType)); } else { if (methodFilter) { eventNameSwitchBlock.append("ignored = false;\n"); eventNameSwitchBlock.append("valueIn.sequence(this, (f, v) -> {\n"); eventNameSwitchBlock.append(argumentRead(m, 0, true, parameterTypes)); eventNameSwitchBlock.append(format("if (((MethodFilterOnFirstArg) f.%s)." + "ignoreMethodBasedOnFirstArg(\"%s\", f.%sarg%d)) {\n", instanceFieldName, m.getName(), m.getName(), 0)); eventNameSwitchBlock.append("f.ignored = true;\n"); for (int i = 1; i < parameterTypes.length; i++) eventNameSwitchBlock.append("v.skipValue();\n"); eventNameSwitchBlock.append("}\n"); eventNameSwitchBlock.append("else {\n"); for (int i = 1; i < parameterTypes.length; i++) eventNameSwitchBlock.append(argumentRead(m, i, true, parameterTypes)); eventNameSwitchBlock.append("}\n"); eventNameSwitchBlock.append("});\n"); eventNameSwitchBlock.append("if (!ignored) {\n"); } else { eventNameSwitchBlock.append("valueIn.sequence(this, (f, v) -> { " + "// todo optimize megamorphic lambda call\n"); for (int i = 0; i < parameterTypes.length; i++) eventNameSwitchBlock.append(argumentRead(m, i, true, parameterTypes)); eventNameSwitchBlock.append("});\n"); } eventNameSwitchBlock.append(methodCall(m, instanceFieldName, chainedCallPrefix, chainReturnType)); if (methodFilter) eventNameSwitchBlock.append("}\n"); } // Handling chainable return types specifically for DocumentContext if (chainReturnType == DocumentContext.class) { eventNameSwitchBlock.append("wireIn.copyTo(((") .append(DocumentContext.class.getName()) .append(") chainedCallReturnResult).wire());\n") .append(Closeable.class.getName()) .append(".closeQuietly(chainedCallReturnResult);\n" + "chainedCallReturnResult = null;\n"); chainReturnType = null; } eventNameSwitchBlock.append("break;\n\n"); // Further handling of chainable return types if (chainReturnType != null) handleInterface(chainReturnType, "chainedCallReturnResult", false, eventNameSwitchBlock, eventIdSwitchBlock); } /** * Creates a switch block in the provided SourceCodeFormatter for a given method ID. * This method facilitates the dynamic selection of methods based on their assigned IDs, allowing * efficient routing and method invocation. * * @param methodName The name of the method for which the switch case is generated. * @param methodId The ID assigned to the method. * @param eventIdSwitchBlock Code block where the generated switch case is appended. */ private void addMethodIdSwitch(String methodName, int methodId, SourceCodeFormatter eventIdSwitchBlock) { // Append the switch case based on method ID eventIdSwitchBlock.append(format("case %d:\n", methodId)); // Set the last executed method's name eventIdSwitchBlock.append(format("lastEventName = \"%s\";\n", methodName)); // End of the switch case eventIdSwitchBlock.append("break;\n\n"); } /** * Generates code that invokes passed method, saves method return value (in case it's a chained call) * and handles {@link MethodReaderInterceptorReturns} if it's specified. * * @param m Method that is being processed. * @param instanceFieldName In generated code, method is executed on field with this name. * @param chainedCallPrefix Prefix for method call statement, passed in order to save method result for chaining. * @return Code that performs a method call. */ private String methodCall(Method m, String instanceFieldName, String chainedCallPrefix, @Nullable Class returnType) { StringBuilder res = new StringBuilder(); Class[] parameterTypes = m.getParameterTypes(); // Array to store argument references used for method call String[] args = new String[parameterTypes.length]; for (int i = 0; i < parameterTypes.length; i++) args[i] = m.getName() + "arg" + i; // Begin try block for method invocation res.append("try {\n"); // Flag indicating a data event has been processed res.append("dataEventProcessed = true;\n"); // called for no interceptor and a generating interceptor if (!hasRealInterceptorReturns()) { // Potential cast to the generating interceptor type GeneratingMethodReaderInterceptorReturns generatingInterceptor = interceptor != null ? (GeneratingMethodReaderInterceptorReturns) interceptor : null; // If a generating interceptor is present, append the code before method call if (generatingInterceptor != null) { final String codeBefore = generatingInterceptor.codeBeforeCall(m, instanceFieldName, args); if (codeBefore != null) res.append(codeBefore).append("\n"); } // Generate the method call code res.append(format("%s((%s) %s).%s(%s);%n", chainedCallPrefix, m.getDeclaringClass().getCanonicalName(), instanceFieldName, m.getName(), String.join(", ", args))); // If a generating interceptor is present, append the code after method call if (generatingInterceptor != null) { final String codeAfter = generatingInterceptor.codeAfterCall(m, instanceFieldName, args); if (codeAfter != null) res.append(codeAfter).append("\n"); } } else { // called for non generating interceptor for (int i = 0; i < parameterTypes.length; i++) { res.append(format("interceptor%sArgs[%d] = %sarg%d;\n", m.getName(), i, m.getName(), i)); } // Determine if a return type cast is needed for the method String castPrefix = chainedCallPrefix.isEmpty() || returnType == null ? "" : "(" + returnType.getCanonicalName() + ")"; // Generate the code for interceptor invocation res.append(format("%s%sinterceptor.intercept(%smethod, %s, " + "interceptor%sArgs, this::actualInvoke);\n", chainedCallPrefix, castPrefix, m.getName(), instanceFieldName, m.getName())); } // Catch block for handling exceptions during method invocation res.append("} \n" + "catch (Exception e) {\n" + "throw new InvocationTargetRuntimeException(e);\n" + "}\n"); return res.toString(); } /** * Generates code for reading an argument of a method from a {@link ValueIn} object. * The argument's index and type, and whether it is read in a lambda function, * influence the generated code. If {@link LongConversion} * annotations are present on the argument, a converter field is registered. * * @param m Method for which an argument is read. * @param argIndex Index of an argument to be read. * @param inLambda {@code true} if argument is read in a lambda passed to a * {@link ValueIn#sequence(Object, BiConsumer)} call. * @param parameterTypes The types of the method parameters. * @return Code in the form of a String that retrieves the specified argument from {@link ValueIn} input. * * @see LongConversion * @see ValueIn */ private String argumentRead(Method m, int argIndex, boolean inLambda, Type[] parameterTypes) { Class numericConversionClass = null; // Numeric conversion is not supported for binary wire if (wireType == WireType.TEXT || wireType == WireType.YAML) { Annotation[] annotations = m.getParameterAnnotations()[argIndex]; // Loop through annotations to identify and set the numeric conversion class. for (Annotation a : annotations) { if (a instanceof LongConversion) { numericConversionClass = ((LongConversion) a).value(); break; } else { // For other annotations, find if they have an associated LongConversion. LongConversion lc = Jvm.findAnnotation(a.annotationType(), LongConversion.class); if (lc != null) { numericConversionClass = lc.value(); break; } } } } // Determine the type of the argument being read. final Class argumentType = erase(parameterTypes[argIndex]); String trueArgumentName = m.getName() + "arg" + argIndex; // Decide on the naming pattern based on whether the read is happening inside a lambda. String argumentName = (inLambda ? "f." : "") + trueArgumentName; String valueInName = inLambda ? "v" : "valueIn"; // Generate code based on the type of the argument. if (boolean.class.equals(argumentType)) { return format("%s = %s.bool();\n", argumentName, valueInName); } else if (byte.class.equals(argumentType)) { // If numeric conversion is available and has a shared instance. if (numericConversionClass != null && hasInstance(numericConversionClass)) { return format("%s = (byte) %s.INSTANCE.parse(%s.text());\n", argumentName, numericConversionClass.getName(), valueInName); } // If numeric conversion is available and is an instance of LongConverter. else if (numericConversionClass != null && LongConverter.class.isAssignableFrom(numericConversionClass)) { // Register a converter for this type. numericConverters.append(format("private final %s %sConverter = ObjectUtils.newInstance(%s.class);\n", numericConversionClass.getCanonicalName(), trueArgumentName, numericConversionClass.getCanonicalName())); return format("%s = (byte) %sConverter.parse(%s.text());\n", argumentName, argumentName, valueInName); } else // Default byte reading logic. return format("%s = %s.readByte();\n", argumentName, valueInName); } else if (char.class.equals(argumentType)) { // Handling character type arguments. return format("%s = %s.character();\n", argumentName, valueInName); } else if (short.class.equals(argumentType)) { // Generate code based on the type of the argument and presence of numeric conversion. if (numericConversionClass != null && hasInstance(numericConversionClass)) { return format("%s = (short) %s.INSTANCE.parse(%s.text());\n", argumentName, numericConversionClass.getName(), valueInName); } else if (numericConversionClass != null && LongConverter.class.isAssignableFrom(numericConversionClass)) { numericConverters.append(format("private final %s %sConverter = ObjectUtils.newInstance(%s.class);\n", numericConversionClass.getCanonicalName(), trueArgumentName, numericConversionClass.getCanonicalName())); return format("%s = (short) %sConverter.parse(%s.text());\n", argumentName, argumentName, valueInName); } else return format("%s = %s.int16();\n", argumentName, valueInName); } else if (int.class.equals(argumentType)) { // Generate code based on the type of the argument and presence of numeric conversion. if (numericConversionClass != null && hasInstance(numericConversionClass)) { return format("%s = (int) %s.INSTANCE.parse(%s.text());\n", argumentName, numericConversionClass.getName(), valueInName); } else if (numericConversionClass != null && LongConverter.class.isAssignableFrom(numericConversionClass)) { numericConverters.append(format("private final %s %sConverter = ObjectUtils.newInstance(%s.class);\n", numericConversionClass.getCanonicalName(), trueArgumentName, numericConversionClass.getCanonicalName())); return format("%s = (int) %sConverter.parse(%s.text());\n", argumentName, argumentName, valueInName); } else return format("%s = %s.int32();\n", argumentName, valueInName); } else if (long.class.equals(argumentType)) { // Generate code based on the type of the argument and presence of numeric conversion. if (numericConversionClass != null && hasInstance(numericConversionClass)) { return format("%s = %s.INSTANCE.parse(%s.text());\n", argumentName, numericConversionClass.getName(), valueInName); } else if (numericConversionClass != null && LongConverter.class.isAssignableFrom(numericConversionClass)) { numericConverters.append(format("private final %s %sConverter = ObjectUtils.newInstance(%s.class);\n", numericConversionClass.getCanonicalName(), trueArgumentName, numericConversionClass.getCanonicalName())); return format("%s = %sConverter.parse(%s.text());\n", argumentName, argumentName, valueInName); } else { return format("%s = %s.int64();\n", argumentName, valueInName); } } else if (float.class.equals(argumentType)) { // Handling float type arguments. return format("%s = %s.float32();\n", argumentName, valueInName); } else if (double.class.equals(argumentType)) { // Handling double type arguments. return format("%s = %s.float64();\n", argumentName, valueInName); } else if (Bytes.class.isAssignableFrom(argumentType)) { // Handling Bytes type arguments. return format("%s.bytes(%s);\n", valueInName, argumentName); } else if (CharSequence.class.isAssignableFrom(argumentType)) { // Handling CharSequence type arguments. return format("%s = %s.text();\n", argumentName, valueInName); } else { // Handling other object types. final String typeName = argumentType.getCanonicalName(); if (!argumentType.isArray() && !AbstractMarshallableCfg.class.isAssignableFrom(argumentType) && !Collection.class.isAssignableFrom(argumentType) && !Map.class.isAssignableFrom(argumentType) && Object.class != argumentType && !argumentType.isInterface()) { return format("%s = %s.object(%s, %s.class);\n", argumentName, valueInName, argumentName, typeName); } return format("%s = %s.object(checkRecycle(%s), %s.class);\n", argumentName, valueInName, argumentName, typeName); } } /** * Checks if a real interceptor is present that returns. * * @return {@code true} if there's a real interceptor present, {@code false} otherwise. */ private boolean hasRealInterceptorReturns() { return interceptor != null && !(interceptor instanceof GeneratingMethodReaderInterceptorReturns); } /** * Retrieves the package name for the generated class. * The package name is determined based on the class name of the first instance. * * @return The package name of the generated class. */ public String packageName() { Class firstClass = instances[0].getClass(); String firstClassFullName = firstClass.getName(); return ReflectionUtil.generatedPackageName(firstClassFullName); } /** * Gets the simple name of the generated class. * * @return The simple name of the generated class. */ public String generatedClassName() { return generatedClassName; } /** * Constructs the generated class name using various components such as * the names of instances, metadata handlers, wire type, and potential interceptor. * Special characters, such as underscores and slashes, are handled to format the class name. * * @return The constructed name for the generated class. */ @NotNull private String generatedClassName0() { final StringBuilder sb = new StringBuilder(); // Append names of instances to the class name. for (Object i : instances) appendInstanceName(sb, i); // Append names of metadata handlers if available. if (metaDataHandler != null) for (Object i : metaDataHandler) appendInstanceName(sb, i); // Append wire type to the class name. if (wireType != null) sb.append(wireType.toString() .replace("_", "")); // Append interceptor details to the class name. if (interceptor instanceof GeneratingMethodReaderInterceptorReturns) sb.append(((GeneratingMethodReaderInterceptorReturns) interceptor).generatorId()); else if (hasRealInterceptorReturns()) sb.append("Intercepting"); sb.append("MethodReader"); // Convert slashes for nested or inner classes. return sb.toString().replace("/", "$"); } /** * Appends the simplified name of the instance's class to the provided StringBuilder. * The method takes into account various scenarios including: * - If the class is a proxy class. * - If the class is enclosed within another class. * - If the class is synthetic, anonymous, or local. * The method aims to provide a more concise and meaningful name for the instance's class * that can be used in contexts like generating a class name. * * @param sb The StringBuilder to which the instance name should be appended. * @param i The instance for which the class name is determined. */ private void appendInstanceName(StringBuilder sb, Object i) { Class aClass = i.getClass(); // Check if it's a proxy class. if (Proxy.isProxyClass(aClass)) { aClass = aClass.getInterfaces()[0]; } // Check for enclosing class. if (aClass.getEnclosingClass() != null) sb.append(aClass.getEnclosingClass().getSimpleName()); String name = Jvm.isLambdaClass(aClass) ? aClass.getInterfaces()[0].getName() : aClass.getName(); final int packageDelimiterIndex = name.lastIndexOf('.'); // Intentionally using this instead of class.simpleName() in order to support anonymous class String nameWithoutPackage = packageDelimiterIndex == -1 ? name : name.substring(packageDelimiterIndex + 1); // Further refine the name for specific synthetic class scenarios. if (aClass.isSynthetic() && !aClass.isAnonymousClass() && !aClass.isLocalClass()) { int lambdaSlashIndex = nameWithoutPackage.lastIndexOf("/"); if (lambdaSlashIndex != -1) nameWithoutPackage = nameWithoutPackage.substring(0, lambdaSlashIndex); } // Append the refined name to the StringBuilder. sb.append(nameWithoutPackage); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy