合理处理JSR223 Rhino抛出的ScriptException

Jas*_*n S 7 java exception rhino jsr223

我开始遇到一个非常有用的JSR223脚本环境的肮脏的小秘密.

我正在使用Java 6 SE附带的内置版Rhino,通过JSR223 ScriptingEngine等访问它.

当我得到一个由我导出到Javascript环境的Java对象引起的异常时,它会ScriptingException包装一个sun.org.mozilla.javascript.internal.WrappedException包装我的真正异常(例如UnsupportedOperationException或者其他)

ScriptingExceptiongetFileName()的返回值为null,getLineNumber()的返回值为-1.但是当我查看消息和调试器时,它WrappedException具有正确的文件名和行号,它只是不通过ScriptingException的getter方法发布它.

大.现在我该怎么办?我不知道我将如何使用sun.org.mozilla.javascript.internal.wrappedException,它不是公共类.

Jas*_*n S 1

啊。Java 6 的 Rhino 也做了同样的事情(不通过 ScriptingException 的方法发布文件名/行号/等),sun.org.mozilla.javascript.internal.EvaluatorException谁知道还有多少其他异常。

我能想到的处理这个问题的唯一合理方法是使用反射。这是我的解决方案。

void handleScriptingException(ScriptingException se)
{ 
    final Throwable t1 = se.getCause();
    String lineSource = null;
    String filename = null;
    Integer lineNumber = null;

    if (hasGetterMethod(t1, "sourceName"))
    {
        lineNumber = getProperty(t1, "lineNumber", Integer.class);
        filename = getProperty(t1, "sourceName", String.class);
        lineSource = getProperty(t1, "lineSource", String.class);
    }
    else
    {
        filename = se.getFileName();
        lineNumber = se.getLineNumber();
    }
    /* do something with this info */
}

static private Method getGetterMethod(Object object, String propertyName)
{
    String methodName = "get"+getBeanSuffix(propertyName);
    try {
        Class<?> cl = object.getClass();
        return cl.getMethod(methodName);
    }
    catch (NoSuchMethodException e) { 
        return null;
        /* gulp */ 
    }
}
static private String getBeanSuffix(String propertyName) {
    return propertyName.substring(0,1).toUpperCase()
       +propertyName.substring(1);  
}   
static private boolean hasGetterMethod(Object object, String propertyName) 
{
    return getGetterMethod(object, propertyName) != null;
}
static private <T> T getProperty(Object object, String propertyName, 
        Class<T> cl) {
    try {
        Object result = getGetterMethod(object, propertyName).invoke(object);
        return cl.cast(result);
    }
    catch (Exception e) {
        e.printStackTrace();
    }
    return null;
}
Run Code Online (Sandbox Code Playgroud)