JSF 2 Глобальная обработка исключений, переход к странице ошибок не происходит
Я занимаюсь разработкой веб-приложения на основе JSF 2.0. Я пытаюсь реализовать глобальный обработчик исключений, который будет перенаправлять пользователя на общую страницу ошибок при возникновении любого исключения (например, NullPointerException,ServletException,ViewExpiredException и т. Д.)
Всякий раз, когда в моем приложении возникает NPE, достигается точка останова My customnavhandler, и выполняется код NavigationHandler, но каким-то образом перенаправление на страницу с ошибкой не происходит, запрашиваемая страница остается частично обработанной. Есть идеи, что здесь может быть не так? Одна информация состоит в том, что я специально выбрасываю NPE на запрошенную страницу (которая была частично визуализирована после NPE)
Моя запись Face-config.xml
<factory>
<exception-handler-factory>
com.common.exceptions.CustomExceptionHandlerFactory
</exception-handler-factory>
</factory>
Мой CustomNavHandler
public class CustomExceptionHandler extends ExceptionHandlerWrapper {
private static final Logger logger = Logger.getLogger("com.gbdreports.common.exception.CustomExceptionHandler");
private final ExceptionHandler wrapped;
public CustomExceptionHandler(ExceptionHandler wrapped) {
this.wrapped = wrapped;
}
@Override
public ExceptionHandler getWrapped() {
return this.wrapped;
}
public void handle() throws FacesException {
final Iterator<ExceptionQueuedEvent> i = getUnhandledExceptionQueuedEvents().iterator();
while (i.hasNext()) {
ExceptionQueuedEvent event = i.next();
ExceptionQueuedEventContext context =
(ExceptionQueuedEventContext) event.getSource();
// get the exception from context
Throwable t = context.getException();
final FacesContext fc = FacesContext.getCurrentInstance();
final ExternalContext externalContext = fc.getExternalContext();
final Map<String, Object> requestMap = fc.getExternalContext().getRequestMap();
final ConfigurableNavigationHandler nav = (ConfigurableNavigationHandler) fc.getApplication().getNavigationHandler();
//here you do what ever you want with exception
try {
//log error ?
logger.error("Severe Exception Occured");
//log.log(Level.SEVERE, "Critical Exception!", t);
//redirect error page
requestMap.put("exceptionMessage", t.getMessage());
nav.performNavigation("/TestPRoject/error.xhtml");
fc.renderResponse();
// remove the comment below if you want to report the error in a jsf error message
//JsfUtil.addErrorMessage(t.getMessage());
}
finally {
//remove it from queue
i.remove(); }
}
//parent hanle
getWrapped().handle();
}
}
Моя кастомная фабрика
public class CustomExceptionHandlerFactory extends ExceptionHandlerFactory {
private ExceptionHandlerFactory parent;
public CustomExceptionHandlerFactory(ExceptionHandlerFactory parent) {
this.parent = parent;
}
@Override
public ExceptionHandler getExceptionHandler() {
return new CustomExceptionHandler (parent.getExceptionHandler());
}
}
2 ответа
Скорее всего, потому что текущий запрос является ajax (асинхронным) запросом. Ваш обработчик исключений предназначен для регулярных (синхронных) запросов.
Правильный способ изменить представление в случае исключения ajax заключается в следующем:
String viewId = "/error.xhtml";
ViewHandler viewHandler = context.getApplication().getViewHandler();
context.setViewRoot(viewHandler.createView(context, viewId));
context.getPartialViewContext().setRenderAll(true);
context.renderResponse();
Это, однако, несколько наивно. Это не будет работать, если исключение ajax было сгенерировано во время рендеринга ответа ajax.
Я бы посоветовал не изобретать велосипед. Утилита JSF OmniFaces имеет законченное рабочее решение во FullAjaxExceptionHandler
, Вы можете найти полный исходный код здесь и пример витрины здесь. Он использует стандартный сервлет API <error-page>
декларации в web.xml
, Таким образом, страницы ошибок также можно повторно использовать для синхронных запросов, с небольшой помощью FacesExceptionFilter
также предоставлено OmniFaces.
Смотрите также:
Единый способ обработки исключений как ajax, так и non ajax запросов может быть упрощен для вашего кода. Вместо
requestMap.put("exceptionMessage", t.getMessage()); nav.performNavigation("/TestPRoject/error.xhtml"); fc.renderResponse();
достаточно использовать:
fc.getExternalContext().redirect("/TestPRoject/error.xhtml");