From 983193071e8f2062e0350cb18b4af0e4e921aff5 Mon Sep 17 00:00:00 2001 From: Greg Wilkins Date: Thu, 26 Mar 2020 13:56:00 +0100 Subject: [PATCH 1/2] Issue #4713 Async dispatch with query. + Preserve the entire URI with query when startAsync(req,res) is used. + merge any query string from dispatch path with either original query or preserved query from forward Signed-off-by: Greg Wilkins --- .../java/org/eclipse/jetty/http/HttpURI.java | 6 +- .../jetty/server/AsyncContextEvent.java | 26 ++-- .../org/eclipse/jetty/server/Request.java | 22 ++-- .../java/org/eclipse/jetty/server/Server.java | 76 ++++++++++-- .../jetty/servlet/AsyncServletTest.java | 116 +++++++++--------- .../java/org/eclipse/jetty/util/URIUtil.java | 14 +++ 6 files changed, 167 insertions(+), 93 deletions(-) diff --git a/jetty-http/src/main/java/org/eclipse/jetty/http/HttpURI.java b/jetty-http/src/main/java/org/eclipse/jetty/http/HttpURI.java index 2bde2c1deaad..2c112940f8bb 100644 --- a/jetty-http/src/main/java/org/eclipse/jetty/http/HttpURI.java +++ b/jetty-http/src/main/java/org/eclipse/jetty/http/HttpURI.java @@ -760,15 +760,15 @@ public void setDecodedPath(String path) _decodedPath = path; } - public void setPathQuery(String path) + public void setPathQuery(String pathQuery) { _uri = null; _path = null; _decodedPath = null; _param = null; _fragment = null; - if (path != null) - parse(State.PATH, path); + if (pathQuery != null) + parse(State.PATH, pathQuery); } public void setQuery(String query) diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java b/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java index 4da7cc87e67b..f9faa5cba873 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java @@ -25,6 +25,7 @@ import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; +import org.eclipse.jetty.http.HttpURI; import org.eclipse.jetty.server.handler.ContextHandler.Context; import org.eclipse.jetty.util.thread.Scheduler; @@ -32,18 +33,20 @@ public class AsyncContextEvent extends AsyncEvent implements Runnable { private final Context _context; private final AsyncContextState _asyncContext; + private final HttpURI _providedUri; private volatile HttpChannelState _state; private ServletContext _dispatchContext; private String _dispatchPath; private volatile Scheduler.Task _timeoutTask; private Throwable _throwable; - public AsyncContextEvent(Context context, AsyncContextState asyncContext, HttpChannelState state, Request baseRequest, ServletRequest request, ServletResponse response) + public AsyncContextEvent(Context context, AsyncContextState asyncContext, HttpChannelState state, Request baseRequest, ServletRequest request, ServletResponse response, boolean provided) { super(null, request, response, null); _context = context; _asyncContext = asyncContext; _state = state; + _providedUri = provided ? new HttpURI(baseRequest.getHttpURI()) : null; // If we haven't been async dispatched before if (baseRequest.getAttribute(AsyncContext.ASYNC_REQUEST_URI) == null) @@ -74,6 +77,11 @@ public AsyncContextEvent(Context context, AsyncContextState asyncContext, HttpCh } } + public HttpURI getProvidedUri() + { + return _providedUri; + } + public ServletContext getSuspendedContext() { return _context; @@ -94,14 +102,6 @@ public ServletContext getServletContext() return _dispatchContext == null ? _context : _dispatchContext; } - /** - * @return The path in the context (encoded with possible query string) - */ - public String getPath() - { - return _dispatchPath; - } - public void setTimeoutTask(Scheduler.Task task) { _timeoutTask = task; @@ -137,6 +137,14 @@ public void setDispatchContext(ServletContext context) _dispatchContext = context; } + /** + * @return The path in the context (encoded with possible query string) + */ + public String getDispatchPath() + { + return _dispatchPath; + } + /** * @param path encoded URI */ diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java b/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java index 6abada24f582..d2e184ffb1eb 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java @@ -1598,7 +1598,10 @@ public void setHttpURI(HttpURI uri) { MetaData.Request metadata = _metaData; if (metadata != null) + { metadata.setURI(uri); + _queryParameters = null; + } } public UserIdentity getUserIdentity() @@ -2165,7 +2168,7 @@ public AsyncContext startAsync() throws IllegalStateException HttpChannelState state = getHttpChannelState(); if (_async == null) _async = new AsyncContextState(state); - AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, this, getResponse()); + AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, this, getResponse(), false); state.startAsync(event); return _async; } @@ -2178,17 +2181,8 @@ public AsyncContext startAsync(ServletRequest servletRequest, ServletResponse se HttpChannelState state = getHttpChannelState(); if (_async == null) _async = new AsyncContextState(state); - AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, servletRequest, servletResponse); + AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, servletRequest, servletResponse, true); event.setDispatchContext(getServletContext()); - - String uri = unwrap(servletRequest).getRequestURI(); - if (_contextPath != null && uri.startsWith(_contextPath)) - uri = uri.substring(_contextPath.length()); - else - // TODO probably need to strip encoded context from requestURI, but will do this for now: - uri = URIUtil.encodePath(URIUtil.addPaths(getServletPath(), getPathInfo())); - - event.setDispatchPath(uri); state.startAsync(event); return _async; } @@ -2391,7 +2385,7 @@ else if (oldQueryParams == null || oldQueryParams.size() == 0) setQueryString(oldQuery); else if (oldQuery == null) setQueryString(newQuery); - else + else if (oldQueryParams.keySet().stream().anyMatch(newQueryParams.keySet()::contains)) { // Build the new merged query string, parameters in the // new query string hide parameters in the old query string. @@ -2413,6 +2407,10 @@ else if (oldQuery == null) } setQueryString(mergedQuery.toString()); } + else + { + setQueryString(newQuery + '&' + oldQuery); + } } } diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java b/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java index d9ade6e27d4c..af33ae9a67ef 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java @@ -47,6 +47,8 @@ import org.eclipse.jetty.util.Attributes; import org.eclipse.jetty.util.Jetty; import org.eclipse.jetty.util.MultiException; +import org.eclipse.jetty.util.MultiMap; +import org.eclipse.jetty.util.StringUtil; import org.eclipse.jetty.util.URIUtil; import org.eclipse.jetty.util.Uptime; import org.eclipse.jetty.util.annotation.ManagedAttribute; @@ -563,22 +565,74 @@ public void handleAsync(HttpChannel channel) throws IOException, ServletExceptio { final HttpChannelState state = channel.getRequest().getHttpChannelState(); final AsyncContextEvent event = state.getAsyncContextEvent(); - final Request baseRequest = channel.getRequest(); - final String path = event.getPath(); + final HttpURI providedUri = event.getProvidedUri(); + String encodedPathQuery = event.getDispatchPath(); - if (path != null) + if (encodedPathQuery == null && providedUri == null) { - // this is a dispatch with a path - ServletContext context = event.getServletContext(); - String query = baseRequest.getQueryString(); - baseRequest.setURIPathQuery(URIUtil.addEncodedPaths(context == null ? null : URIUtil.encodePath(context.getContextPath()), path)); - HttpURI uri = baseRequest.getHttpURI(); - baseRequest.setPathInfo(uri.getDecodedPath()); - if (uri.getQuery() != null) - baseRequest.mergeQueryParameters(query, uri.getQuery(), true); //we have to assume dispatch path and query are UTF8 + // Simple case, no request modification or merging needed + handleAsync(channel, event, baseRequest); + return; } + // this is a dispatch with either a provided URI and/or a dispatched path + // We will have to modify the request and then revert + final ServletContext context = event.getServletContext(); + final HttpURI oldUri = baseRequest.getHttpURI(); + final String oldQuery = baseRequest.getQueryString(); + final MultiMap oldQueryParams = baseRequest.getQueryParameters(); + try + { + baseRequest.resetParameters(); + HttpURI newUri = providedUri == null ? new HttpURI(oldUri) : providedUri; + if (encodedPathQuery == null) + { + baseRequest.setHttpURI(newUri); + } + else + { + if (context != null && !StringUtil.isEmpty(context.getContextPath())) + encodedPathQuery = URIUtil.addEncodedPaths(URIUtil.encodePath(context.getContextPath()), encodedPathQuery); + + if (newUri.getQuery() == null) + { + // parse new path and query + newUri.setPathQuery(encodedPathQuery); + baseRequest.setHttpURI(newUri); + } + else + { + // do we have a new query in the encodedPathQuery + int q = encodedPathQuery.indexOf('?'); + if (q < 0) + { + // No query, so we can just set the encoded path + newUri.setPath(encodedPathQuery); + baseRequest.setHttpURI(newUri); + } + else + { + newUri.setPath(encodedPathQuery.substring(0, q)); + baseRequest.setHttpURI(newUri); + baseRequest.mergeQueryParameters(oldQuery, encodedPathQuery.substring(q + 1), true); + } + } + } + + baseRequest.setPathInfo(newUri.getDecodedPath()); + handleAsync(channel, event, baseRequest); + } + finally + { + baseRequest.setHttpURI(oldUri); + baseRequest.setQueryParameters(oldQueryParams); + baseRequest.resetParameters(); + } + } + + private void handleAsync(HttpChannel channel, AsyncContextEvent event, Request baseRequest) throws IOException, ServletException + { final String target = baseRequest.getPathInfo(); final HttpServletRequest request = Request.unwrap(event.getSuppliedRequest()); final HttpServletResponse response = Response.unwrap(event.getSuppliedResponse()); diff --git a/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncServletTest.java b/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncServletTest.java index ac16a759a6a3..f04cffee271b 100644 --- a/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncServletTest.java +++ b/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncServletTest.java @@ -163,7 +163,7 @@ public void testSleep() throws Exception String response = process("sleep=200", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?sleep=200", "initial")); assertContains("SLEPT", response); assertFalse(__history.contains("onTimeout")); @@ -173,7 +173,7 @@ public void testSleep() throws Exception @Test public void testNonAsync() throws Exception { - String response = process("", null); + String response = process(null, null); assertThat(response, Matchers.startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( "REQUEST /ctx/path/info", @@ -186,7 +186,7 @@ public void testNonAsync() throws Exception public void testAsyncNotSupportedNoAsync() throws Exception { _expectedCode = "200 "; - String response = process("noasync", "", null); + String response = process("noasync", null, null); assertThat(response, Matchers.startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( "REQUEST /ctx/noasync/info", @@ -205,9 +205,9 @@ public void testAsyncNotSupportedAsync() throws Exception String response = process("noasync", "start=200", null); assertThat(response, Matchers.startsWith("HTTP/1.1 500 ")); assertThat(__history, contains( - "REQUEST /ctx/noasync/info", + "REQUEST /ctx/noasync/info?start=200", "initial", - "ERROR /ctx/error/custom", + "ERROR /ctx/error/custom?start=200", "!initial" )); @@ -224,11 +224,11 @@ public void testStart() throws Exception String response = process("start=200", null); assertThat(response, Matchers.startsWith("HTTP/1.1 500 Server Error")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200", "initial", "start", "onTimeout", - "ERROR /ctx/error/custom", + "ERROR /ctx/error/custom?start=200", "!initial", "onComplete")); @@ -241,12 +241,12 @@ public void testStartOnTimeoutDispatch() throws Exception String response = process("start=200&timeout=dispatch", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&timeout=dispatch", "initial", "start", "onTimeout", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=200&timeout=dispatch", "!initial", "onComplete")); @@ -260,12 +260,12 @@ public void testStartOnTimeoutError() throws Exception String response = process("start=200&timeout=error", null); assertThat(response, startsWith("HTTP/1.1 500 Server Error")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&timeout=error", "initial", "start", "onTimeout", "error", - "ERROR /ctx/error/custom", + "ERROR /ctx/error/custom?start=200&timeout=error", "!initial", "onComplete")); @@ -278,7 +278,7 @@ public void testStartOnTimeoutComplete() throws Exception String response = process("start=200&timeout=complete", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&timeout=complete", "initial", "start", "onTimeout", @@ -294,11 +294,11 @@ public void testStartWaitDispatch() throws Exception String response = process("start=200&dispatch=10", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&dispatch=10", "initial", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=200&dispatch=10", "!initial", "onComplete")); assertFalse(__history.contains("onTimeout")); @@ -310,11 +310,11 @@ public void testStartDispatch() throws Exception String response = process("start=200&dispatch=0", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&dispatch=0", "initial", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=200&dispatch=0", "!initial", "onComplete")); } @@ -326,11 +326,11 @@ public void testStartError() throws Exception String response = process("start=200&throw=1", null); assertThat(response, startsWith("HTTP/1.1 500 Server Error")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&throw=1", "initial", "start", "onError", - "ERROR /ctx/error/custom", + "ERROR /ctx/error/custom?start=200&throw=1", "!initial", "onComplete")); assertContains("ERROR DISPATCH: /ctx/error/custom", response); @@ -342,7 +342,7 @@ public void testStartWaitComplete() throws Exception String response = process("start=200&complete=50", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&complete=50", "initial", "start", "complete", @@ -358,7 +358,7 @@ public void testStartComplete() throws Exception String response = process("start=200&complete=0", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&complete=0", "initial", "start", "complete", @@ -374,16 +374,16 @@ public void testStartWaitDispatchStartWaitDispatch() throws Exception String response = process("start=1000&dispatch=10&start2=1000&dispatch2=10", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=1000&dispatch=10&start2=1000&dispatch2=10", "initial", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=1000&dispatch=10&start2=1000&dispatch2=10", "!initial", "onStartAsync", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=1000&dispatch=10&start2=1000&dispatch2=10", "!initial", "onComplete")); assertContains("DISPATCHED", response); @@ -395,11 +395,11 @@ public void testStartWaitDispatchStartComplete() throws Exception String response = process("start=1000&dispatch=10&start2=1000&complete2=10", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=1000&dispatch=10&start2=1000&complete2=10", "initial", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=1000&dispatch=10&start2=1000&complete2=10", "!initial", "onStartAsync", "start", @@ -415,16 +415,16 @@ public void testStartWaitDispatchStart() throws Exception String response = process("start=1000&dispatch=10&start2=10", null); assertThat(response, startsWith("HTTP/1.1 500 Server Error")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=1000&dispatch=10&start2=10", "initial", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=1000&dispatch=10&start2=10", "!initial", "onStartAsync", "start", "onTimeout", - "ERROR /ctx/error/custom", + "ERROR /ctx/error/custom?start=1000&dispatch=10&start2=10", "!initial", "onComplete")); assertContains("ERROR DISPATCH: /ctx/error/custom", response); @@ -436,16 +436,16 @@ public void testStartTimeoutStartDispatch() throws Exception String response = process("start=10&start2=1000&dispatch2=10", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=10&start2=1000&dispatch2=10", "initial", "start", "onTimeout", - "ERROR /ctx/error/custom", + "ERROR /ctx/error/custom?start=10&start2=1000&dispatch2=10", "!initial", "onStartAsync", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=10&start2=1000&dispatch2=10", "!initial", "onComplete")); assertContains("DISPATCHED", response); @@ -457,11 +457,11 @@ public void testStartTimeoutStartComplete() throws Exception String response = process("start=10&start2=1000&complete2=10", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=10&start2=1000&complete2=10", "initial", "start", "onTimeout", - "ERROR /ctx/error/custom", + "ERROR /ctx/error/custom?start=10&start2=1000&complete2=10", "!initial", "onStartAsync", "start", @@ -478,16 +478,16 @@ public void testStartTimeoutStart() throws Exception String response = process("start=10&start2=10", null); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=10&start2=10", "initial", "start", "onTimeout", - "ERROR /ctx/path/error", + "ERROR /ctx/path/error?start=10&start2=10", "!initial", "onStartAsync", "start", "onTimeout", - "ERROR /ctx/path/error", + "ERROR /ctx/path/error?start=10&start2=10", "!initial", "onComplete")); // Error Page Loop! assertContains("AsyncContext timeout", response); @@ -499,11 +499,11 @@ public void testWrapStartDispatch() throws Exception String response = process("wrap=true&start=200&dispatch=20", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?wrap=true&start=200&dispatch=20", "initial", "start", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?wrap=true&start=200&dispatch=20", "wrapped REQ RSP", "!initial", "onComplete")); @@ -516,11 +516,11 @@ public void testStartDispatchEncodedPath() throws Exception String response = process("start=200&dispatch=20&path=/p%20th3", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=200&dispatch=20&path=/p%20th3", "initial", "start", "dispatch", - "ASYNC /ctx/p%20th3", + "ASYNC /ctx/p%20th3?start=200&dispatch=20&path=/p%20th3", "!initial", "onComplete")); assertContains("DISPATCHED", response); @@ -532,13 +532,13 @@ public void testFwdStartDispatch() throws Exception String response = process("fwd", "start=200&dispatch=20", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "FWD REQUEST /ctx/fwd/info", - "FORWARD /ctx/path1", + "FWD REQUEST /ctx/fwd/info?start=200&dispatch=20", + "FORWARD /ctx/path1?forward=true&start=200&dispatch=20", "initial", "start", "dispatch", - "FWD ASYNC /ctx/fwd/info", - "FORWARD /ctx/path1", + "FWD ASYNC /ctx/fwd/info?start=200&dispatch=20", + "FORWARD /ctx/path1?forward=true&start=200&dispatch=20", "!initial", "onComplete")); assertContains("DISPATCHED", response); @@ -550,12 +550,12 @@ public void testFwdStartDispatchPath() throws Exception String response = process("fwd", "start=200&dispatch=20&path=/path2", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "FWD REQUEST /ctx/fwd/info", - "FORWARD /ctx/path1", + "FWD REQUEST /ctx/fwd/info?start=200&dispatch=20&path=/path2", + "FORWARD /ctx/path1?forward=true&start=200&dispatch=20&path=/path2", "initial", "start", "dispatch", - "ASYNC /ctx/path2", + "ASYNC /ctx/path2?start=200&dispatch=20&path=/path2", "!initial", "onComplete")); assertContains("DISPATCHED", response); @@ -567,12 +567,12 @@ public void testFwdWrapStartDispatch() throws Exception String response = process("fwd", "wrap=true&start=200&dispatch=20", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "FWD REQUEST /ctx/fwd/info", - "FORWARD /ctx/path1", + "FWD REQUEST /ctx/fwd/info?wrap=true&start=200&dispatch=20", + "FORWARD /ctx/path1?forward=true&wrap=true&start=200&dispatch=20", "initial", "start", "dispatch", - "ASYNC /ctx/path1", + "ASYNC /ctx/path1?forward=true&wrap=true&start=200&dispatch=20", "wrapped REQ RSP", "!initial", "onComplete")); @@ -585,12 +585,12 @@ public void testFwdWrapStartDispatchPath() throws Exception String response = process("fwd", "wrap=true&start=200&dispatch=20&path=/path2", null); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "FWD REQUEST /ctx/fwd/info", - "FORWARD /ctx/path1", + "FWD REQUEST /ctx/fwd/info?wrap=true&start=200&dispatch=20&path=/path2", + "FORWARD /ctx/path1?forward=true&wrap=true&start=200&dispatch=20&path=/path2", "initial", "start", "dispatch", - "ASYNC /ctx/path2", + "ASYNC /ctx/path2?forward=true&wrap=true&start=200&dispatch=20&path=/path2", "wrapped REQ RSP", "!initial", "onComplete")); @@ -619,12 +619,12 @@ public void testAsyncRead() throws Exception __latch.await(1, TimeUnit.SECONDS); assertThat(response, startsWith("HTTP/1.1 200 OK")); assertThat(__history, contains( - "REQUEST /ctx/path/info", + "REQUEST /ctx/path/info?start=2000&dispatch=1500", "initial", "start", "async-read=10", "dispatch", - "ASYNC /ctx/path/info", + "ASYNC /ctx/path/info?start=2000&dispatch=1500", "!initial", "onComplete")); } @@ -685,10 +685,10 @@ private static class FwdServlet extends HttpServlet @Override public void doGet(final HttpServletRequest request, final HttpServletResponse response) throws ServletException, IOException { - historyAdd("FWD " + request.getDispatcherType() + " " + request.getRequestURI()); + historyAdd("FWD " + request.getDispatcherType() + " " + URIUtil.addPathQuery(request.getRequestURI(), request.getQueryString())); if (request instanceof ServletRequestWrapper || response instanceof ServletResponseWrapper) historyAdd("wrapped" + ((request instanceof ServletRequestWrapper) ? " REQ" : "") + ((response instanceof ServletResponseWrapper) ? " RSP" : "")); - request.getServletContext().getRequestDispatcher("/path1").forward(request, response); + request.getServletContext().getRequestDispatcher("/path1?forward=true").forward(request, response); } } @@ -711,7 +711,7 @@ public void doGet(final HttpServletRequest request, final HttpServletResponse re // ignored } - historyAdd(request.getDispatcherType() + " " + request.getRequestURI()); + historyAdd(request.getDispatcherType() + " " + URIUtil.addPathQuery(request.getRequestURI(),request.getQueryString())); if (request instanceof ServletRequestWrapper || response instanceof ServletResponseWrapper) historyAdd("wrapped" + ((request instanceof ServletRequestWrapper) ? " REQ" : "") + ((response instanceof ServletResponseWrapper) ? " RSP" : "")); diff --git a/jetty-util/src/main/java/org/eclipse/jetty/util/URIUtil.java b/jetty-util/src/main/java/org/eclipse/jetty/util/URIUtil.java index 1909e1022425..6ae6f609cd96 100644 --- a/jetty-util/src/main/java/org/eclipse/jetty/util/URIUtil.java +++ b/jetty-util/src/main/java/org/eclipse/jetty/util/URIUtil.java @@ -714,6 +714,20 @@ public static String addPaths(String p1, String p2) return buf.toString(); } + /** Add a path and a query string + * @param path The path which may already contain contain a query + * @param query The query string or null if no query to be added + * @return The path with any non null query added after a '?' or '&' as appropriate. + */ + public static String addPathQuery(String path, String query) + { + if (query == null) + return path; + if (path.indexOf('?') >= 0) + return path + '&' + query; + return path + '?' + query; + } + /** * Given a URI, attempt to get the last segment. *

From a74f36c38cd2a1c6d981290dc233eea73d014859 Mon Sep 17 00:00:00 2001 From: Greg Wilkins Date: Mon, 30 Mar 2020 12:29:47 +0200 Subject: [PATCH 2/2] Issue #4713 asyncDispatch with query parameters Signed-off-by: Greg Wilkins --- .../eclipse/jetty/server/AsyncContextEvent.java | 15 ++++++++++----- .../java/org/eclipse/jetty/server/Request.java | 4 ++-- .../java/org/eclipse/jetty/server/Server.java | 6 +++--- 3 files changed, 15 insertions(+), 10 deletions(-) diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java b/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java index f9faa5cba873..c4bc0095aee0 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContextEvent.java @@ -33,20 +33,25 @@ public class AsyncContextEvent extends AsyncEvent implements Runnable { private final Context _context; private final AsyncContextState _asyncContext; - private final HttpURI _providedUri; + private final HttpURI _baseURI; private volatile HttpChannelState _state; private ServletContext _dispatchContext; private String _dispatchPath; private volatile Scheduler.Task _timeoutTask; private Throwable _throwable; - public AsyncContextEvent(Context context, AsyncContextState asyncContext, HttpChannelState state, Request baseRequest, ServletRequest request, ServletResponse response, boolean provided) + public AsyncContextEvent(Context context, AsyncContextState asyncContext, HttpChannelState state, Request baseRequest, ServletRequest request, ServletResponse response) + { + this (context, asyncContext, state, baseRequest, request, response, null); + } + + public AsyncContextEvent(Context context, AsyncContextState asyncContext, HttpChannelState state, Request baseRequest, ServletRequest request, ServletResponse response, HttpURI baseURI) { super(null, request, response, null); _context = context; _asyncContext = asyncContext; _state = state; - _providedUri = provided ? new HttpURI(baseRequest.getHttpURI()) : null; + _baseURI = baseURI; // If we haven't been async dispatched before if (baseRequest.getAttribute(AsyncContext.ASYNC_REQUEST_URI) == null) @@ -77,9 +82,9 @@ public AsyncContextEvent(Context context, AsyncContextState asyncContext, HttpCh } } - public HttpURI getProvidedUri() + public HttpURI getBaseURI() { - return _providedUri; + return _baseURI; } public ServletContext getSuspendedContext() diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java b/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java index d2e184ffb1eb..07f2d318be4c 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/Request.java @@ -2168,7 +2168,7 @@ public AsyncContext startAsync() throws IllegalStateException HttpChannelState state = getHttpChannelState(); if (_async == null) _async = new AsyncContextState(state); - AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, this, getResponse(), false); + AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, this, getResponse()); state.startAsync(event); return _async; } @@ -2181,7 +2181,7 @@ public AsyncContext startAsync(ServletRequest servletRequest, ServletResponse se HttpChannelState state = getHttpChannelState(); if (_async == null) _async = new AsyncContextState(state); - AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, servletRequest, servletResponse, true); + AsyncContextEvent event = new AsyncContextEvent(_context, _async, state, this, servletRequest, servletResponse, getHttpURI()); event.setDispatchContext(getServletContext()); state.startAsync(event); return _async; diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java b/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java index af33ae9a67ef..1566d6ba2f70 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/Server.java @@ -566,10 +566,10 @@ public void handleAsync(HttpChannel channel) throws IOException, ServletExceptio final HttpChannelState state = channel.getRequest().getHttpChannelState(); final AsyncContextEvent event = state.getAsyncContextEvent(); final Request baseRequest = channel.getRequest(); - final HttpURI providedUri = event.getProvidedUri(); + final HttpURI baseUri = event.getBaseURI(); String encodedPathQuery = event.getDispatchPath(); - if (encodedPathQuery == null && providedUri == null) + if (encodedPathQuery == null && baseUri == null) { // Simple case, no request modification or merging needed handleAsync(channel, event, baseRequest); @@ -585,7 +585,7 @@ public void handleAsync(HttpChannel channel) throws IOException, ServletExceptio try { baseRequest.resetParameters(); - HttpURI newUri = providedUri == null ? new HttpURI(oldUri) : providedUri; + HttpURI newUri = baseUri == null ? new HttpURI(oldUri) : baseUri; if (encodedPathQuery == null) { baseRequest.setHttpURI(newUri);