Glassfishv3でJSF2.0アプリケーションを開発していて、ViewExpiredExceptionを処理しようとしています。しかし、私が何をするにしても、自分のエラーページではなく、常にGlassfishエラーレポートを受け取ります。
VEEの発生をシミュレートするために、VEEを起動する次の関数をバッキングBeanに挿入しました。この関数は、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ステータス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
次に試したのは、 JavaServerFaces 2.0-完全なリファレンスで説明されているように、ExceptionHandlerFactoryとCustomExceptionHandlerを作成することでした。そこで、faces-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();
}
}
しかし、それでもエラーページにリダイレクトされません-上記と同じHTTP500エラーが発生します。何が間違っているのですか、例外が正しく処理されないという私の実装に何が欠けている可能性がありますか?どんな助けでも大歓迎です!
編集
はい、正直です。実際、私のコードは実際にはScalaで書かれていますが、それは長い話です。いつもJavaの問題だと思っていました。この場合の本当のエラーは私自身の愚かさでした。私の(Scala)コードのCustomExceptionHandlerで、「i.remove();」の行を追加するのを忘れました。そのため、ViewExpiredExceptionは、処理後もUnhandledExceptionsQueueにとどまり、「バブルアップ」しました。そして、それがバブルアップすると、ServletExceptionになります。
お二人ともご迷惑をおかけして申し訳ございません。