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

com.fluxtion.runtime.partition.LambdaReflection Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2025 gregory higgins.
 * All rights reserved.
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the Server Side Public License, version 1,
 * as published by MongoDB, Inc.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * Server Side Public License for more details.
 *
 * You should have received a copy of the Server Side Public License
 * along with this program.  If not, see
 * .
 */
package com.fluxtion.runtime.partition;

import lombok.SneakyThrows;

import java.io.Serializable;
import java.lang.invoke.*;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.Objects;
import java.util.function.*;

/**
 * @author Greg Higgins
 */
public interface LambdaReflection {

    interface MethodReferenceReflection {

        //inspired by: http://benjiweber.co.uk/blog/2015/08/17/lambda-parameter-names-with-reflection/
        default SerializedLambda serialized() {
            try {
                Method replaceMethod = getClass().getDeclaredMethod("writeReplace");
                replaceMethod.setAccessible(true);
                return (SerializedLambda) replaceMethod.invoke(this);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }

        default Class getContainingClass(ClassLoader loader) {
            try {
                String className = serialized().getImplClass().replaceAll("/", ".");
                return Class.forName(className, true, loader);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }

        default Class getContainingClass() {
            try {
                String className = serialized().getImplClass().replaceAll("/", ".");
                return Class.forName(className);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }

        default Object[] captured() {
            final SerializedLambda serialized = serialized();
            Object[] args = new Object[serialized.getCapturedArgCount()];
            for (int i = 0; i < serialized.getCapturedArgCount(); i++) {
                args[i] = serialized.getCapturedArg(i);
            }
            return args;
        }

        default Method method(ClassLoader loader) {
            SerializedLambda lambda = serialized();
            Class containingClass = getContainingClass(loader);
            return Arrays.stream(containingClass.getDeclaredMethods())
                    .filter(method -> Objects.equals(method.getName(), lambda.getImplMethodName()))
                    .findFirst()
                    .orElseThrow(UnableToGuessMethodException::new);
        }

        default boolean isDefaultConstructor() {
            return serialized().getImplMethodName().equalsIgnoreCase("");
        }

        @SneakyThrows
        default Method method() {
            SerializedLambda lambda = serialized();
            Class containingClass = getContainingClass();
            return Arrays.stream(containingClass.getDeclaredMethods())
                    .filter(method -> Objects.equals(method.getName(), lambda.getImplMethodName()))
                    .findFirst()
                    .orElseThrow(UnableToGuessMethodException::new);
        }

        class UnableToGuessMethodException extends RuntimeException {
        }
    }

    interface SerializableRunnable extends Runnable, Serializable, MethodReferenceReflection {
    }

    interface SerializableSupplier extends Supplier, Serializable, MethodReferenceReflection {
    }

    interface SerializableIntSupplier extends IntSupplier, Serializable, MethodReferenceReflection {
    }

    interface SerializableDoubleSupplier extends DoubleSupplier, Serializable, MethodReferenceReflection {
    }

    interface SerializableLongSupplier extends LongSupplier, Serializable, MethodReferenceReflection {
    }

    interface SerializableConsumer extends Consumer, Serializable, MethodReferenceReflection {
    }

    interface SerializableIntConsumer extends IntConsumer, Serializable, MethodReferenceReflection {
    }

    interface SerializableDoubleConsumer extends DoubleConsumer, Serializable, MethodReferenceReflection {
    }

    interface SerializableLongConsumer extends LongConsumer, Serializable, MethodReferenceReflection {
    }

    interface SerializableBiConsumer extends BiConsumer, Serializable, MethodReferenceReflection {
    }

    interface SerializableFunction extends Function, Serializable, MethodReferenceReflection {
    }

    interface SerializableIntFunction extends IntFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableDoubleFunction extends DoubleFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableLongFunction extends LongFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableToIntFunction extends ToIntFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableIntUnaryOperator extends IntUnaryOperator, Serializable, MethodReferenceReflection {
    }

    interface SerializableDoubleUnaryOperator extends DoubleUnaryOperator, Serializable, MethodReferenceReflection {
    }

    interface SerializableLongUnaryOperator extends LongUnaryOperator, Serializable, MethodReferenceReflection {
    }

    interface SerializableToDoubleFunction extends ToDoubleFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableToLongFunction extends ToLongFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableDoubleToIntFunction extends DoubleToIntFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableLongToIntFunction extends LongToIntFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableIntToDoubleFunction extends IntToDoubleFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableLongToDoubleFunction extends LongToDoubleFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableIntToLongFunction extends IntToLongFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableDoubleToLongFunction extends DoubleToLongFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableBiFunction extends BiFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableBiIntFunction extends IntBinaryOperator, Serializable, MethodReferenceReflection {
    }

    interface SerializableBiIntPredicate extends IntBinaryPredicate, Serializable, MethodReferenceReflection {
    }

    interface SerializableBiDoublePredicate extends DoubleBinaryPredicate, Serializable, MethodReferenceReflection {
    }

    interface SerializableBiLongPredicate extends LongBinaryPredicate, Serializable, MethodReferenceReflection {
    }


    interface SerializableBiDoubleFunction extends DoubleBinaryOperator, Serializable, MethodReferenceReflection {
    }

    interface SerializableBiLongFunction extends LongBinaryOperator, Serializable, MethodReferenceReflection {
    }

    interface SerializableTriFunction extends TriFunction, Serializable, MethodReferenceReflection {
    }

    interface SerializableQuadFunction extends QuadFunction, Serializable, MethodReferenceReflection {
    }

    @FunctionalInterface
    interface IntBinaryPredicate {
        boolean apply(int argument1, int argument2);
    }

    @FunctionalInterface
    interface DoubleBinaryPredicate {
        boolean apply(double argument1, double argument2);
    }

    @FunctionalInterface
    interface LongBinaryPredicate {
        boolean apply(long argument1, long argument2);
    }

    @FunctionalInterface
    interface TriFunction {
        R apply(F f, T t, U u);
    }

    @FunctionalInterface
    interface QuadFunction {
        R apply(F f, T t, U u, V v);
    }

    @FunctionalInterface
    interface SerializableTriConsumer extends Serializable, MethodReferenceReflection {
        void accept(A a, B b, C c);
    }

    @FunctionalInterface
    interface SerializableQuadConsumer extends Serializable, MethodReferenceReflection {
        void accept(A a, B b, C c, D d);
    }

    @FunctionalInterface
    interface SerializableQuinConsumer extends Serializable, MethodReferenceReflection {
        void accept(A a, B b, C c, D d, E e);
    }

    @FunctionalInterface
    interface SerializableSextConsumer extends Serializable, MethodReferenceReflection {
        void accept(A a, B b, C c, D d, E e, F f);
    }

    @FunctionalInterface
    interface SerializableSeptConsumer extends Serializable, MethodReferenceReflection {
        void accept(A a, B b, C c, D d, E e, F f, G g);
    }

    static  Method getMethod(LambdaReflection.SerializableConsumer supplier) {
        return supplier.method();
    }

    static  Method getMethod(LambdaReflection.SerializableFunction supplier) {
        return supplier.method();
    }

    static  Method getMethod(LambdaReflection.SerializableBiFunction supplier) {
        return supplier.method();
    }

    @SneakyThrows
    @SuppressWarnings("all")
    public static  LambdaReflection.SerializableFunction method2Function(Method keyMethod) {
        MethodHandles.Lookup lookup = MethodHandles.lookup();
        Class clazz = keyMethod.getDeclaringClass();
        CallSite site = LambdaMetafactory.altMetafactory(lookup,
                "apply",
                MethodType.methodType(LambdaReflection.SerializableFunction.class),
                MethodType.methodType(Object.class, Object.class),
                lookup.findVirtual(clazz, keyMethod.getName(), MethodType.methodType(keyMethod.getReturnType())),
                MethodType.methodType(keyMethod.getReturnType(), clazz),
                LambdaMetafactory.FLAG_SERIALIZABLE);
        return (LambdaReflection.SerializableFunction) site.getTarget().invokeExact();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy