Skip to main content
aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--jetty-server/src/main/java/org/eclipse/jetty/server/HttpConnection.java3
-rw-r--r--jetty-server/src/main/java/org/eclipse/jetty/server/HttpInput.java10
-rw-r--r--jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncIOServletTest.java79
3 files changed, 85 insertions, 7 deletions
diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/HttpConnection.java b/jetty-server/src/main/java/org/eclipse/jetty/server/HttpConnection.java
index 69bb3072af..5d4ce4de6b 100644
--- a/jetty-server/src/main/java/org/eclipse/jetty/server/HttpConnection.java
+++ b/jetty-server/src/main/java/org/eclipse/jetty/server/HttpConnection.java
@@ -366,7 +366,8 @@ public class HttpConnection extends AbstractConnection implements Runnable, Http
if (LOG.isDebugEnabled())
LOG.debug("unconsumed input {}", this);
// Complete reading the request
- _channel.getRequest().getHttpInput().consumeAll();
+ if (!_channel.getRequest().getHttpInput().consumeAll())
+ _channel.abort();
}
}
diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/HttpInput.java b/jetty-server/src/main/java/org/eclipse/jetty/server/HttpInput.java
index a906a9b9c2..77456ca5e1 100644
--- a/jetty-server/src/main/java/org/eclipse/jetty/server/HttpInput.java
+++ b/jetty-server/src/main/java/org/eclipse/jetty/server/HttpInput.java
@@ -277,10 +277,14 @@ public abstract class HttpInput<T> extends ServletInputStream implements Runnabl
_channelState.onReadPossible();
}
- public void consumeAll()
+ public boolean consumeAll()
{
synchronized (lock())
- {
+ {
+ // Don't bother reading if we already know there was an error.
+ if (_onError != null)
+ return false;
+
try
{
while (!isFinished())
@@ -291,10 +295,12 @@ public abstract class HttpInput<T> extends ServletInputStream implements Runnabl
else
consume(item, remaining(item));
}
+ return true;
}
catch (IOException e)
{
LOG.debug(e);
+ return false;
}
}
}
diff --git a/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncIOServletTest.java b/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncIOServletTest.java
index 194a39898c..b6d6487e51 100644
--- a/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncIOServletTest.java
+++ b/jetty-servlet/src/test/java/org/eclipse/jetty/servlet/AsyncIOServletTest.java
@@ -18,9 +18,6 @@
package org.eclipse.jetty.servlet;
-import static org.hamcrest.Matchers.*;
-import static org.junit.Assert.assertThat;
-
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
@@ -30,10 +27,10 @@ import java.nio.charset.StandardCharsets;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
-
import javax.servlet.AsyncContext;
import javax.servlet.ReadListener;
import javax.servlet.ServletException;
+import javax.servlet.ServletInputStream;
import javax.servlet.ServletOutputStream;
import javax.servlet.WriteListener;
import javax.servlet.http.HttpServlet;
@@ -48,6 +45,13 @@ import org.junit.After;
import org.junit.Assert;
import org.junit.Test;
+import static org.hamcrest.Matchers.containsString;
+import static org.hamcrest.Matchers.instanceOf;
+import static org.hamcrest.Matchers.is;
+import static org.hamcrest.Matchers.not;
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertThat;
+
public class AsyncIOServletTest
{
private Server server;
@@ -148,6 +152,73 @@ public class AsyncIOServletTest
}
@Test
+ public void testAsyncReadIdleTimeout() throws Exception
+ {
+ final int status = 567;
+ startServer(new HttpServlet()
+ {
+ @Override
+ protected void service(HttpServletRequest request, final HttpServletResponse response) throws ServletException, IOException
+ {
+ final AsyncContext asyncContext = request.startAsync(request, response);
+ asyncContext.setTimeout(0);
+ final ServletInputStream inputStream = request.getInputStream();
+ inputStream.setReadListener(new ReadListener()
+ {
+ @Override
+ public void onDataAvailable() throws IOException
+ {
+ while (inputStream.isReady() && !inputStream.isFinished())
+ inputStream.read();
+ }
+
+ @Override
+ public void onAllDataRead() throws IOException
+ {
+ }
+
+ @Override
+ public void onError(Throwable t)
+ {
+ response.setStatus(status);
+ // Do not put Connection: close header here, the test
+ // verifies that the server closes no matter what.
+ asyncContext.complete();
+ }
+ });
+ }
+ });
+ server.stop();
+ long idleTimeout = 1000;
+ connector.setIdleTimeout(idleTimeout);
+ server.start();
+
+ String data1 = "0123456789";
+ String data2 = "ABCDEF";
+ // Only send the first chunk of data and then let it idle timeout.
+ String request = "GET " + path + " HTTP/1.1\r\n" +
+ "Host: localhost:" + connector.getLocalPort() + "\r\n" +
+ "Content-Length: " + (data1.length() + data2.length()) + "\r\n" +
+ "\r\n" +
+ data1;
+
+ try (Socket client = new Socket("localhost", connector.getLocalPort()))
+ {
+ OutputStream output = client.getOutputStream();
+ output.write(request.getBytes("UTF-8"));
+ output.flush();
+
+ SimpleHttpParser parser = new SimpleHttpParser();
+ SimpleHttpResponse response = parser.readResponse(new BufferedReader(new InputStreamReader(client.getInputStream(), "UTF-8")));
+
+ assertEquals(String.valueOf(status), response.getCode());
+
+ // Make sure the connection was closed by the server.
+ assertEquals(-1, client.getInputStream().read());
+ }
+ }
+
+ @Test
public void testOnErrorThrows() throws Exception
{
final AtomicInteger errors = new AtomicInteger();

Back to the top