9

JSF 2.0 ベースの Web アプリケーションを開発しています。例外が発生するたびにユーザーを一般的なエラーページにリダイレクトするグローバル例外ハンドラーを実装しようとしています (NullPointerException、ServletException、ViewExpiredException など)。

アプリで NPE が発生するたびに、customnavhandler ブレークポイントがヒットし、NavigationHandler コードが実行されますが、何らかの理由でエラー ページへのリダイレクトが行われず、要求されたページが部分的にレンダリングされたままになります。ここで何が間違っているのでしょうか?1 つの情報は、要求されたページ (NPE の後に部分的にレンダリングされたもの) で意図的に NPE をスローしていることです。

私の faces-config.xml エントリ

<factory>
  <exception-handler-factory>
    com.common.exceptions.CustomExceptionHandlerFactory
  </exception-handler-factory>
</factory>

私の CustomNavHandler

public class CustomExceptionHandler extends ExceptionHandlerWrapper {

private static final Logger logger = Logger.getLogger("com.gbdreports.common.exception.CustomExceptionHandler");
private final ExceptionHandler wrapped;

public CustomExceptionHandler(ExceptionHandler wrapped) {
    this.wrapped = wrapped;
}

@Override
public ExceptionHandler getWrapped() {
    return this.wrapped;

}
public void handle() throws FacesException {
    final Iterator<ExceptionQueuedEvent> i = getUnhandledExceptionQueuedEvents().iterator();         

    while (i.hasNext()) {             
        ExceptionQueuedEvent event = i.next();             
        ExceptionQueuedEventContext context =                    
                (ExceptionQueuedEventContext) event.getSource();               
        // get the exception from context             
        Throwable t = context.getException();               
        final FacesContext fc = FacesContext.getCurrentInstance();   
        final ExternalContext externalContext = fc.getExternalContext();
        final Map<String, Object> requestMap = fc.getExternalContext().getRequestMap();            
        final ConfigurableNavigationHandler nav = (ConfigurableNavigationHandler) fc.getApplication().getNavigationHandler();               
        //here you do what ever you want with exception             
        try {                   
            //log error ?      
            logger.error("Severe Exception Occured");
            //log.log(Level.SEVERE, "Critical Exception!", t);                   
            //redirect error page                 
            requestMap.put("exceptionMessage", t.getMessage());                 
            nav.performNavigation("/TestPRoject/error.xhtml");                 
            fc.renderResponse();                   
            // remove the comment below if you want to report the error in a jsf error message                 
            //JsfUtil.addErrorMessage(t.getMessage());               
            } 
        finally {                 
            //remove it from queue                 
            i.remove();             }         
        }         
    //parent hanle         
    getWrapped().handle(); 
        }

}

私の customNavhandler ファクトリー

public class CustomExceptionHandlerFactory extends ExceptionHandlerFactory {


 private ExceptionHandlerFactory parent;

  public CustomExceptionHandlerFactory(ExceptionHandlerFactory parent) {
    this.parent = parent;
  }

  @Override
  public ExceptionHandler getExceptionHandler() {
      return new CustomExceptionHandler (parent.getExceptionHandler());

  }

}
4

2 に答える 2

16

現在のリクエストが ajax (非同期) リクエストである可能性が最も高いです。あなたが持っている例外ハンドラは、通常の(同期)リクエスト用に設計されています。

ajax 例外が発生した場合にビューを変更する適切な方法は次のとおりです。

String viewId = "/error.xhtml";
ViewHandler viewHandler = context.getApplication().getViewHandler();
context.setViewRoot(viewHandler.createView(context, viewId));
context.getPartialViewContext().setRenderAll(true);
context.renderResponse();

ただし、これはやや単純です。ajax 応答のレンダリング中に ajax 例外がスローされた場合、これは機能しません。

車輪を再発明しないことをお勧めします。JSF ユーティリティ ライブラリOmniFacesには、FullAjaxExceptionHandler. 完全なソース コードはここにあり、ショーケースの例はここにあります。の標準サーブレット API<error-page>宣言を使用しweb.xmlます。FacesExceptionFilterこのようにして、同じく OmniFaces によって提供されるの助けを借りて、エラー ページを同期要求に再利用することもできます。

以下も参照してください。

于 2013-08-23T20:32:39.077 に答える
0

ajax リクエストと非 ajax リクエストの両方の例外を処理する統一された方法により、コードを簡素化できます。それ以外の

requestMap.put("exceptionMessage", t.getMessage());                 
nav.performNavigation("/TestPRoject/error.xhtml");                 
fc.renderResponse();

使用するのに十分です:

fc.getExternalContext().redirect("/TestPRoject/error.xhtml");
于 2015-03-16T17:16:28.603 に答える