JSF: не удается поймать исключение ViewExpiredException
Я разрабатываю приложение JSF 2.0 на Glassfish v3 и пытаюсь обработать исключение ViewExpiredException. Но что бы я ни делал, я всегда получаю сообщение об ошибке Glassfish вместо своей собственной страницы ошибки.
Чтобы смоделировать возникновение VEE, я вставил следующую функцию в свой компонент поддержки, который запускает VEE. Я запускаю эту функцию со своей JSF-страницы через commandLink. Код:
@Named
public class PersonHome {
(...)
public void throwVEE() {
throw new ViewExpiredException();
}
}
Сначала я попробовал это, просто добавив страницу ошибки в мой web.xml:
<error-page>
<exception-type>javax.faces.application.ViewExpiredException</exception-type>
<location>/error.xhtml</location>
</error-page>
Но это не работает, я не перенаправлен на ошибку, но мне показывают страницу ошибки Glassfish, которая показывает страницу HTTP Status 500 со следующим содержанием:
description:The server encountered an internal error () that prevented it from fulfilling this request.
exception: javax.servlet.ServletException: javax.faces.application.ViewExpiredException
root cause: javax.faces.el.EvaluationException:javax.faces.application.ViewExpiredException
root cause:javax.faces.application.ViewExpiredException
Затем я попытался написать ExceptionHandlerFactory и CustomExceptionHandler, как описано в JavaServerFaces 2.0 - Полный справочник. Поэтому я вставил следующий тег в Face-config.xml:
<factory>
<exception-handler-factory>
exceptions.ExceptionHandlerFactory
</exception-handler-factory>
</factory>
И добавил эти классы: Фабрика:
package exceptions;
import javax.faces.context.ExceptionHandler;
public class ExceptionHandlerFactory extends javax.faces.context.ExceptionHandlerFactory {
private javax.faces.context.ExceptionHandlerFactory parent;
public ExceptionHandlerFactory(javax.faces.context.ExceptionHandlerFactory parent) {
this.parent = parent;
}
@Override
public ExceptionHandler getExceptionHandler() {
ExceptionHandler result = parent.getExceptionHandler();
result = new CustomExceptionHandler(result);
return result;
}
}
Пользовательский обработчик исключений:
package exceptions;
import java.util.Iterator;
import javax.faces.FacesException;
import javax.faces.application.NavigationHandler;
import javax.faces.application.ViewExpiredException;
import javax.faces.context.ExceptionHandler;
import javax.faces.context.ExceptionHandlerWrapper;
import javax.faces.context.FacesContext;
import javax.faces.event.ExceptionQueuedEvent;
import javax.faces.event.ExceptionQueuedEventContext;
class CustomExceptionHandler extends ExceptionHandlerWrapper {
private ExceptionHandler parent;
public CustomExceptionHandler(ExceptionHandler parent) {
this.parent = parent;
}
@Override
public ExceptionHandler getWrapped() {
return this.parent;
}
@Override
public void handle() throws FacesException {
for (Iterator<ExceptionQueuedEvent> i = getUnhandledExceptionQueuedEvents().iterator(); i.hasNext();) {
ExceptionQueuedEvent event = i.next();
System.out.println("Iterating over ExceptionQueuedEvents. Current:" + event.toString());
ExceptionQueuedEventContext context = (ExceptionQueuedEventContext) event.getSource();
Throwable t = context.getException();
if (t instanceof ViewExpiredException) {
ViewExpiredException vee = (ViewExpiredException) t;
FacesContext fc = FacesContext.getCurrentInstance();
NavigationHandler nav =
fc.getApplication().getNavigationHandler();
try {
// Push some useful stuff to the flash scope for
// use in the page
fc.getExternalContext().getFlash().put("expiredViewId", vee.getViewId());
nav.handleNavigation(fc, null, "/login?faces-redirect=true");
fc.renderResponse();
} finally {
i.remove();
}
}
}
// At this point, the queue will not contain any ViewExpiredEvents.
// Therefore, let the parent handle them.
getWrapped().handle();
}
}
Но все же я НЕ перенаправлен на свою страницу ошибок - я получаю ту же ошибку HTTP 500, как описано выше. Что я делаю не так, что может быть упущено в моей реализации, что исключение не обрабатывается правильно? Любая помощь высоко ценится!
РЕДАКТИРОВАТЬ
Хорошо, я честен На самом деле мой код написан на Scala, но это длинная история. я думал, что это была проблема Java все время. НАСТОЯЩЕЙ ошибкой в этом случае была моя собственная глупость. В моем (Scala) коде, в CustomExceptionHandler, я забыл добавить строку с помощью "i.remove();" Таким образом, ViewExpiredException остался в UnhandledExceptionsQueue после его обработки, и он "всплыл". И когда он вспыхивает, он становится ServletException.
Мне очень жаль, что я вас запутал!
2 ответа
Этот тест является поддельным. ViewExpiredException
обычно генерируется только во время восстановления представления (потому что оно отсутствует в сеансе), а не во время рендеринга ответа или создания экземпляра bean-компонента. В вашем случае это исключение выдается во время создания экземпляра bean-компонента, и это исключение помещается в ServletException
,
Настоящий ViewExpiredException
обычно генерируется только при отправке HTTP-запроса POST на сервер, когда сеанс HTTP истек. Таким образом, существует два способа надежного воспроизведения:
Откройте страницу JSF с формой POST (
h:form
по умолчанию уже POST) в веб-браузере, выключите сервер и очистите его рабочий каталог (важно, потому что большинство серверов будут сериализовать открытые сеансы на диск при завершении работы и десериализовать их при запуске), перезапустите сервер и отправьте уже открытую форму.ViewExpiredException
будет брошен.Установить
<session-timeout>
вweb.xml
в1
минуту и отправьте форму в течение 1 минуты после открытия страницы JSF с формой POST. Это броситViewExpiredException
также.
Я не эксперт. Это просто дикие догадки или предложения.
1) Попробуйте перенаправить на стандартную HTML-страницу, чтобы увидеть, работает ли она 2) Исходя из этого, она должна работать с первым подходом, попробуйте написать JSF PhaseListener и сгенерировать то же исключение в RESTORE VIEW Phase.Right прямо сейчас, вы выбрасывание в фазе INVOKE APPLICATION или UPDATE MODEL. 3) С помощью sysout, убедитесь, что страница ошибки настроена - с использованием контекста сервлета (я не пробовал это, но это должно быть возможно)
Даже мне любопытно, в чем может быть проблема!!!