JSF: Não é possível capturar ViewExpiredException
Estou desenvolvendo um aplicativo JSF 2.0 no Glassfish v3 e estou tentando lidar com o ViewExpiredException. Mas o que quer que eu faça, sempre recebo um relatório de erro do Glassfish em vez da minha própria página de erro.
Para simular a ocorrência do VEE, inseri a seguinte função no meu backing bean, que aciona o VEE. Estou acionando essa função da minha página JSF através de um commandLink. O código:
@Named
public class PersonHome {
(...)
public void throwVEE() {
throw new ViewExpiredException();
}
}
No começo, tentei adicionando uma página de erro ao meu web.xml:
<error-page>
<exception-type>javax.faces.application.ViewExpiredException</exception-type>
<location>/error.xhtml</location>
</error-page>
Mas isso não funciona, não sou redirecionado para o erro, mas é exibida a página de erro do Glassfish, que mostra uma página HTTP Status 500 com o seguinte conteúdo:
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
A próxima coisa que tentei foi escrever ExceptionHandlerFactory e um CustomExceptionHandler, conforme descrito emJavaServerFaces 2.0 - a referência completa. Então, eu inseri a seguinte tag no faces-config.xml:
<factory>
<exception-handler-factory>
exceptions.ExceptionHandlerFactory
</exception-handler-factory>
</factory>
E adicionou estas classes: A fábrica:
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;
}
}
O manipulador de exceção personalizado:
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();
}
}
Mas ainda não sou redirecionado para a minha página de erro - estou recebendo o mesmo erro HTTP 500 como acima. O que estou fazendo de errado, o que pode estar faltando na minha implementação que a exceção não é tratada corretamente?Qualquer ajuda muito apreciada!
EDITAR
Ok, eu sou honesto. De fato, meu código está realmente escrito em Scala, mas é uma longa história. Eu pensei que era um problema de Java o tempo todo. O erro REAL neste caso foi minha própria estupidez. No meu código (Scala), no CustomExceptionHandler, esqueci de adicionar a linha com o "i.remove ();" Portanto, o ViewExpiredException permaneceu no UnhandledExceptionsQueue após manipulá-lo e "borbulhou". E quando borbulha, torna-se uma ServletException.
Sinto muito por confundir vocês dois!