JSF 2.0 Custom Exception Handler throws IllegalSta

2019-05-16 23:15发布

问题:

I wrote a custom exception handler for JSF to log the exception and navigate to an error page to be shown to the user. Unfortunately, I get a IllegalStateException "Cannot call sendRedirect() after the response has been commited" when calling "handleNavigation" inside the Handler. Any ideas, what I'm doing wrong?

My Handler:

public class MyExceptionHandler extends ExceptionHandlerWrapper {
    private Logger           log            = ...;
    public  ExceptionHandler wrappedHandler = null;

    public MyExceptionHandler (ExceptionHandler wrappedHandler) {
        this.wrappedHandler = wrappedHandler;
    }

    @Override
    public ExceptionHandler getWrapped() {
        return wrappedHandler;
    }   

    @Override
    public void handle() throws FacesException {
        Iterator<ExceptionQueuedEvent> iter         = null;
        ExceptionQueuedEvent           event        = null;
        ExceptionQueuedEventContext    eventContext = null;
        FacesContext                   facesContext = null;       

        iter = getUnhandledExceptionQueuedEvents().iterator();      

        while (iter.hasNext()) {
            try {
                event        = iter.next();
                eventContext = (ExceptionQueuedEventContext) event.getSource(); 

                log.error("JSF Exception aufgetreten", eventContext.getException());          

                facesContext = FacesContext.getCurrentInstance();                    

                // !!!!!!!! Exception occurs here !!!!!!!!!!!
                facesContext
                   .getApplication()
                   .getNavigationHandler()
                   .handleNavigation(facesContext, null, "error");

                facesContext.renderResponse();        

            } catch (RuntimeException ex) {
                throw ex; // just to set break point
            } finally {
                iter.remove();
            }
          }
          getWrapped().handle();       
    }
}

Navigation Definition in faces-config.xml

<navigation-rule>
    <from-view-id>*</from-view-id>
    <navigation-case>
        <from-outcome>error</from-outcome>
        <to-view-id>/faces/error.xhtml</to-view-id>
        <redirect/>
    </navigation-case>
</navigation-rule>

回答1:

The exception being handled is apparently been thrown during render response phase, at that point when the HTTP response is already been committed. A committed response means that the first part of the HTTP response, including the headers, is already been sent to the client side. This is a point of no return. You can't take the already sent bytes back from the client.

A response will usually be auto-committed when the written content exceeds the buffer size which defaults usually to ~2KB, depending on servletcontainer and Facelets configuration. The average HTML <head> occuppies already 1~2KB. So the change is big that the response is already committed before JSF starts to render the <body> or only a small part of it.

In your particular case there's however one more cause: when you receive more than one unhandled exception, then you will also run into trouble, because you're not aborting the while loop after having navigated, but continuing to handling the next exception. You can't return multiple responses (error pages) to a single request. You should either collect all exceptions and navigate only once, or to abort the loop after the first one.

In any case, handling exceptions which are thrown during render response is only possible as long as the response is not (auto)-committed. You can try several approaches to prevent the response from being auto committed too soon:

  1. Set the Facelets buffer size to the size of your largest HTML response. E.g. 64KB:

    <context-param>
        <param-name>javax.faces.FACELETS_BUFFER_SIZE</param-name>
        <param-value>65535</param-value><!-- 64KB -->
    </context-param>
    
  2. Perform the exception-sensitive business job before rendering the view (i.e. don't do it in (post)constructor of a bean which is constructed during a GET request):

    <f:event type="preRenderView" listener="#{bean.init}" />
    

As to handling with exceptions in JSF in general, you may find OmniFaces FullAjaxExceptionHandler helpful. You can find its source code here.

See also:

  • Exception thrown in @PostConstruct causes IllegalStateException in JSF 2.1
  • Error page defined in web.xml is embedded in partially rendered JSF page