Merge pull request #6 from aNNiMON/java-interop-fix

Java interop fixes and improvements
This commit is contained in:
Victor Melnik 2019-10-04 22:27:44 +03:00 committed by GitHub
commit 0614d0fa53
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 144 additions and 35 deletions

View File

@ -1,15 +1,14 @@
package com.annimon.ownlang.lib; package com.annimon.ownlang.lib;
import com.annimon.ownlang.exceptions.UnknownFunctionException;
import com.annimon.ownlang.parser.ast.ClassDeclarationStatement; import com.annimon.ownlang.parser.ast.ClassDeclarationStatement;
import java.util.HashMap;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
public final class ClassDeclarations { public final class ClassDeclarations {
private static final Map<String, ClassDeclarationStatement> declarations; private static final Map<String, ClassDeclarationStatement> declarations;
static { static {
declarations = new HashMap<>(); declarations = new ConcurrentHashMap<>();
} }
private ClassDeclarations() { } private ClassDeclarations() { }
@ -22,12 +21,7 @@ public final class ClassDeclarations {
return declarations; return declarations;
} }
public static boolean isExists(String key) {
return declarations.containsKey(key);
}
public static ClassDeclarationStatement get(String key) { public static ClassDeclarationStatement get(String key) {
if (!isExists(key)) throw new UnknownFunctionException(key);
return declarations.get(key); return declarations.get(key);
} }

View File

@ -0,0 +1,9 @@
package com.annimon.ownlang.lib;
/**
* Interface for values that supports creating instances with `new` keyword.
*/
public interface Instantiable {
Value newInstance(Value[] args);
}

View File

@ -3,10 +3,12 @@ package com.annimon.ownlang.modules.java;
import com.annimon.ownlang.lib.*; import com.annimon.ownlang.lib.*;
import com.annimon.ownlang.modules.Module; import com.annimon.ownlang.modules.Module;
import java.lang.reflect.Array; import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field; import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException; import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method; import java.lang.reflect.Method;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.List; import java.util.List;
/** /**
@ -102,7 +104,7 @@ public final class java implements Module {
} }
} }
private static class ClassValue extends MapValue { private static class ClassValue extends MapValue implements Instantiable {
public static Value classOrNull(Class<?> clazz) { public static Value classOrNull(Class<?> clazz) {
if (clazz == null) return NULL; if (clazz == null) return NULL;
@ -151,25 +153,22 @@ public final class java implements Module {
set("cast", new FunctionValue(this::cast)); set("cast", new FunctionValue(this::cast));
} }
private Value asSubclass(Value... args) { private Value asSubclass(Value[] args) {
Arguments.check(1, args.length); Arguments.check(1, args.length);
return new ClassValue(clazz.asSubclass( ((ClassValue)args[0]).clazz )); return new ClassValue(clazz.asSubclass( ((ClassValue)args[0]).clazz ));
} }
private Value isAssignableFrom(Value... args) { private Value isAssignableFrom(Value[] args) {
Arguments.check(1, args.length); Arguments.check(1, args.length);
return NumberValue.fromBoolean(clazz.isAssignableFrom( ((ClassValue)args[0]).clazz )); return NumberValue.fromBoolean(clazz.isAssignableFrom( ((ClassValue)args[0]).clazz ));
} }
private Value newInstance(Value... args) { @Override
try { public Value newInstance(Value[] args) {
return new ObjectValue(clazz.newInstance()); return findConstructorAndInstantiate(args, clazz.getConstructors());
} catch (InstantiationException | IllegalAccessException ex) {
return NULL;
}
} }
private Value cast(Value... args) { private Value cast(Value[] args) {
Arguments.check(1, args.length); Arguments.check(1, args.length);
return objectToValue(clazz, clazz.cast(((ObjectValue)args[0]).object)); return objectToValue(clazz, clazz.cast(((ObjectValue)args[0]).object));
} }
@ -209,7 +208,7 @@ public final class java implements Module {
@Override @Override
public boolean containsKey(Value key) { public boolean containsKey(Value key) {
return getValue(object.getClass(), object, key.asString()) != null; return get(key) != null;
} }
@Override @Override
@ -229,7 +228,7 @@ public final class java implements Module {
} }
//</editor-fold> //</editor-fold>
private Value isNull(Value... args) { private Value isNull(Value[] args) {
Arguments.checkAtLeast(1, args.length); Arguments.checkAtLeast(1, args.length);
for (Value arg : args) { for (Value arg : args) {
if (arg.raw() == null) return NumberValue.ONE; if (arg.raw() == null) return NumberValue.ONE;
@ -237,24 +236,24 @@ public final class java implements Module {
return NumberValue.ZERO; return NumberValue.ZERO;
} }
private Value newClass(Value... args) { private Value newClass(Value[] args) {
Arguments.check(1, args.length); Arguments.check(1, args.length);
final String className = args[0].asString(); final String className = args[0].asString();
try { try {
return new ClassValue(Class.forName(className)); return new ClassValue(Class.forName(className));
} catch (ClassNotFoundException ce) { } catch (ClassNotFoundException ce) {
return NULL; throw new RuntimeException("Class " + className + " not found.", ce);
} }
} }
private Value toObject(Value... args) { private Value toObject(Value[] args) {
Arguments.check(1, args.length); Arguments.check(1, args.length);
if (args[0] == NULL) return NULL; if (args[0] == NULL) return NULL;
return new ObjectValue(valueToObject(args[0])); return new ObjectValue(valueToObject(args[0]));
} }
private Value toValue(Value... args) { private Value toValue(Value[] args) {
Arguments.check(1, args.length); Arguments.check(1, args.length);
if (args[0] instanceof ObjectValue) { if (args[0] instanceof ObjectValue) {
return objectToValue( ((ObjectValue) args[0]).object ); return objectToValue( ((ObjectValue) args[0]).object );
@ -294,6 +293,22 @@ public final class java implements Module {
return NULL; return NULL;
} }
private static Value findConstructorAndInstantiate(Value[] args, Constructor<?>[] ctors) {
for (Constructor<?> ctor : ctors) {
if (ctor.getParameterCount() != args.length) continue;
if (!isMatch(args, ctor.getParameterTypes())) continue;
try {
final Object result = ctor.newInstance(valuesToObjects(args));
return new ObjectValue(result);
} catch (InstantiationException | IllegalAccessException
| IllegalArgumentException | InvocationTargetException ex) {
// skip
}
}
throw new RuntimeException("Constructor for " + args.length + " arguments"
+ " not found or non accessible");
}
private static Function methodsToFunction(Object object, List<Method> methods) { private static Function methodsToFunction(Object object, List<Method> methods) {
return (args) -> { return (args) -> {
for (Method method : methods) { for (Method method : methods) {
@ -309,7 +324,9 @@ public final class java implements Module {
// skip // skip
} }
} }
return null; final String className = (object == null ? "null" : object.getClass().getName());
throw new RuntimeException("Method for " + args.length + " arguments"
+ " not found or non accessible in " + className);
}; };
} }
@ -324,7 +341,15 @@ public final class java implements Module {
boolean assignable = unboxed != null; boolean assignable = unboxed != null;
final Object object = valueToObject(arg); final Object object = valueToObject(arg);
assignable = assignable && (object != null); assignable = assignable && (object != null);
if (assignable && unboxed.isArray() && object.getClass().isArray()) {
final Class<?> uComponentType = unboxed.getComponentType();
final Class<?> oComponentType = object.getClass().getComponentType();
assignable = assignable && (uComponentType != null);
assignable = assignable && (oComponentType != null);
assignable = assignable && (uComponentType.isAssignableFrom(oComponentType));
} else {
assignable = assignable && (unboxed.isAssignableFrom(object.getClass())); assignable = assignable && (unboxed.isAssignableFrom(object.getClass()));
}
if (assignable) continue; if (assignable) continue;
return false; return false;
@ -477,11 +502,72 @@ public final class java implements Module {
private static Object arrayToObject(ArrayValue value) { private static Object arrayToObject(ArrayValue value) {
final int size = value.size(); final int size = value.size();
final Object[] result = new Object[size]; final Object[] array = new Object[size];
for (int i = 0; i < size; i++) { if (size == 0) {
result[i] = valueToObject(value.get(i)); return array;
}
Class<?> elementsType = null;
for (int i = 0; i < size; i++) {
array[i] = valueToObject(value.get(i));
if (i == 0) {
elementsType = array[0].getClass();
} else {
elementsType = mostCommonType(elementsType, array[i].getClass());
}
}
if (elementsType.equals(Object[].class)) {
return array;
}
return typedArray(array, size, elementsType);
}
private static <T, U> T[] typedArray(U[] elements, int newLength, Class<?> elementsType) {
@SuppressWarnings("unchecked")
T[] copy = (T[]) Array.newInstance(elementsType, newLength);
System.arraycopy(elements, 0, copy, 0, Math.min(elements.length, newLength));
return copy;
}
private static Class<?> mostCommonType(Class<?> c1, Class<?> c2) {
if (c1.equals(c2)) {
return c1;
} else if (c1.isAssignableFrom(c2)) {
return c1;
} else if (c2.isAssignableFrom(c1)) {
return c2;
}
final Class<?> s1 = c1.getSuperclass();
final Class<?> s2 = c2.getSuperclass();
if (s1 == null && s2 == null) {
final List<Class<?>> upperTypes = Arrays.asList(
Object.class, void.class, boolean.class, char.class,
byte.class, short.class, int.class, long.class,
float.class, double.class);
for (Class<?> type : upperTypes) {
if (c1.equals(type) && c2.equals(type)) {
return s1;
}
}
return Object.class;
} else if (s1 == null || s2 == null) {
if (c1.equals(c2)) {
return c1;
}
if (c1.isInterface() && c1.isAssignableFrom(c2)) {
return c1;
}
if (c2.isInterface() && c2.isAssignableFrom(c1)) {
return c2;
}
}
if (s1 != null) {
return mostCommonType(s1, c2);
} else {
return mostCommonType(c1, s2);
} }
return result;
} }
//</editor-fold> //</editor-fold>
} }

View File

@ -1,5 +1,6 @@
package com.annimon.ownlang.parser.ast; package com.annimon.ownlang.parser.ast;
import com.annimon.ownlang.exceptions.UnknownClassException;
import com.annimon.ownlang.lib.*; import com.annimon.ownlang.lib.*;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
@ -17,6 +18,16 @@ public final class ObjectCreationExpression implements Expression {
@Override @Override
public Value eval() { public Value eval() {
final ClassDeclarationStatement cd = ClassDeclarations.get(className); final ClassDeclarationStatement cd = ClassDeclarations.get(className);
if (cd == null) {
// Is Instantiable?
if (Variables.isExists(className)) {
final Value variable = Variables.get(className);
if (variable instanceof Instantiable) {
return ((Instantiable) variable).newInstance(ctorArgs());
}
}
throw new UnknownClassException(className);
}
// Create an instance and put evaluated fields with method declarations // Create an instance and put evaluated fields with method declarations
final ClassInstanceValue instance = new ClassInstanceValue(className); final ClassInstanceValue instance = new ClassInstanceValue(className);
@ -30,14 +41,17 @@ public final class ObjectCreationExpression implements Expression {
} }
// Call a constructor // Call a constructor
instance.callConstructor(ctorArgs());
return instance;
}
private Value[] ctorArgs() {
final int argsSize = constructorArguments.size(); final int argsSize = constructorArguments.size();
final Value[] ctorArgs = new Value[argsSize]; final Value[] ctorArgs = new Value[argsSize];
for (int i = 0; i < argsSize; i++) { for (int i = 0; i < argsSize; i++) {
ctorArgs[i] = constructorArguments.get(i).eval(); ctorArgs[i] = constructorArguments.get(i).eval();
} }
instance.callConstructor(ctorArgs); return ctorArgs;
return instance;
} }
@Override @Override

View File

@ -36,6 +36,12 @@ def testInvokeMethodSameName() {
assertEquals("text", data.getText()) assertEquals("text", data.getText())
} }
def testNonDefaultConstructor() {
StringBuilder = newClass("java.lang.StringBuilder")
sb = new StringBuilder("text")
assertEquals("text", sb.toString())
}
def createObject() { def createObject() {
dataClass = newClass("interop.Data") dataClass = newClass("interop.Data")