разумная обработка исключения ScriptException, вызванного JSR223 Rhino

#java #exception #rhino #jsr223

#java #исключение #rhino #jsr223

Вопрос:

Я начинаю разбираться в маленьких грязных секретах того, что в остальном является очень полезной средой сценариев JSR223.

Я использую встроенную версию Rhino, поставляемую с Java 6 SE, получая доступ к ней через JSR223 ScriptingEngine и др.

Когда я получаю исключение, вызванное объектом Java, который я экспортировал в среду Javascript, это a, ScriptingException которое обертывает a sun.org.mozilla.javascript.internal.WrappedException , которое обертывает мое реальное исключение (например, UnsupportedOperationException или что-то еще)

ScriptingException Возвращает null для GetFileName() и -1 для getLineNumber(). Но когда я смотрю на сообщение и на отладчик, у WrappedException него правильное имя файла и номер строки, оно просто не публикует его с помощью методов получения ScriptingException.

Отлично. Что мне теперь делать? Я не знаю, как я собираюсь использовать исключение sun.org.mozilla.javascript.internal.wrappedException, которое в любом случае не является общедоступным классом.

Ответ №1:

Ага. Rhino в Java 6 делает то же самое (не публикует имя файла / номер строки / и т.д. С помощью методов 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;
}