/*
* JBoss, Home of Professional Open Source.
* Copyright 2014 Red Hat, Inc., and individual contributors
* as indicated by the @author tags.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.undertow.servlet.test.streams;
import java.io.IOException;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import io.undertow.servlet.ServletExtension;
import io.undertow.servlet.api.DeploymentInfo;
import io.undertow.servlet.api.ServletInfo;
import io.undertow.servlet.test.util.DeploymentUtils;
import io.undertow.testutils.DefaultServer;
import io.undertow.testutils.HttpClientUtils;
import io.undertow.testutils.TestHttpClient;
import io.undertow.util.StatusCodes;
import org.apache.http.HttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.junit.Assert;
import org.junit.BeforeClass;
import org.junit.Test;
import org.junit.runner.RunWith;
/**
* @author Stuart Douglas
*/
@RunWith(DefaultServer.class)
public class ServletOutputStreamTestCase {
public static String message;
public static final String HELLO_WORLD = "Hello World";
public static final String BLOCKING_SERVLET = "blockingOutput";
public static final String ASYNC_SERVLET = "asyncOutput";
public static final String CONTENT_LENGTH_SERVLET = "contentLength";
public static final String RESET = "reset";
public static final String START = "START";
public static final String END = "END";
@BeforeClass
public static void setup() throws ServletException {
DeploymentUtils.setupServlet(new ServletExtension() {
@Override
public void handleDeployment(DeploymentInfo deploymentInfo, ServletContext servletContext) {
deploymentInfo.setIgnoreFlush(false);
}
},
new ServletInfo(BLOCKING_SERVLET, BlockingOutputStreamServlet.class)
.addMapping("/" + BLOCKING_SERVLET),
new ServletInfo(ASYNC_SERVLET, AsyncOutputStreamServlet.class)
.addMapping("/" + ASYNC_SERVLET)
.setAsyncSupported(true),
new ServletInfo(CONTENT_LENGTH_SERVLET, ContentLengthCloseFlushServlet.class)
.addMapping("/" + CONTENT_LENGTH_SERVLET),
new ServletInfo(RESET, ResetBufferServlet.class).addMapping("/" + RESET));
}
@Test
public void testFlushAndCloseWithContentLength() throws Exception {
TestHttpClient client = new TestHttpClient();
try {
String uri = DefaultServer.getDefaultServerURL() + "/servletContext/" + CONTENT_LENGTH_SERVLET;
HttpGet get = new HttpGet(uri);
HttpResponse result = client.execute(get);
Assert.assertEquals(StatusCodes.OK, result.getStatusLine().getStatusCode());
String response = HttpClientUtils.readResponse(result);
Assert.assertEquals("a", response);
get = new HttpGet(uri);
result = client.execute(get);
Assert.assertEquals(StatusCodes.OK, result.getStatusLine().getStatusCode());
response = HttpClientUtils.readResponse(result);
Assert.assertEquals("OK", response);
} finally {
client.getConnectionManager().shutdown();
}
}
@Test
public void testResetBuffer() throws Exception {
TestHttpClient client = new TestHttpClient();
try {
String uri = DefaultServer.getDefaultServerURL() + "/servletContext/" + RESET;
HttpGet get = new HttpGet(uri);
HttpResponse result = client.execute(get);
Assert.assertEquals(StatusCodes.OK, result.getStatusLine().getStatusCode());
String response = HttpClientUtils.readResponse(result);
Assert.assertEquals("hello world", response);
} finally {
client.getConnectionManager().shutdown();
}
}
@Test
public void testBlockingServletOutputStream() throws IOException {
message = START + HELLO_WORLD + END;
runTest(message, BLOCKING_SERVLET, false, true, 1, true, false);
StringBuilder builder = new StringBuilder(1000 * HELLO_WORLD.length());
builder.append(START);
for (int i = 0; i < 10; ++i) {
try {
for (int j = 0; j < 1000; ++j) {
builder.append(HELLO_WORLD);
}
String message = builder.toString() + END;
runTest(message, BLOCKING_SERVLET, false, false, 1, false, false);
runTest(message, BLOCKING_SERVLET, true, false, 10, false, false);
runTest(message, BLOCKING_SERVLET, false, true, 3, false, false);
runTest(message, BLOCKING_SERVLET, true, true, 7, false, false);
} catch (Throwable e) {
throw new RuntimeException("test failed with i equal to " + i, e);
}
}
}
@Test
public void testChunkedResponseWithInitialFlush() throws IOException {
message = START + HELLO_WORLD + END;
runTest(message, BLOCKING_SERVLET, false, true, 1, true, false);
}
@Test
public void testAsyncServletOutputStream() {
StringBuilder builder = new StringBuilder(1000 * HELLO_WORLD.length());
builder.append(START);
for (int i = 0; i < 10; ++i) {
try {
for (int j = 0; j < 10000; ++j) {
builder.append(HELLO_WORLD);
}
String message = builder.toString() + END;
runTest(message, ASYNC_SERVLET, false, false, 1, false, false);
runTest(message, ASYNC_SERVLET, true, false, 10, false, false);
runTest(message, ASYNC_SERVLET, false, true, 3, false, false);
runTest(message, ASYNC_SERVLET, true, true, 7, false, false);
} catch (Exception e) {
throw new RuntimeException("test failed with i equal to " + i, e);
}
}
}
@Test
public void testAsyncServletOutputStreamWithPreable() {
StringBuilder builder = new StringBuilder(1000 * HELLO_WORLD.length());
builder.append(START);
for (int i = 0; i < 10; ++i) {
try {
for (int j = 0; j < 10000; ++j) {
builder.append(HELLO_WORLD);
}
String message = builder.toString() + END;
runTest(message, ASYNC_SERVLET, false, false, 1, false, true);
runTest(message, ASYNC_SERVLET, true, false, 10, false, true);
runTest(message, ASYNC_SERVLET, false, true, 3, false, true);
runTest(message, ASYNC_SERVLET, true, true, 7, false, true);
} catch (Exception e) {
throw new RuntimeException("test failed with i equal to " + i, e);
}
}
}
public void runTest(final String message, String url, final boolean flush, final boolean close, int reps, boolean initialFlush, boolean writePreable) throws IOException {
TestHttpClient client = new TestHttpClient();
try {
ServletOutputStreamTestCase.message = message;
String uri = DefaultServer.getDefaultServerURL() + "/servletContext/" + url + "?reps=" + reps + "&";
if (flush) {
uri = uri + "flush=true&";
}
if (close) {
uri = uri + "close=true&";
}
if(initialFlush) {
uri = uri + "initialFlush=true&";
}
if(writePreable) {
uri = uri + "preamble=true&";
}
HttpGet get = new HttpGet(uri);
HttpResponse result = client.execute(get);
Assert.assertEquals(StatusCodes.OK, result.getStatusLine().getStatusCode());
StringBuilder builder = new StringBuilder(reps * message.length());
for (int j = 0; j < reps; ++j) {
builder.append(message);
}
if(writePreable) {
builder.append(builder.toString()); //content gets written twice in this case
}
final String response = HttpClientUtils.readResponse(result);
String expected = builder.toString();
Assert.assertTrue("Must start with START", response.startsWith(START));
Assert.assertTrue("Must end with END", response.endsWith(END));
Assert.assertEquals(expected.length(), response.length());
Assert.assertEquals(expected, response);
} finally {
client.getConnectionManager().shutdown();
}
}
}