Merge branch 'master' into servlet-3.1-api

This commit is contained in:
Greg Wilkins 2013-04-22 12:10:13 +10:00
commit a275c8fb37
7 changed files with 415 additions and 249 deletions

View File

@ -0,0 +1,153 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.server;
import javax.servlet.AsyncContext;
import javax.servlet.AsyncEvent;
import javax.servlet.RequestDispatcher;
import javax.servlet.ServletContext;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import org.eclipse.jetty.server.handler.ContextHandler.Context;
import org.eclipse.jetty.util.thread.Scheduler;
public class AsyncContextEvent extends AsyncEvent
{
final private Context _context;
final private AsyncContextState _asyncContext;
volatile HttpChannelState _state;
private ServletContext _dispatchContext;
private String _pathInContext;
private Scheduler.Task _timeoutTask;
private Throwable _throwable;
public AsyncContextEvent(Context context,AsyncContextState asyncContext, HttpChannelState state, Request baseRequest, ServletRequest request, ServletResponse response)
{
super(null,request,response,null);
_context=context;
_asyncContext=asyncContext;
_state=state;
// If we haven't been async dispatched before
if (baseRequest.getAttribute(AsyncContext.ASYNC_REQUEST_URI)==null)
{
// We are setting these attributes during startAsync, when the spec implies that
// they are only available after a call to AsyncContext.dispatch(...);
// have we been forwarded before?
String uri=(String)baseRequest.getAttribute(RequestDispatcher.FORWARD_REQUEST_URI);
if (uri!=null)
{
baseRequest.setAttribute(AsyncContext.ASYNC_REQUEST_URI,uri);
baseRequest.setAttribute(AsyncContext.ASYNC_CONTEXT_PATH,baseRequest.getAttribute(RequestDispatcher.FORWARD_CONTEXT_PATH));
baseRequest.setAttribute(AsyncContext.ASYNC_SERVLET_PATH,baseRequest.getAttribute(RequestDispatcher.FORWARD_SERVLET_PATH));
baseRequest.setAttribute(AsyncContext.ASYNC_PATH_INFO,baseRequest.getAttribute(RequestDispatcher.FORWARD_PATH_INFO));
baseRequest.setAttribute(AsyncContext.ASYNC_QUERY_STRING,baseRequest.getAttribute(RequestDispatcher.FORWARD_QUERY_STRING));
}
else
{
baseRequest.setAttribute(AsyncContext.ASYNC_REQUEST_URI,baseRequest.getRequestURI());
baseRequest.setAttribute(AsyncContext.ASYNC_CONTEXT_PATH,baseRequest.getContextPath());
baseRequest.setAttribute(AsyncContext.ASYNC_SERVLET_PATH,baseRequest.getServletPath());
baseRequest.setAttribute(AsyncContext.ASYNC_PATH_INFO,baseRequest.getPathInfo());
baseRequest.setAttribute(AsyncContext.ASYNC_QUERY_STRING,baseRequest.getQueryString());
}
}
}
public ServletContext getSuspendedContext()
{
return _context;
}
public Context getContext()
{
return _context;
}
public ServletContext getDispatchContext()
{
return _dispatchContext;
}
public ServletContext getServletContext()
{
return _dispatchContext==null?_context:_dispatchContext;
}
/**
* @return The path in the context
*/
public String getPath()
{
return _pathInContext;
}
public void setTimeoutTask(Scheduler.Task task)
{
_timeoutTask = task;
}
public void cancelTimeoutTask()
{
Scheduler.Task task=_timeoutTask;
_timeoutTask=null;
if (task!=null)
task.cancel();
}
@Override
public AsyncContext getAsyncContext()
{
return _asyncContext;
}
@Override
public Throwable getThrowable()
{
return _throwable;
}
public void setThrowable(Throwable throwable)
{
_throwable=throwable;
}
public void setDispatchTarget(ServletContext context, String path)
{
if (context!=null)
_dispatchContext=context;
if (path!=null)
_pathInContext=path;
}
public void completed()
{
_asyncContext.reset();
}
public HttpChannelState getHttpChannelState()
{
return _state;
}
}

View File

@ -0,0 +1,180 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.server;
import java.io.IOException;
import javax.servlet.AsyncContext;
import javax.servlet.AsyncEvent;
import javax.servlet.AsyncListener;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
public class AsyncContextState implements AsyncContext
{
volatile HttpChannelState _state;
public AsyncContextState(HttpChannelState state)
{
_state=state;
}
private HttpChannelState state()
{
HttpChannelState state=_state;
if (state==null)
throw new IllegalStateException("AsyncContext completed");
return state;
}
@Override
public void addListener(final AsyncListener listener, final ServletRequest request, final ServletResponse response)
{
AsyncListener wrap = new AsyncListener()
{
@Override
public void onTimeout(AsyncEvent event) throws IOException
{
listener.onTimeout(new AsyncEvent(event.getAsyncContext(),request,response,event.getThrowable()));
}
@Override
public void onStartAsync(AsyncEvent event) throws IOException
{
listener.onStartAsync(new AsyncEvent(event.getAsyncContext(),request,response,event.getThrowable()));
}
@Override
public void onError(AsyncEvent event) throws IOException
{
listener.onComplete(new AsyncEvent(event.getAsyncContext(),request,response,event.getThrowable()));
}
@Override
public void onComplete(AsyncEvent event) throws IOException
{
listener.onComplete(new AsyncEvent(event.getAsyncContext(),request,response,event.getThrowable()));
}
};
state().addListener(wrap);
}
@Override
public void addListener(AsyncListener listener)
{
state().addListener(listener);
}
@Override
public void complete()
{
state().complete();
}
@Override
public <T extends AsyncListener> T createListener(Class<T> clazz) throws ServletException
{
try
{
return clazz.newInstance();
}
catch(Exception e)
{
throw new ServletException(e);
}
}
@Override
public void dispatch()
{
state().dispatch(null,null);
}
@Override
public void dispatch(String path)
{
state().dispatch(null,path);
}
@Override
public void dispatch(ServletContext context, String path)
{
state().dispatch(context,path);
}
@Override
public ServletRequest getRequest()
{
return state().getAsyncContextEvent().getSuppliedRequest();
}
@Override
public ServletResponse getResponse()
{
return state().getAsyncContextEvent().getSuppliedResponse();
}
@Override
public long getTimeout()
{
return state().getTimeout();
}
@Override
public boolean hasOriginalRequestAndResponse()
{
HttpChannel<?> channel=state().getHttpChannel();
return channel.getRequest()==getRequest() && channel.getResponse()==getResponse();
}
@Override
public void setTimeout(long arg0)
{
state().setTimeout(arg0);
}
@Override
public void start(final Runnable task)
{
state().getHttpChannel().execute(new Runnable()
{
@Override
public void run()
{
state().getAsyncContextEvent().getContext().getContextHandler().handle(task);
}
});
}
public void reset()
{
_state=null;
}
public HttpChannelState getHttpChannelState()
{
return state();
}
}

View File

@ -21,19 +21,14 @@ package org.eclipse.jetty.server;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import javax.servlet.AsyncContext;
import javax.servlet.AsyncEvent;
import javax.servlet.AsyncListener; import javax.servlet.AsyncListener;
import javax.servlet.RequestDispatcher; import javax.servlet.RequestDispatcher;
import javax.servlet.ServletContext; import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse; import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import org.eclipse.jetty.server.handler.ContextHandler; import org.eclipse.jetty.server.handler.ContextHandler;
import org.eclipse.jetty.server.handler.ContextHandler.Context; import org.eclipse.jetty.server.handler.ContextHandler.Context;
import org.eclipse.jetty.util.URIUtil;
import org.eclipse.jetty.util.log.Log; import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger; import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.thread.Scheduler; import org.eclipse.jetty.util.thread.Scheduler;
@ -56,7 +51,7 @@ import org.eclipse.jetty.util.thread.Scheduler;
* <tr><th align=right>COMPLETED:</th> <td></td> <td></td> <td></td> <td></td> <td></td> <td></td></tr> * <tr><th align=right>COMPLETED:</th> <td></td> <td></td> <td></td> <td></td> <td></td> <td></td></tr>
* </table> * </table>
*/ */
public class HttpChannelState implements AsyncContext public class HttpChannelState
{ {
private static final Logger LOG = Log.getLogger(HttpChannelState.class); private static final Logger LOG = Log.getLogger(HttpChannelState.class);
@ -86,7 +81,7 @@ public class HttpChannelState implements AsyncContext
private boolean _expired; private boolean _expired;
private volatile boolean _responseWrapped; private volatile boolean _responseWrapped;
private long _timeoutMs=DEFAULT_TIMEOUT; private long _timeoutMs=DEFAULT_TIMEOUT;
private AsyncEventState _event; private AsyncContextEvent _event;
protected HttpChannelState(HttpChannel<?> channel) protected HttpChannelState(HttpChannel<?> channel)
{ {
@ -103,7 +98,6 @@ public class HttpChannelState implements AsyncContext
} }
} }
@Override
public void addListener(AsyncListener listener) public void addListener(AsyncListener listener)
{ {
synchronized(this) synchronized(this)
@ -114,19 +108,6 @@ public class HttpChannelState implements AsyncContext
} }
} }
@Override
public void addListener(AsyncListener listener,ServletRequest request, ServletResponse response)
{
synchronized(this)
{
if (_asyncListeners==null)
_asyncListeners=new ArrayList<>();
_asyncListeners.add(listener);
}
}
@Override
public void setTimeout(long ms) public void setTimeout(long ms)
{ {
synchronized(this) synchronized(this)
@ -135,7 +116,6 @@ public class HttpChannelState implements AsyncContext
} }
} }
@Override
public long getTimeout() public long getTimeout()
{ {
synchronized(this) synchronized(this)
@ -144,7 +124,7 @@ public class HttpChannelState implements AsyncContext
} }
} }
public AsyncEventState getAsyncEventState() public AsyncContextEvent getAsyncContextEvent()
{ {
synchronized(this) synchronized(this)
{ {
@ -218,7 +198,8 @@ public class HttpChannelState implements AsyncContext
} }
} }
public void startAsync()
public void startAsync(AsyncContextEvent event)
{ {
synchronized (this) synchronized (this)
{ {
@ -228,56 +209,15 @@ public class HttpChannelState implements AsyncContext
case REDISPATCHED: case REDISPATCHED:
_dispatched=false; _dispatched=false;
_expired=false; _expired=false;
_responseWrapped=event.getSuppliedResponse()!=_channel.getResponse();
_responseWrapped=false; _responseWrapped=false;
_event=new AsyncEventState(_channel.getRequest().getServletContext(),_channel.getRequest(),_channel.getResponse()); _event=event;
_state=State.ASYNCSTARTED; _state=State.ASYNCSTARTED;
List<AsyncListener> listeners=_lastAsyncListeners; List<AsyncListener> listeners=_lastAsyncListeners;
_lastAsyncListeners=_asyncListeners; _lastAsyncListeners=_asyncListeners;
if (listeners!=null)
listeners.clear();
_asyncListeners=listeners; _asyncListeners=listeners;
if (_asyncListeners!=null)
_asyncListeners.clear();
break;
default:
throw new IllegalStateException(this.getStatusString());
}
}
if (_lastAsyncListeners!=null)
{
for (AsyncListener listener : _lastAsyncListeners)
{
try
{
listener.onStartAsync(_event);
}
catch(Exception e)
{
LOG.warn(e);
}
}
}
}
public void startAsync(final ServletContext context,final ServletRequest request,final ServletResponse response)
{
synchronized (this)
{
switch(_state)
{
case DISPATCHED:
case REDISPATCHED:
_dispatched=false;
_expired=false;
_responseWrapped=response!=_channel.getResponse();
_event=new AsyncEventState(context,request,response);
_event._pathInContext = (request instanceof HttpServletRequest)?URIUtil.addPaths(((HttpServletRequest)request).getServletPath(),((HttpServletRequest)request).getPathInfo()):null;
_state=State.ASYNCSTARTED;
List<AsyncListener> listeners=_lastAsyncListeners;
_lastAsyncListeners=_asyncListeners;
_asyncListeners=listeners;
if (_asyncListeners!=null)
_asyncListeners.clear();
break; break;
default: default:
@ -306,7 +246,7 @@ public class HttpChannelState implements AsyncContext
synchronized (this) synchronized (this)
{ {
if (_event!=null) if (_event!=null)
_event._cause=th; _event.setThrowable(th);
} }
} }
@ -362,26 +302,29 @@ public class HttpChannelState implements AsyncContext
} }
} }
@Override public void dispatch(ServletContext context, String path)
public void dispatch()
{ {
boolean dispatch; boolean dispatch;
synchronized (this) synchronized (this)
{ {
switch(_state) switch(_state)
{ {
case ASYNCSTARTED: case ASYNCSTARTED:
_state=State.REDISPATCHING; _state=State.REDISPATCHING;
_event.setDispatchTarget(context,path);
_dispatched=true; _dispatched=true;
return; return;
case ASYNCWAIT: case ASYNCWAIT:
dispatch=!_expired; dispatch=!_expired;
_state=State.REDISPATCH; _state=State.REDISPATCH;
_event.setDispatchTarget(context,path);
_dispatched=true; _dispatched=true;
break; break;
case REDISPATCH: case REDISPATCH:
_event.setDispatchTarget(context,path);
return; return;
default: default:
@ -453,7 +396,6 @@ public class HttpChannelState implements AsyncContext
scheduleDispatch(); scheduleDispatch();
} }
@Override
public void complete() public void complete()
{ {
// just like resume, except don't set _dispatched=true; // just like resume, except don't set _dispatched=true;
@ -488,19 +430,6 @@ public class HttpChannelState implements AsyncContext
} }
} }
@Override
public <T extends AsyncListener> T createListener(Class<T> clazz) throws ServletException
{
try
{
return clazz.newInstance();
}
catch(Exception e)
{
throw new ServletException(e);
}
}
protected void completed() protected void completed()
{ {
final List<AsyncListener> aListeners; final List<AsyncListener> aListeners;
@ -524,10 +453,10 @@ public class HttpChannelState implements AsyncContext
{ {
try try
{ {
if (_event!=null && _event._cause!=null) if (_event!=null && _event.getThrowable()!=null)
{ {
_event.getSuppliedRequest().setAttribute(RequestDispatcher.ERROR_EXCEPTION,_event._cause); _event.getSuppliedRequest().setAttribute(RequestDispatcher.ERROR_EXCEPTION,_event.getThrowable());
_event.getSuppliedRequest().setAttribute(RequestDispatcher.ERROR_MESSAGE,_event._cause.getMessage()); _event.getSuppliedRequest().setAttribute(RequestDispatcher.ERROR_MESSAGE,_event.getThrowable().getMessage());
listener.onError(_event); listener.onError(_event);
} }
else else
@ -539,6 +468,7 @@ public class HttpChannelState implements AsyncContext
} }
} }
} }
_event.completed();
} }
protected void recycle() protected void recycle()
@ -563,14 +493,6 @@ public class HttpChannelState implements AsyncContext
} }
} }
public void cancel()
{
synchronized (this)
{
cancelTimeout();
}
}
protected void scheduleDispatch() protected void scheduleDispatch()
{ {
_channel.execute(_channel); _channel.execute(_channel);
@ -580,18 +502,14 @@ public class HttpChannelState implements AsyncContext
{ {
Scheduler scheduler = _channel.getScheduler(); Scheduler scheduler = _channel.getScheduler();
if (scheduler!=null && _timeoutMs>0) if (scheduler!=null && _timeoutMs>0)
_event._timeout=scheduler.schedule(new AsyncTimeout(),_timeoutMs,TimeUnit.MILLISECONDS); _event.setTimeoutTask(scheduler.schedule(new AsyncTimeout(),_timeoutMs,TimeUnit.MILLISECONDS));
} }
protected void cancelTimeout() protected void cancelTimeout()
{ {
AsyncEventState event=_event; AsyncContextEvent event=_event;
if (event!=null) if (event!=null)
{ event.cancelTimeoutTask();
Scheduler.Task task=event._timeout;
if (task!=null)
task.cancel();
}
} }
public boolean isExpired() public boolean isExpired()
@ -656,71 +574,19 @@ public class HttpChannelState implements AsyncContext
} }
} }
@Override
public void dispatch(ServletContext context, String path)
{
_event._dispatchContext=context;
_event._pathInContext=path;
dispatch();
}
@Override
public void dispatch(String path)
{
_event._pathInContext=path;
dispatch();
}
public Request getBaseRequest() public Request getBaseRequest()
{ {
return _channel.getRequest(); return _channel.getRequest();
} }
@Override public HttpChannel<?> getHttpChannel()
public ServletRequest getRequest()
{ {
if (_event!=null) return _channel;
return _event.getSuppliedRequest();
return _channel.getRequest();
}
@Override
public ServletResponse getResponse()
{
if (_responseWrapped && _event!=null && _event.getSuppliedResponse()!=null)
return _event.getSuppliedResponse();
return _channel.getResponse();
}
@Override
public void start(final Runnable run)
{
final AsyncEventState event=_event;
if (event!=null)
{
_channel.execute(new Runnable()
{
@Override
public void run()
{
((Context)event.getServletContext()).getContextHandler().handle(run);
}
});
}
}
@Override
public boolean hasOriginalRequestAndResponse()
{
synchronized (this)
{
return (_event!=null && _event.getSuppliedRequest()==_channel.getRequest() && _event.getSuppliedResponse()==_channel.getResponse());
}
} }
public ContextHandler getContextHandler() public ContextHandler getContextHandler()
{ {
final AsyncEventState event=_event; final AsyncContextEvent event=_event;
if (event!=null) if (event!=null)
return ((Context)event.getServletContext()).getContextHandler(); return ((Context)event.getServletContext()).getContextHandler();
return null; return null;
@ -757,70 +623,4 @@ public class HttpChannelState implements AsyncContext
} }
} }
public class AsyncEventState extends AsyncEvent
{
final private ServletContext _suspendedContext;
private String _pathInContext;
private Scheduler.Task _timeout;
private ServletContext _dispatchContext;
private Throwable _cause;
public AsyncEventState(ServletContext context, ServletRequest request, ServletResponse response)
{
super(HttpChannelState.this, request,response);
_suspendedContext=context;
// Get the base request So we can remember the initial paths
Request r=_channel.getRequest();
// If we haven't been async dispatched before
if (r.getAttribute(AsyncContext.ASYNC_REQUEST_URI)==null)
{
// We are setting these attributes during startAsync, when the spec implies that
// they are only available after a call to AsyncContext.dispatch(...);
// have we been forwarded before?
String uri=(String)r.getAttribute(RequestDispatcher.FORWARD_REQUEST_URI);
if (uri!=null)
{
r.setAttribute(AsyncContext.ASYNC_REQUEST_URI,uri);
r.setAttribute(AsyncContext.ASYNC_CONTEXT_PATH,r.getAttribute(RequestDispatcher.FORWARD_CONTEXT_PATH));
r.setAttribute(AsyncContext.ASYNC_SERVLET_PATH,r.getAttribute(RequestDispatcher.FORWARD_SERVLET_PATH));
r.setAttribute(AsyncContext.ASYNC_PATH_INFO,r.getAttribute(RequestDispatcher.FORWARD_PATH_INFO));
r.setAttribute(AsyncContext.ASYNC_QUERY_STRING,r.getAttribute(RequestDispatcher.FORWARD_QUERY_STRING));
}
else
{
r.setAttribute(AsyncContext.ASYNC_REQUEST_URI,r.getRequestURI());
r.setAttribute(AsyncContext.ASYNC_CONTEXT_PATH,r.getContextPath());
r.setAttribute(AsyncContext.ASYNC_SERVLET_PATH,r.getServletPath());
r.setAttribute(AsyncContext.ASYNC_PATH_INFO,r.getPathInfo());
r.setAttribute(AsyncContext.ASYNC_QUERY_STRING,r.getQueryString());
}
}
}
public ServletContext getSuspendedContext()
{
return _suspendedContext;
}
public ServletContext getDispatchContext()
{
return _dispatchContext;
}
public ServletContext getServletContext()
{
return _dispatchContext==null?_suspendedContext:_dispatchContext;
}
/**
* @return The path in the context
*/
public String getPath()
{
return _pathInContext;
}
}
} }

View File

@ -204,7 +204,8 @@ public class Request implements HttpServletRequest
private long _dispatchTime; private long _dispatchTime;
private HttpURI _uri; private HttpURI _uri;
private MultiPartInputStreamParser _multiPartInputStream; //if the request is a multi-part mime private MultiPartInputStreamParser _multiPartInputStream; //if the request is a multi-part mime
private AsyncContextState _async;
/* ------------------------------------------------------------ */ /* ------------------------------------------------------------ */
public Request(HttpChannel<?> channel, HttpInput<?> input) public Request(HttpChannel<?> channel, HttpInput<?> input)
{ {
@ -370,10 +371,11 @@ public class Request implements HttpServletRequest
@Override @Override
public AsyncContext getAsyncContext() public AsyncContext getAsyncContext()
{ {
HttpChannelState continuation = getHttpChannelState(); HttpChannelState state = getHttpChannelState();
if (continuation.isInitial() && !continuation.isAsync()) if (_async==null || state.isInitial() && !state.isAsync())
throw new IllegalStateException(continuation.getStatusString()); throw new IllegalStateException(state.getStatusString());
return continuation;
return _async;
} }
/* ------------------------------------------------------------ */ /* ------------------------------------------------------------ */
@ -1517,6 +1519,9 @@ public class Request implements HttpServletRequest
setAuthentication(Authentication.NOT_CHECKED); setAuthentication(Authentication.NOT_CHECKED);
getHttpChannelState().recycle(); getHttpChannelState().recycle();
if (_async!=null)
_async.reset();
_async=null;
_asyncSupported = true; _asyncSupported = true;
_handled = false; _handled = false;
if (_context != null) if (_context != null)
@ -2000,8 +2005,11 @@ public class Request implements HttpServletRequest
if (!_asyncSupported) if (!_asyncSupported)
throw new IllegalStateException("!asyncSupported"); throw new IllegalStateException("!asyncSupported");
HttpChannelState state = getHttpChannelState(); HttpChannelState state = getHttpChannelState();
state.startAsync(); if (_async==null)
return state; _async=new AsyncContextState(state);
AsyncContextEvent event = new AsyncContextEvent(_context,_async,state,this,this,getResponse());
state.startAsync(event);
return _async;
} }
/* ------------------------------------------------------------ */ /* ------------------------------------------------------------ */
@ -2011,8 +2019,12 @@ public class Request implements HttpServletRequest
if (!_asyncSupported) if (!_asyncSupported)
throw new IllegalStateException("!asyncSupported"); throw new IllegalStateException("!asyncSupported");
HttpChannelState state = getHttpChannelState(); HttpChannelState state = getHttpChannelState();
state.startAsync(_context, servletRequest, servletResponse); if (_async==null)
return state; _async=new AsyncContextState(state);
AsyncContextEvent event = new AsyncContextEvent(_context,_async,state,this,servletRequest,servletResponse);
event.setDispatchTarget(getServletContext(),URIUtil.addPaths(getServletPath(),getPathInfo()));
state.startAsync(event);
return _async;
} }
/* ------------------------------------------------------------ */ /* ------------------------------------------------------------ */

View File

@ -476,16 +476,16 @@ public class Server extends HandlerWrapper implements Attributes
*/ */
public void handleAsync(HttpChannel<?> connection) throws IOException, ServletException public void handleAsync(HttpChannel<?> connection) throws IOException, ServletException
{ {
final HttpChannelState async = connection.getRequest().getHttpChannelState(); final HttpChannelState state = connection.getRequest().getHttpChannelState();
final HttpChannelState.AsyncEventState state = async.getAsyncEventState(); final AsyncContextEvent event = state.getAsyncContextEvent();
final Request baseRequest=connection.getRequest(); final Request baseRequest=connection.getRequest();
final String path=state.getPath(); final String path=event.getPath();
if (path!=null) if (path!=null)
{ {
// this is a dispatch with a path // this is a dispatch with a path
ServletContext context=state.getServletContext(); ServletContext context=event.getServletContext();
HttpURI uri = new HttpURI(context==null?path:URIUtil.addPaths(context.getContextPath(),path)); HttpURI uri = new HttpURI(context==null?path:URIUtil.addPaths(context.getContextPath(),path));
baseRequest.setUri(uri); baseRequest.setUri(uri);
baseRequest.setRequestURI(null); baseRequest.setRequestURI(null);
@ -495,8 +495,8 @@ public class Server extends HandlerWrapper implements Attributes
} }
final String target=baseRequest.getPathInfo(); final String target=baseRequest.getPathInfo();
final HttpServletRequest request=(HttpServletRequest)async.getRequest(); final HttpServletRequest request=(HttpServletRequest)event.getSuppliedRequest();
final HttpServletResponse response=(HttpServletResponse)async.getResponse(); final HttpServletResponse response=(HttpServletResponse)event.getSuppliedResponse();
if (LOG.isDebugEnabled()) if (LOG.isDebugEnabled())
{ {
@ -509,7 +509,6 @@ public class Server extends HandlerWrapper implements Attributes
} }
/* ------------------------------------------------------------ */ /* ------------------------------------------------------------ */
public void join() throws InterruptedException public void join() throws InterruptedException
{ {

View File

@ -28,6 +28,7 @@ import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.server.AsyncContextEvent;
import org.eclipse.jetty.server.HttpChannelState; import org.eclipse.jetty.server.HttpChannelState;
import org.eclipse.jetty.server.Request; import org.eclipse.jetty.server.Request;
import org.eclipse.jetty.server.Response; import org.eclipse.jetty.server.Response;
@ -76,12 +77,13 @@ public class StatisticsHandler extends HandlerWrapper
public void onError(AsyncEvent event) throws IOException public void onError(AsyncEvent event) throws IOException
{ {
} }
@Override @Override
public void onComplete(AsyncEvent event) throws IOException public void onComplete(AsyncEvent event) throws IOException
{ {
HttpChannelState state = (HttpChannelState)event.getAsyncContext();
HttpChannelState state = ((AsyncContextEvent)event).getHttpChannelState();
Request request = state.getBaseRequest(); Request request = state.getBaseRequest();
final long elapsed = System.currentTimeMillis()-request.getTimeStamp(); final long elapsed = System.currentTimeMillis()-request.getTimeStamp();
@ -93,7 +95,7 @@ public class StatisticsHandler extends HandlerWrapper
if (!state.isDispatched()) if (!state.isDispatched())
_asyncWaitStats.decrement(); _asyncWaitStats.decrement();
} }
}; };
/** /**

View File

@ -103,6 +103,7 @@ public class AsyncContextTest
Assert.assertEquals("servlet gets right path", "doGet:getServletPath:/servletPath", br.readLine()); Assert.assertEquals("servlet gets right path", "doGet:getServletPath:/servletPath", br.readLine());
Assert.assertEquals("async context gets right path in get","doGet:async:getServletPath:/servletPath",br.readLine()); Assert.assertEquals("async context gets right path in get","doGet:async:getServletPath:/servletPath",br.readLine());
Assert.assertEquals("async context gets right path in async","async:run:attr:servletPath:/servletPath",br.readLine()); Assert.assertEquals("async context gets right path in async","async:run:attr:servletPath:/servletPath",br.readLine());
} }
@Test @Test
@ -121,6 +122,16 @@ public class AsyncContextTest
Assert.assertEquals("query string attr is correct","async:run:attr:queryString:dispatch=true",br.readLine()); Assert.assertEquals("query string attr is correct","async:run:attr:queryString:dispatch=true",br.readLine());
Assert.assertEquals("context path attr is correct","async:run:attr:contextPath:",br.readLine()); Assert.assertEquals("context path attr is correct","async:run:attr:contextPath:",br.readLine());
Assert.assertEquals("request uri attr is correct","async:run:attr:requestURI:/servletPath",br.readLine()); Assert.assertEquals("request uri attr is correct","async:run:attr:requestURI:/servletPath",br.readLine());
try
{
__asyncContext.getRequest();
Assert.fail();
}
catch (IllegalStateException e)
{
}
} }
@Test @Test
@ -193,8 +204,11 @@ public class AsyncContextTest
@Test @Test
public void testDispatchRequestResponse() throws Exception public void testDispatchRequestResponse() throws Exception
{ {
String request = "GET /forward?dispatchRequestResponse=true HTTP/1.1\r\n" + "Host: localhost\r\n" String request = "GET /forward?dispatchRequestResponse=true HTTP/1.1\r\n" +
+ "Content-Type: application/x-www-form-urlencoded\r\n" + "Connection: close\r\n" + "\r\n"; "Host: localhost\r\n" +
"Content-Type: application/x-www-form-urlencoded\r\n" +
"Connection: close\r\n" +
"\r\n";
String responseString = _connector.getResponses(request); String responseString = _connector.getResponses(request);
@ -233,10 +247,12 @@ public class AsyncContextTest
} }
} }
public static volatile AsyncContext __asyncContext;
private class AsyncDispatchingServlet extends HttpServlet private class AsyncDispatchingServlet extends HttpServlet
{ {
private static final long serialVersionUID = 1L; private static final long serialVersionUID = 1L;
@Override @Override
protected void doGet(HttpServletRequest req, final HttpServletResponse response) throws ServletException, IOException protected void doGet(HttpServletRequest req, final HttpServletResponse response) throws ServletException, IOException
{ {
@ -253,13 +269,14 @@ public class AsyncContextTest
{ {
wrapped = true; wrapped = true;
asyncContext = request.startAsync(request, new Wrapper(response)); asyncContext = request.startAsync(request, new Wrapper(response));
__asyncContext=asyncContext;
} }
else else
{ {
asyncContext = request.startAsync(); asyncContext = request.startAsync();
__asyncContext=asyncContext;
} }
new Thread(new DispatchingRunnable(asyncContext, wrapped)).start(); new Thread(new DispatchingRunnable(asyncContext, wrapped)).start();
} }
} }
@ -301,12 +318,14 @@ public class AsyncContextTest
if (request.getParameter("dispatch") != null) if (request.getParameter("dispatch") != null)
{ {
AsyncContext asyncContext = request.startAsync(request,response); AsyncContext asyncContext = request.startAsync(request,response);
__asyncContext=asyncContext;
asyncContext.dispatch("/servletPath2"); asyncContext.dispatch("/servletPath2");
} }
else else
{ {
response.getOutputStream().print("doGet:getServletPath:" + request.getServletPath() + "\n"); response.getOutputStream().print("doGet:getServletPath:" + request.getServletPath() + "\n");
AsyncContext asyncContext = request.startAsync(request,response); AsyncContext asyncContext = request.startAsync(request,response);
__asyncContext=asyncContext;
response.getOutputStream().print("doGet:async:getServletPath:" + ((HttpServletRequest)asyncContext.getRequest()).getServletPath() + "\n"); response.getOutputStream().print("doGet:async:getServletPath:" + ((HttpServletRequest)asyncContext.getRequest()).getServletPath() + "\n");
asyncContext.start(new AsyncRunnable(asyncContext)); asyncContext.start(new AsyncRunnable(asyncContext));
@ -323,6 +342,7 @@ public class AsyncContextTest
{ {
response.getOutputStream().print("doGet:getServletPath:" + request.getServletPath() + "\n"); response.getOutputStream().print("doGet:getServletPath:" + request.getServletPath() + "\n");
AsyncContext asyncContext = request.startAsync(request, response); AsyncContext asyncContext = request.startAsync(request, response);
__asyncContext=asyncContext;
response.getOutputStream().print("doGet:async:getServletPath:" + ((HttpServletRequest)asyncContext.getRequest()).getServletPath() + "\n"); response.getOutputStream().print("doGet:async:getServletPath:" + ((HttpServletRequest)asyncContext.getRequest()).getServletPath() + "\n");
asyncContext.start(new AsyncRunnable(asyncContext)); asyncContext.start(new AsyncRunnable(asyncContext));
} }