package dev.w1zzrd.spigot.wizcompat.packet; import java.lang.reflect.*; public final class Reflect { private Reflect() { throw new UnsupportedOperationException("Functional class"); } public static boolean contains(final String[] array, final String find) { for (final String check : array) if (check.equals(find)) return true; return false; } public static Method findDeclaredMethod(final Class rootType, final String[] methodNames, final Object[] args) { Class current = rootType; do { for (final Method check : current.getDeclaredMethods()) if (contains(methodNames, check.getName()) && argsMatch(check.getParameterTypes(), args)) return check; current = current.getSuperclass(); } while (true); } public static Field findDeclaredField(final Class rootType, final Class expectedType, final String... fieldNames) { Class current = rootType; do { for (final Field check : current.getDeclaredFields()) if (contains(fieldNames, check.getName()) && (expectedType == null || check.getType().equals(expectedType))) return check; current = current.getSuperclass(); } while (true); } public static Constructor findDeclaredConstructor(final Class type, final Object[] args) { for (final Constructor check : type.getDeclaredConstructors()) if (argsMatch(check.getParameterTypes(), args)) return (Constructor) check; return null; } public static Object reflectInvoke(final Object target, final String[] methodNames, final Object... args) { final Method targetMethod = findDeclaredMethod(target.getClass(), methodNames, args); targetMethod.setAccessible(true); try { return targetMethod.invoke(target, args); } catch (IllegalAccessException | InvocationTargetException e) { e.printStackTrace(); } return null; } public static void reflectSetStaticField(final Class target, final Object value, final String... fieldNames) { reflectSetStaticField(target, null, value, fieldNames); } public static void reflectSetStaticField(final Class target, final Class expectedType, final Object value, final String... fieldNames) { final Field targetField = findDeclaredField(target, expectedType, fieldNames); targetField.setAccessible(true); try { targetField.set(null, value); } catch (IllegalAccessException e) { e.printStackTrace(); } } public static T reflectGetStaticField(final Class target, final String... fieldNames) { return (T) reflectGetStaticField(target, null, fieldNames); } public static T reflectGetGenericStaticField(final Class target, final Class fieldType, final Class genericType) { for (final Field check : target.getDeclaredFields()) { if (fieldType.isAssignableFrom(check.getType())) { final Type checkFieldType = check.getGenericType(); if (checkFieldType instanceof ParameterizedType) { final ParameterizedType pCFT = (ParameterizedType) checkFieldType; if (pCFT.getActualTypeArguments().length != 0) { for (final Type typeArg : pCFT.getActualTypeArguments()) { if (typeArg == genericType) { try { return (T) check.get(null); } catch (IllegalAccessException e) { e.printStackTrace(); return null; } } } } } } } return null; } public static T reflectGetStaticField(final Class target, final Class expectedType, final String... fieldNames) { final Field targetField = findDeclaredField(target, expectedType, fieldNames); targetField.setAccessible(true); try { return (T)targetField.get(null); } catch (IllegalAccessException e) { e.printStackTrace(); } return null; } public static void reflectSetField(final Object target, final Object value, final String... fieldNames) { reflectSetField(target, null, value, fieldNames); } public static void reflectSetField(final Object target, final Class expectedType, final Object value, final String... fieldNames) { final Field targetField = findDeclaredField(target.getClass(), expectedType, fieldNames); targetField.setAccessible(true); try { targetField.set(target, value); } catch (IllegalAccessException e) { e.printStackTrace(); } } public static Object reflectGetField(final Object target, final String... fieldNames) { return reflectGetField(target, null, fieldNames); } public static T reflectGetField(final Object target, final Class expectedType, final String... fieldNames) { final Field targetField = findDeclaredField(target.getClass(), expectedType, fieldNames); targetField.setAccessible(true); try { return (T)targetField.get(target); } catch (IllegalAccessException e) { e.printStackTrace(); } return null; } public static T reflectConstruct(final Class targetType, final Object... args) { final Constructor targetConstructor = findDeclaredConstructor(targetType, args); assert targetConstructor != null; targetConstructor.setAccessible(true); try { return targetConstructor.newInstance(args); } catch (InstantiationException | InvocationTargetException | IllegalAccessException e) { e.printStackTrace(); } return null; } public static Class loadClass(final Package from, final String... names) { for (final String possibleName : names) try { return Class.forName(from.getName() + "." + possibleName); } catch (ClassNotFoundException e) { } return null; } private static boolean argsMatch(final Class[] types, final Object[] args) { if (types.length != args.length) return false; for (int i = 0; i < args.length; ++i) if (isNotSoftAssignable(types[i], args[i])) return false; return true; } private static boolean isNotSoftAssignable(final Class type, final Object arg) { return (arg == null && type.isPrimitive()) || (arg != null && !type.isAssignableFrom(arg.getClass()) && !isBoxedPrimitive(type, arg.getClass())); } private static boolean isBoxedPrimitive(final Class primitive, final Class objectType) { return (primitive == boolean.class && objectType == Boolean.class) || (primitive == byte.class && objectType == Byte.class) || (primitive == short.class && objectType == Short.class) || (primitive == int.class && objectType == Integer.class) || (primitive == long.class && objectType == Long.class) || (primitive == float.class && objectType == Float.class) || (primitive == double.class && objectType == Double.class); } private interface DeclarationGetter { R[] getDeclared(final Class t); } private interface NameGetter { String getName(final T t); } }