474634 - AsyncListener.onError() handling.
Handle errors thrown from dispatch when async is started with onError
This commit is contained in:
parent
0f6a83f710
commit
f21ea15725
|
@ -29,6 +29,7 @@ import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import javax.servlet.DispatcherType;
|
import javax.servlet.DispatcherType;
|
||||||
import javax.servlet.RequestDispatcher;
|
import javax.servlet.RequestDispatcher;
|
||||||
|
import javax.servlet.UnavailableException;
|
||||||
import javax.servlet.http.HttpServletRequest;
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
|
||||||
import org.eclipse.jetty.http.BadMessageException;
|
import org.eclipse.jetty.http.BadMessageException;
|
||||||
|
@ -277,7 +278,6 @@ public class HttpChannel implements Runnable, HttpOutput.Interceptor
|
||||||
// already happened when unhandle is called.
|
// already happened when unhandle is called.
|
||||||
loop: while (!getServer().isStopped())
|
loop: while (!getServer().isStopped())
|
||||||
{
|
{
|
||||||
boolean error=false;
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (LOG.isDebugEnabled())
|
if (LOG.isDebugEnabled())
|
||||||
|
@ -323,14 +323,14 @@ public class HttpChannel implements Runnable, HttpOutput.Interceptor
|
||||||
_request.setDispatcherType(DispatcherType.ERROR);
|
_request.setDispatcherType(DispatcherType.ERROR);
|
||||||
|
|
||||||
Throwable ex = _state.getAsyncContextEvent().getThrowable();
|
Throwable ex = _state.getAsyncContextEvent().getThrowable();
|
||||||
String reason;
|
String reason=null;
|
||||||
if (ex == null || ex instanceof TimeoutException)
|
if (ex == null || ex instanceof TimeoutException)
|
||||||
{
|
{
|
||||||
reason = "Async Timeout";
|
reason = "Async Timeout";
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
reason = "Async Exception";
|
reason = HttpStatus.Code.INTERNAL_SERVER_ERROR.getMessage();
|
||||||
_request.setAttribute(RequestDispatcher.ERROR_EXCEPTION, ex);
|
_request.setAttribute(RequestDispatcher.ERROR_EXCEPTION, ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -338,7 +338,7 @@ public class HttpChannel implements Runnable, HttpOutput.Interceptor
|
||||||
_request.setAttribute(RequestDispatcher.ERROR_MESSAGE, reason);
|
_request.setAttribute(RequestDispatcher.ERROR_MESSAGE, reason);
|
||||||
_request.setAttribute(RequestDispatcher.ERROR_REQUEST_URI, _request.getRequestURI());
|
_request.setAttribute(RequestDispatcher.ERROR_REQUEST_URI, _request.getRequestURI());
|
||||||
|
|
||||||
_response.setStatusWithReason(500, reason);
|
_response.setStatusWithReason(HttpStatus.INTERNAL_SERVER_ERROR_500, reason);
|
||||||
|
|
||||||
ErrorHandler eh = ErrorHandler.getErrorHandler(getServer(), _state.getContextHandler());
|
ErrorHandler eh = ErrorHandler.getErrorHandler(getServer(), _state.getContextHandler());
|
||||||
if (eh instanceof ErrorHandler.ErrorPageMapper)
|
if (eh instanceof ErrorHandler.ErrorPageMapper)
|
||||||
|
@ -408,21 +408,15 @@ public class HttpChannel implements Runnable, HttpOutput.Interceptor
|
||||||
}
|
}
|
||||||
catch (EofException|QuietServletException|BadMessageException e)
|
catch (EofException|QuietServletException|BadMessageException e)
|
||||||
{
|
{
|
||||||
error=true;
|
|
||||||
LOG.debug(e);
|
LOG.debug(e);
|
||||||
_state.error(e);
|
|
||||||
_request.setHandled(true);
|
|
||||||
handleException(e);
|
handleException(e);
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
error=true;
|
|
||||||
if (_connector.isStarted())
|
if (_connector.isStarted())
|
||||||
LOG.warn(String.valueOf(_request.getHttpURI()), e);
|
LOG.warn(String.valueOf(_request.getHttpURI()), e);
|
||||||
else
|
else
|
||||||
LOG.debug(String.valueOf(_request.getHttpURI()), e);
|
LOG.debug(String.valueOf(_request.getHttpURI()), e);
|
||||||
_state.error(e);
|
|
||||||
_request.setHandled(true);
|
|
||||||
handleException(e);
|
handleException(e);
|
||||||
}
|
}
|
||||||
catch (Throwable e)
|
catch (Throwable e)
|
||||||
|
@ -431,20 +425,15 @@ public class HttpChannel implements Runnable, HttpOutput.Interceptor
|
||||||
LOG.ignore(e);
|
LOG.ignore(e);
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
error=true;
|
|
||||||
if (_connector.isStarted())
|
if (_connector.isStarted())
|
||||||
LOG.warn(String.valueOf(_request.getHttpURI()), e);
|
LOG.warn(String.valueOf(_request.getHttpURI()), e);
|
||||||
else
|
else
|
||||||
LOG.debug(String.valueOf(_request.getHttpURI()), e);
|
LOG.debug(String.valueOf(_request.getHttpURI()), e);
|
||||||
LOG.warn(String.valueOf(_request.getHttpURI()), e);
|
LOG.warn(String.valueOf(_request.getHttpURI()), e);
|
||||||
_state.error(e);
|
|
||||||
_request.setHandled(true);
|
|
||||||
handleException(e);
|
handleException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (error && _state.isAsyncStarted())
|
|
||||||
_state.errorComplete();
|
|
||||||
action = _state.unhandle();
|
action = _state.unhandle();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -468,19 +457,30 @@ public class HttpChannel implements Runnable, HttpOutput.Interceptor
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
if (_state.isAsyncStarted())
|
||||||
|
{
|
||||||
|
// Handle exception via AsyncListener onError
|
||||||
|
Throwable root = _state.getAsyncContextEvent().getThrowable();
|
||||||
|
if (root==null)
|
||||||
|
{
|
||||||
|
_state.error(x);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We've already processed an error before!
|
||||||
|
root.addSuppressed(x);
|
||||||
|
LOG.warn("Error while handling async error: ",root);
|
||||||
|
abort(x);
|
||||||
|
_state.errorComplete();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle error normally
|
||||||
|
_request.setHandled(true);
|
||||||
_request.setAttribute(RequestDispatcher.ERROR_EXCEPTION,x);
|
_request.setAttribute(RequestDispatcher.ERROR_EXCEPTION,x);
|
||||||
_request.setAttribute(RequestDispatcher.ERROR_EXCEPTION_TYPE,x.getClass());
|
_request.setAttribute(RequestDispatcher.ERROR_EXCEPTION_TYPE,x.getClass());
|
||||||
if (_state.isSuspended())
|
|
||||||
{
|
if (isCommitted())
|
||||||
HttpFields fields = new HttpFields();
|
|
||||||
fields.add(HttpHeader.CONNECTION,HttpHeaderValue.CLOSE);
|
|
||||||
MetaData.Response info = new MetaData.Response(_request.getHttpVersion(), HttpStatus.INTERNAL_SERVER_ERROR_500, null, fields, 0);
|
|
||||||
boolean committed = sendResponse(info,null, true);
|
|
||||||
if (!committed)
|
|
||||||
LOG.warn("Could not send response error 500: "+x);
|
|
||||||
_request.getAsyncContext().complete();
|
|
||||||
}
|
|
||||||
else if (isCommitted())
|
|
||||||
{
|
{
|
||||||
abort(x);
|
abort(x);
|
||||||
if (!(x instanceof EofException))
|
if (!(x instanceof EofException))
|
||||||
|
@ -495,8 +495,15 @@ public class HttpChannel implements Runnable, HttpOutput.Interceptor
|
||||||
BadMessageException bme = (BadMessageException)x;
|
BadMessageException bme = (BadMessageException)x;
|
||||||
_response.sendError(bme.getCode(), bme.getReason());
|
_response.sendError(bme.getCode(), bme.getReason());
|
||||||
}
|
}
|
||||||
|
else if (x instanceof UnavailableException)
|
||||||
|
{
|
||||||
|
if (((UnavailableException)x).isPermanent())
|
||||||
|
_response.sendError(HttpStatus.NOT_FOUND_404);
|
||||||
|
else
|
||||||
|
_response.sendError(HttpStatus.SERVICE_UNAVAILABLE_503);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
_response.sendError(500, x.getClass().toString());
|
_response.sendError(HttpStatus.INTERNAL_SERVER_ERROR_500, x.getClass().toString());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (IOException e)
|
catch (IOException e)
|
||||||
|
|
|
@ -296,6 +296,7 @@ public class HttpChannelState
|
||||||
{
|
{
|
||||||
if (_event!=null)
|
if (_event!=null)
|
||||||
_event.addThrowable(th);
|
_event.addThrowable(th);
|
||||||
|
_async=Async.ERRORING;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -598,7 +598,9 @@ public class ServletHandler extends ScopedHandler
|
||||||
}
|
}
|
||||||
catch(Exception e)
|
catch(Exception e)
|
||||||
{
|
{
|
||||||
if (!(DispatcherType.REQUEST.equals(type) || DispatcherType.ASYNC.equals(type)))
|
//TODO, can we let all error handling fall through to HttpChannel?
|
||||||
|
|
||||||
|
if (baseRequest.isAsyncStarted() || !(DispatcherType.REQUEST.equals(type) || DispatcherType.ASYNC.equals(type)))
|
||||||
{
|
{
|
||||||
if (e instanceof IOException)
|
if (e instanceof IOException)
|
||||||
throw (IOException)e;
|
throw (IOException)e;
|
||||||
|
|
|
@ -173,6 +173,7 @@ public class AsyncContextTest
|
||||||
"Connection: close\r\n" +
|
"Connection: close\r\n" +
|
||||||
"\r\n";
|
"\r\n";
|
||||||
String responseString = _connector.getResponses(request);
|
String responseString = _connector.getResponses(request);
|
||||||
|
System.err.println(responseString);
|
||||||
|
|
||||||
BufferedReader br = new BufferedReader(new StringReader(responseString));
|
BufferedReader br = new BufferedReader(new StringReader(responseString));
|
||||||
|
|
||||||
|
|
|
@ -81,17 +81,7 @@ public class AsyncServletTest
|
||||||
protected List<String> _log;
|
protected List<String> _log;
|
||||||
protected int _expectedLogs;
|
protected int _expectedLogs;
|
||||||
protected String _expectedCode;
|
protected String _expectedCode;
|
||||||
protected static List<String> __history=new CopyOnWriteArrayList()
|
protected static List<String> __history=new CopyOnWriteArrayList<>();
|
||||||
{
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean add(Object e)
|
|
||||||
{
|
|
||||||
System.err.println("H: "+e);
|
|
||||||
return super.add(e);
|
|
||||||
}
|
|
||||||
|
|
||||||
};
|
|
||||||
protected static CountDownLatch __latch;
|
protected static CountDownLatch __latch;
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
|
@ -213,7 +203,7 @@ public class AsyncServletTest
|
||||||
{
|
{
|
||||||
_expectedCode="500 ";
|
_expectedCode="500 ";
|
||||||
String response=process("start=200&timeout=error",null);
|
String response=process("start=200&timeout=error",null);
|
||||||
assertThat(response,startsWith("HTTP/1.1 500 Async Exception"));
|
assertThat(response,startsWith("HTTP/1.1 500 Server Error"));
|
||||||
assertThat(__history,contains(
|
assertThat(__history,contains(
|
||||||
"REQUEST /ctx/path/info",
|
"REQUEST /ctx/path/info",
|
||||||
"initial",
|
"initial",
|
||||||
|
@ -323,12 +313,11 @@ public class AsyncServletTest
|
||||||
"REQUEST /ctx/path/info",
|
"REQUEST /ctx/path/info",
|
||||||
"initial",
|
"initial",
|
||||||
"start",
|
"start",
|
||||||
/* TODO should there be an onError call?
|
|
||||||
"onError",
|
"onError",
|
||||||
"history: onComplete\r\n"
|
"ERROR /ctx/path/info",
|
||||||
*/""
|
"!initial",
|
||||||
));
|
"onComplete"));
|
||||||
assertContains("HTTP ERROR: 500",response);
|
assertContains("ERROR DISPATCH: /ctx/path/info",response);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
Loading…
Reference in New Issue