8

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になります。

お二人ともご迷惑をおかけして申し訳ございません。

4

2 に答える 2

14

このテストケースは偽物です。これViewExpiredExceptionは通常、ビューの復元中にのみスローされ(セッションで欠落しているため)、応答のレンダリング中やBeanのインスタンス化中にはスローされません。あなたの場合、この例外はBeanのインスタンス化中にスローされ、この例外は。でラップされますServletException

実数 は通常、ViewExpiredExceptionHTTPセッションの有効期限が切れている間にサーバーにHTTPPOSTリクエストを送信した場合にのみスローされます。したがって、これを確実に再現するには、基本的に2つの方法があります。

  1. WebブラウザでPOSTフォーム(デフォルトではすでにPOST)を使用してJSFページを開き、h:formサーバーをシャットダウンしてその作業ディレクトリをクリーンアップします(ほとんどのサーバーはシャットダウン時に開いているセッションをディスクにシリアル化し、起動時にシリアル化を解除するため、重要です)。サーバーにアクセスし、すでに開いているフォームを送信します。AViewExpiredExceptionがスローされます。

  2. を分に設定<session-timeout>し、POSTフォームでJSFページを開いてから1分以上フォームを送信します。これもスローします。web.xml1ViewExpiredException

于 2010-06-05T22:29:55.337 に答える
2

私は専門家ではありません。これらは単なる推測または提案です。

1)標準のHTMLページにリダイレクトして、それが機能するかどうかを確認します2)これに基づいて、最初のアプローチ自体で機能するはずです。JSFPhaseListenerを記述して、RESTOREVIEWフェーズで同じ例外をスローします。 INVOKEAPPLICATIONまたはUPDATEMODELフェーズのいずれかでスローします。3)sysoutによって、エラーページが構成されていることを確認します-サーブレットコンテキストを使用します(これは試していませんが、可能であるはずです)

何が問題なのか気になります!!!

于 2010-06-05T14:51:50.990 に答える