Разумная обработка ScriptException, выданного JSR223 Rhino
Я начинаю сталкиваться с грязными маленькими секретами того, что в противном случае является очень полезной средой сценариев JSR223.
Я использую встроенную версию Rhino, поставляемую с Java 6 SE, и обращаюсь к ней через JSR223 ScriptingEngine
и другие.
Когда я получаю исключение, вызванное объектом Java, который я экспортировал в среду Javascript, это ScriptingException
это оборачивает sun.org.mozilla.javascript.internal.WrappedException
это окутывает мое настоящее исключение (например, UnsupportedOperationException
или что угодно)
ScriptingException
возвращает null для getFileName() и -1 для getLineNumber(). Но когда я смотрю на сообщение и на отладчик, WrappedException
имеет правильное имя файла и номер строки, он просто не публикуется через методы получения ScriptingException.
Отлично. Что мне теперь делать? Я не знаю, как я собираюсь использовать sun.org.mozilla.javascript.internal.wrappedException, который в любом случае не является публичным классом.
1 ответ
Argh. 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;
}