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&nbsp;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!