JSF: не удается поймать исключение ViewExpiredException - PullRequest
8 голосов
/ 05 июня 2010

Я разрабатываю приложение 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, как и выше. Что я делаю не так, что может быть упущено в моей реализации, что исключение не обрабатывается правильно? Любой помощь высоко ценится!

EDIT

Хорошо, я честен. На самом деле мой код написан на Scala, но это длинная история. я думал, что это проблема Java все время. НАСТОЯЩЕЙ ошибкой в ​​этом случае была моя собственная глупость. В моем (Scala) коде, в CustomExceptionHandler, я забыл добавить строку с помощью «i.remove ();» Таким образом, ViewExpiredException остался в UnhandledExceptionsQueue после его обработки, и он «всплыл». И когда он вспыхивает, он становится ServletException.

Мне очень жаль, что я вас запутал!

Ответы [ 2 ]

14 голосов
/ 06 июня 2010

Этот тестовый пример является поддельным. ViewExpiredException обычно генерируется только во время восстановления представления (потому что оно отсутствует в сеансе), а не во время рендеринга ответа или создания экземпляра компонента. В вашем случае это исключение выдается при создании экземпляра bean-компонента, и это исключение заключено в ServletException.

real ViewExpiredException обычно генерируется только при отправке запроса HTTP POST на сервер, когда сеанс HTTP истек. Таким образом, существует два способа надежного воспроизведения:

  1. Откройте страницу JSF с формой POST (по умолчанию h:form уже POST) в веб-браузере, завершите работу сервера и очистите его рабочий каталог (важно, потому что большинство серверов сериализует открытые сессии на диск при завершении работы и десериализации их при запуске), перезагрузите сервер и отправьте уже открытую форму. A ViewExpiredException будет брошено.

  2. Установите <session-timeout> в web.xml на 1 минуту и ​​отправьте форму в течение 1 минуты после открытия страницы JSF с формой POST. Это также выбросит ViewExpiredException.

2 голосов
/ 05 июня 2010

Я не эксперт. Это просто дикие догадки или предположения.

1) Попробуйте перенаправить на стандартную страницу HTML, чтобы увидеть, работает ли это 2) Исходя из этого , он должен работать с самим первым подходом, попробуйте написать JSF PhaseListener и выдать то же исключение в фазе RESTORE VIEW. Прямо сейчас вы добавляете либо в INVOKE APPLICATION, либо в UPDATE. МОДЕЛЬ Фаза. 3) С помощью sysout, убедитесь, что страница ошибки настроена - с использованием контекста сервлета (я не пробовал это, но это должно быть возможно)

Даже мне любопытно, в чем может быть проблема !!!

...