org.babyfish.jimmer.client.meta.impl.JetBrainsMetadata Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jimmer-client Show documentation
Show all versions of jimmer-client Show documentation
A revolutionary ORM framework for both java and kotlin
package org.babyfish.jimmer.client.meta.impl;
import kotlin.Metadata;
import kotlin.jvm.JvmClassMappingKt;
import kotlin.reflect.*;
import kotlin.reflect.full.KClasses;
import kotlin.reflect.jvm.ReflectJvmMapping;
import org.babyfish.jimmer.client.IllegalDocMetaException;
import org.babyfish.jimmer.impl.asm.*;
import org.jetbrains.annotations.Nullable;
import javax.validation.constraints.Null;
import java.io.IOException;
import java.io.InputStream;
import java.lang.annotation.Annotation;
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.*;
class JetBrainsMetadata {
private static final Set> BOX_TYPES = new HashSet<>(
Arrays.asList(
Boolean.class,
Character.class,
Byte.class,
Short.class,
Integer.class,
Long.class,
Float.class,
Double.class
)
);
private final Class> javaClass;
private final KClass> kotlinClass;
private final Map> kCallableMap;
private Nullity nullity;
JetBrainsMetadata(Class> javaClass) {
this.javaClass = javaClass;
if (javaClass.isAnnotationPresent(Metadata.class)) {
this.kotlinClass = JvmClassMappingKt.getKotlinClass(javaClass);
Map> callableMap = new HashMap<>();
for (KProperty1, ?> prop : KClasses.getDeclaredMemberProperties(kotlinClass)) {
Field field = ReflectJvmMapping.getJavaField(prop);
if (field != null) {
callableMap.put(field, prop);
}
Method getter = ReflectJvmMapping.getJavaGetter(prop);
if (getter != null) {
callableMap.put(getter, prop);
}
if (prop instanceof KMutableProperty>) {
Method setter = ReflectJvmMapping.getJavaSetter((KMutableProperty>) prop);
if (getter != null) {
callableMap.put(setter, prop);
}
}
}
for (KFunction> func : KClasses.getDeclaredFunctions(kotlinClass)) {
Method method = ReflectJvmMapping.getJavaMethod(func);
callableMap.put(method, func);
}
this.kCallableMap = callableMap;
} else {
this.kotlinClass = null;
this.kCallableMap = null;
}
}
public boolean isKotlinClass() {
return kotlinClass != null;
}
public KFunction> toKFunction(Method method) {
if (kCallableMap == null) {
throw new IllegalStateException(
"The current class \"" +
javaClass.getName() +
"\" is not kotlin class"
);
}
return (KFunction>) kCallableMap.get(method);
}
private Nullity getNullity() {
Nullity nty = nullity;
if (nty == null) {
if (kotlinClass != null) {
throw new IllegalStateException("It is unnecessary to create nullity for kotlin class");
}
nullity = nty = new Nullity(javaClass);
}
return nty;
}
public boolean isNullable(Field field) {
return isNullable(field, field.getType());
}
public boolean isNullable(Method method) {
return isNullable(method, method.getReturnType());
}
private boolean isNullable(AccessibleObject member, Class> type) {
if (kCallableMap != null) {
KCallable> callable = kCallableMap.get(member);
return callable != null && callable.getReturnType().isMarkedNullable();
}
if (type.isPrimitive()) {
return false;
}
if (BOX_TYPES.contains(type)) {
return true;
}
for (Annotation annotation : member.getAnnotations()) {
String simpleName = annotation.annotationType().getSimpleName();
if (simpleName.equals("Null") || simpleName.equals("Nullable")) {
return true;
}
}
return getNullity().nullableMembers.contains(member);
}
public boolean isNullable(Method method, int parameterIndex) {
if (kCallableMap != null) {
KCallable> callable = kCallableMap.get(method);
return callable != null && callable.getParameters().get(parameterIndex).getType().isMarkedNullable();
}
Set indices = getNullity().nullableParameterIndices.get(method);
return indices != null && indices.contains(parameterIndex);
}
private static class Nullity {
final Set nullableMembers = new HashSet<>();
final Map> nullableParameterIndices = new HashMap<>();
Nullity(Class> clazz) {
if (clazz.isPrimitive() || BOX_TYPES.contains(clazz)) {
return;
}
Map accessibleObjectMap = new HashMap<>();
for (Field field : clazz.getDeclaredFields()) {
accessibleObjectMap.put(new Member(field.getName(), Type.getDescriptor(field.getType())), field);
}
for (Method method : clazz.getDeclaredMethods()) {
accessibleObjectMap.put(new Member(method.getName(), Type.getMethodDescriptor(method)), method);
}
ClassLoader loader = clazz.getClassLoader();
if (loader != null) {
try (InputStream inputStream = loader.getResourceAsStream(clazz.getName().replace('.', '/') + ".class")) {
ClassReader reader = new ClassReader(inputStream);
reader.accept(
new ClassVisitor(Opcodes.ASM9, null) {
@Override
public FieldVisitor visitField(int access, String name, String descriptor, String signature, Object value) {
return (access & Opcodes.ACC_STATIC) == 0 ?
new FieldVisitorImpl(accessibleObjectMap.get(new Member(name, descriptor))) :
null;
}
@Override
public MethodVisitor visitMethod(int access, String name, String descriptor, String signature, String[] exceptions) {
return (access & Opcodes.ACC_STATIC) == 0 ?
new MethodVisitorImpl(accessibleObjectMap.get(new Member(name, descriptor))) :
null;
}
},
ClassReader.SKIP_CODE | ClassReader.SKIP_FRAMES | ClassReader.SKIP_DEBUG
);
} catch (IOException ex) {
throw new IllegalDocMetaException(
"Failed to parse the jetbrains nullity for class \"" +
clazz.getName() +
"\"",
ex
);
}
}
}
private class FieldVisitorImpl extends FieldVisitor {
private final AccessibleObject field;
protected FieldVisitorImpl(AccessibleObject field) {
super(Opcodes.ASM9);
this.field = field;
}
@Override
public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
if (descriptor.endsWith("/Nullable;")) {
nullableMembers.add(field);
}
return null;
}
}
private class MethodVisitorImpl extends MethodVisitor {
private final AccessibleObject method;
protected MethodVisitorImpl(AccessibleObject method) {
super(Opcodes.ASM9);
this.method = method;
}
@Override
public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
if (descriptor.endsWith("/Nullable;")) {
nullableMembers.add(method);
}
return null;
}
@Override
public AnnotationVisitor visitParameterAnnotation(int parameter, String descriptor, boolean visible) {
if (descriptor.endsWith("/Nullable;")) {
nullableParameterIndices
.computeIfAbsent(method, it -> new HashSet<>())
.add(parameter);
}
return null;
}
}
}
private static class Member {
final String name;
final String descriptor;
public Member(String name, String descriptor) {
this.name = name;
this.descriptor = descriptor;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Member member = (Member) o;
return name.equals(member.name) && descriptor.equals(member.descriptor);
}
@Override
public int hashCode() {
return Objects.hash(name, descriptor);
}
@Override
public String toString() {
return name + ':' + descriptor;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy