package org.jdbi.v3.core.argument;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;
import java.util.function.Function;
import java.util.stream.Collectors;
import org.jdbi.v3.core.argument.internal.ObjectPropertyNamedArgumentFinder;
import org.jdbi.v3.core.argument.internal.TypedValue;
import org.jdbi.v3.core.config.ConfigRegistry;
import org.jdbi.v3.core.config.JdbiCache;
import org.jdbi.v3.core.config.JdbiCaches;
import org.jdbi.v3.core.internal.exceptions.Unchecked;
import org.jdbi.v3.core.qualifier.QualifiedType;
import org.jdbi.v3.core.qualifier.Qualifiers;
import org.jdbi.v3.core.statement.StatementContext;
@Deprecated
public class ObjectMethodArguments extends ObjectPropertyNamedArgumentFinder {
private static final JdbiCache<Class<?>, Map<String, Function<Object, TypedValue>>> NULLARY_METHOD_CACHE =
JdbiCaches.declare(ObjectMethodArguments::load);
public ObjectMethodArguments(String prefix, Object object) {
super(prefix, object);
}
private static Map<String, Function<Object, TypedValue>> load(ConfigRegistry config, Class<?> type) {
final HashMap<String, Function<Object, TypedValue>> methodMap = new HashMap<>();
if (Modifier.isPublic(type.getModifiers())) {
Arrays.stream(type.getMethods())
.filter(m -> m.getParameterCount() == 0)
.collect(Collectors.toMap(
Method::getName,
Function.identity(),
ObjectMethodArguments::bridgeMethodMerge))
.forEach((name, method) -> {
QualifiedType<?> qualifiedType = QualifiedType.of(method.getGenericReturnType())
.withAnnotations(config.get(Qualifiers.class).findFor(method));
MethodHandle mh = Unchecked.function(MethodHandles.lookup()::unreflect).apply(method);
methodMap.put(name, Unchecked.function(
value -> new TypedValue(qualifiedType, mh.invoke(value))));
});
} else {
Optional.ofNullable(type.getSuperclass()).ifPresent(superclass -> methodMap.putAll(load(config, superclass)));
Arrays.stream(type.getInterfaces()).forEach(interfaceClass -> methodMap.putAll(load(config, interfaceClass)));
}
return methodMap;
}
@Override
protected Optional<TypedValue> getValue(String name, StatementContext ctx) {
return getter(name, ctx.getConfig()).map(m -> m.apply(obj));
}
public Optional<Function<Object, TypedValue>> getter(String name, ConfigRegistry config) {
return Optional.ofNullable(NULLARY_METHOD_CACHE.get(obj.getClass(), config).get(name));
}
@Override
protected NamedArgumentFinder getNestedArgumentFinder(TypedValue value) {
return new ObjectMethodArguments(null, value.getValue());
}
@Override
public String toString() {
return "{lazy object functions arguments \"" + obj + "\"";
}
private static Method bridgeMethodMerge(Method a, Method b) {
return a.isBridge() ? b : a;
}
}