From a9767e5b9c8a8f84171eb0b8c3e63b2ea3c715b0 Mon Sep 17 00:00:00 2001 From: Kenneth Lausdahl Date: Thu, 1 Mar 2018 15:43:08 +0100 Subject: [PATCH] updated delegate to enable context and search org.overture.lib if not found in first attempt #667 added missing class check --- .../overture/interpreter/util/Delegate.java | 134 +++++++++++++++--- 1 file changed, 114 insertions(+), 20 deletions(-) diff --git a/core/interpreter/src/main/java/org/overture/interpreter/util/Delegate.java b/core/interpreter/src/main/java/org/overture/interpreter/util/Delegate.java index 91d1ad967b..254c479f65 100644 --- a/core/interpreter/src/main/java/org/overture/interpreter/util/Delegate.java +++ b/core/interpreter/src/main/java/org/overture/interpreter/util/Delegate.java @@ -24,7 +24,9 @@ package org.overture.interpreter.util; import java.io.IOException; +import java.io.PrintWriter; import java.io.Serializable; +import java.io.StringWriter; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.lang.reflect.Modifier; @@ -55,6 +57,7 @@ public class Delegate implements Serializable private static final long serialVersionUID = 1L; private final String name; private List definitions; + private static final String OVERTURE_LIB_PKG_PREFIX="org.overture.lib."; public Delegate(String name, List definitions) { @@ -73,17 +76,32 @@ public boolean hasDelegate(ITypeCheckerAssistantFactory assistantFactory) { delegateChecked = true; + String classname = name.replace('_', '.'); try { - String classname = name.replace('_', '.'); delegateClass = this.getClass().getClassLoader().loadClass(classname); - delegateMethods = new HashMap(); - delegateArgs = new HashMap(); - definitions = assistantFactory.createPDefinitionListAssistant().singleDefinitions(definitions); } catch (ClassNotFoundException e) { // Fine } + + if(delegateClass==null) + { + try + { + delegateClass = this.getClass().getClassLoader().loadClass(OVERTURE_LIB_PKG_PREFIX+classname); + } catch (ClassNotFoundException e) + { + // Fine + } + } + + + if(delegateClass!=null) { + delegateMethods = new HashMap<>(); + delegateArgs = new HashMap<>(); + definitions = assistantFactory.createPDefinitionListAssistant().singleDefinitions(definitions); + } } return delegateClass != null; @@ -183,24 +201,35 @@ private Method getDelegateMethod(String title, Context ctxt) + title); } - try + // search with no context and default package + org.overture.lib + InternalException searchException = null; + Method basicMathod = null; + try { + basicMathod = getDelegateMethod(mname, ptypes); + }catch(InternalException e) { - Class[] array = new Class[0]; - m = delegateClass.getMethod(mname, ptypes.toArray(array)); + searchException = e; + } - if (!m.getReturnType().equals(Value.class)) - { - throw new InternalException(58, "Native method does not return Value: " - + m); - } - } catch (SecurityException e) + // search with context and default package + org.overture.lib this is preferred over no context + try { + ptypes.add(0,Context.class); + m = getJavaDelegateMethod(mname, ptypes); + }catch(InternalException e) + { + searchException = e; + } + + if(m==null) { - throw new InternalException(60, "Cannot access native method: " - + e.getMessage()); - } catch (NoSuchMethodException e) + m = basicMathod; + } + + + + if(m==null && searchException!=null) { - throw new InternalException(61, "Cannot find native method: " - + e.getMessage()); + throw searchException; } delegateMethods.put(title, m); @@ -209,6 +238,54 @@ private Method getDelegateMethod(String title, Context ctxt) return m; } + private Method getDelegateMethod(String mname, List> ptypes) { + Method m = null; + InternalException searchException = null; + try { + m = getJavaDelegateMethod(mname, ptypes); + }catch(InternalException e) + { + searchException = e; + } + if(m == null) + { + try { + m = getJavaDelegateMethod(OVERTURE_LIB_PKG_PREFIX+mname, ptypes); + }catch(InternalException e) + { + searchException = e; + } + } + if(m==null && searchException!=null) { + throw searchException; + } + return m; + } + + private Method getJavaDelegateMethod(String mname, List> ptypes) { + Method m; + try + { + Class[] array = new Class[0]; + m = delegateClass.getMethod(mname, ptypes.toArray(array)); + + if (!m.getReturnType().equals(Value.class)) + { + throw new InternalException(58, "Native method does not return Value: " + + m); + } + } catch (SecurityException e) + { + throw new InternalException(60, "Cannot access native method: " + + e.getMessage()); + } catch (NoSuchMethodException e) + { + throw new InternalException(61, "Cannot find native method: " + + e.getMessage()); + } + return m; + } + public Value invokeDelegate(Object delegateObject, Context ctxt) { Method m = getDelegateMethod(ctxt.title, ctxt); @@ -219,10 +296,23 @@ public Value invokeDelegate(Object delegateObject, Context ctxt) + m.getName()); } + boolean useContext = m.getParameterTypes().length >0 && m.getParameterTypes()[0].equals(Context.class); + LexNameList anames = delegateArgs.get(ctxt.title); - Object[] avals = new Object[anames.size()]; + + int argCount = anames.size(); + if(useContext) + { + argCount++; + } + Object[] avals = new Object[argCount]; int a = 0; + if(useContext) + { + avals[a++] = ctxt; + } + for (ILexNameToken arg : anames) { avals[a++] = ctxt.get(arg); @@ -253,8 +343,12 @@ public Value invokeDelegate(Object delegateObject, Context ctxt) // { // throw (ValueException)e.getTargetException(); // } + + StringWriter strOut = new StringWriter(); + strOut.append(e.getTargetException().getMessage()+"\n"); + e.getTargetException().printStackTrace(new PrintWriter(strOut)); throw new InternalException(59, "Failed in native method: " - + e.getTargetException().getMessage()); + + strOut); } }