Issue #3888 - Adding HttpClient tests

+ Also applying changes from review

Signed-off-by: Joakim Erdfelt <joakim.erdfelt@gmail.com>
This commit is contained in:
Joakim Erdfelt 2019-07-23 15:24:33 -05:00
parent 122f994aaf
commit 6fd8aeefde
5 changed files with 222 additions and 68 deletions

View File

@ -125,7 +125,7 @@ public class ResourceHttpContent implements HttpContent
@Override @Override
public ByteBuffer getDirectBuffer() public ByteBuffer getDirectBuffer()
{ {
if (_resource.length() <= 0 || _maxBuffer > 0 && _maxBuffer < _resource.length()) if (_resource.length() <= 0 || _maxBuffer > 0 && _resource.length() > _maxBuffer)
return null; return null;
try try
{ {
@ -152,7 +152,7 @@ public class ResourceHttpContent implements HttpContent
@Override @Override
public ByteBuffer getIndirectBuffer() public ByteBuffer getIndirectBuffer()
{ {
if (_resource.length() <= 0 || _maxBuffer > 0 && _maxBuffer < _resource.length()) if (_resource.length() <= 0 || _maxBuffer > 0 && _resource.length() > _maxBuffer)
return null; return null;
try try
{ {

View File

@ -335,13 +335,14 @@ public class CachedContentFactory implements HttpContent.ContentFactory
{ {
try try
{ {
return BufferUtil.toBuffer(resource, true); return BufferUtil.toBuffer(resource, false);
} }
catch (IOException | IllegalArgumentException e) catch (IOException | IllegalArgumentException e)
{ {
LOG.warn(e); if (LOG.isDebugEnabled())
return null; LOG.debug(e);
} }
return null;
} }
protected ByteBuffer getMappedBuffer(Resource resource) protected ByteBuffer getMappedBuffer(Resource resource)
@ -355,7 +356,8 @@ public class CachedContentFactory implements HttpContent.ContentFactory
} }
catch (IOException | IllegalArgumentException e) catch (IOException | IllegalArgumentException e)
{ {
LOG.warn(e); if (LOG.isDebugEnabled())
LOG.debug(e);
} }
return null; return null;
} }
@ -368,7 +370,8 @@ public class CachedContentFactory implements HttpContent.ContentFactory
} }
catch (IOException | IllegalArgumentException e) catch (IOException | IllegalArgumentException e)
{ {
LOG.warn(e); if (LOG.isDebugEnabled())
LOG.debug(e);
} }
return null; return null;
} }
@ -552,7 +555,7 @@ public class CachedContentFactory implements HttpContent.ContentFactory
@Override @Override
public ByteBuffer getIndirectBuffer() public ByteBuffer getIndirectBuffer()
{ {
if (_resource.length() > (long)_maxCachedFileSize) if (_resource.length() > _maxCachedFileSize)
{ {
return null; return null;
} }
@ -561,21 +564,25 @@ public class CachedContentFactory implements HttpContent.ContentFactory
if (buffer == null) if (buffer == null)
{ {
ByteBuffer buffer2 = CachedContentFactory.this.getIndirectBuffer(_resource); ByteBuffer buffer2 = CachedContentFactory.this.getIndirectBuffer(_resource);
if (buffer2 == null) if (buffer2 == null)
LOG.warn("Could not load indirect buffer from " + this); {
else if (_indirectBuffer.compareAndSet(null, buffer2)) if (LOG.isDebugEnabled())
LOG.debug("Could not load indirect buffer from " + this);
return null;
}
if (_indirectBuffer.compareAndSet(null, buffer2))
{ {
buffer = buffer2; buffer = buffer2;
if (_cachedSize.addAndGet(BufferUtil.length(buffer)) > _maxCacheSize) if (_cachedSize.addAndGet(BufferUtil.length(buffer)) > _maxCacheSize)
shrinkCache(); shrinkCache();
} }
else else
{
buffer = _indirectBuffer.get(); buffer = _indirectBuffer.get();
}
} }
if (buffer == null) return buffer == null ? null : buffer.asReadOnlyBuffer();
return null;
return buffer.slice();
} }
@Override @Override
@ -594,7 +601,7 @@ public class CachedContentFactory implements HttpContent.ContentFactory
else else
buffer = _mappedBuffer.get(); buffer = _mappedBuffer.get();
} }
else if (_resource.length() <= (long)_maxCachedFileSize) else if (_resource.length() > _maxCachedFileSize)
{ {
ByteBuffer direct = CachedContentFactory.this.getDirectBuffer(_resource); ByteBuffer direct = CachedContentFactory.this.getDirectBuffer(_resource);
if (direct != null) if (direct != null)
@ -612,7 +619,8 @@ public class CachedContentFactory implements HttpContent.ContentFactory
} }
else else
{ {
LOG.warn("Could not load " + this); if (LOG.isDebugEnabled())
LOG.debug("Could not load " + this);
} }
} }
} }

View File

@ -681,17 +681,21 @@ public class Request implements HttpServletRequest
MetaData.Request metadata = _metaData; MetaData.Request metadata = _metaData;
if (metadata == null) if (metadata == null)
return -1; return -1;
if (metadata.getContentLength() != Long.MIN_VALUE)
long contentLength = metadata.getContentLength();
if (contentLength != Long.MIN_VALUE)
{ {
if (metadata.getContentLength() > (long)Integer.MAX_VALUE) if (contentLength > Integer.MAX_VALUE)
{ {
// Per ServletRequest#getContentLength() javadoc this must return -1 for values exceeding Integer.MAX_VALUE // Per ServletRequest#getContentLength() javadoc this must return -1 for values exceeding Integer.MAX_VALUE
return -1; return -1;
} }
else else
return (int)metadata.getContentLength(); {
return (int)contentLength;
}
} }
return (int)metadata.getFields().getLongField(HttpHeader.CONTENT_LENGTH.toString()); return (int)metadata.getFields().getLongField(HttpHeader.CONTENT_LENGTH.asString());
} }
/* /*
@ -705,7 +709,7 @@ public class Request implements HttpServletRequest
return -1L; return -1L;
if (metadata.getContentLength() != Long.MIN_VALUE) if (metadata.getContentLength() != Long.MIN_VALUE)
return metadata.getContentLength(); return metadata.getContentLength();
return metadata.getFields().getLongField(HttpHeader.CONTENT_LENGTH.toString()); return metadata.getFields().getLongField(HttpHeader.CONTENT_LENGTH.asString());
} }
public long getContentRead() public long getContentRead()

View File

@ -52,11 +52,6 @@
<artifactId>jetty-xml</artifactId> <artifactId>jetty-xml</artifactId>
<version>${project.version}</version> <version>${project.version}</version>
</dependency> </dependency>
<dependency>
<groupId>org.eclipse.jetty.toolchain</groupId>
<artifactId>jetty-test-helper</artifactId>
<scope>test</scope>
</dependency>
<dependency> <dependency>
<groupId>org.eclipse.jetty</groupId> <groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-servlet</artifactId> <artifactId>jetty-servlet</artifactId>
@ -68,6 +63,17 @@
<version>${project.version}</version> <version>${project.version}</version>
<optional>true</optional> <optional>true</optional>
</dependency> </dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-client</artifactId>
<version>${project.version}</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.eclipse.jetty.toolchain</groupId>
<artifactId>jetty-test-helper</artifactId>
<scope>test</scope>
</dependency>
</dependencies> </dependencies>
<profiles> <profiles>
<profile> <profile>

View File

@ -21,19 +21,37 @@ package org.eclipse.jetty.webapp;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.io.OutputStream; import java.io.OutputStream;
import java.net.HttpURLConnection; import java.io.PrintWriter;
import java.net.URI; import java.net.URI;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import java.nio.channels.SeekableByteChannel; import java.nio.channels.SeekableByteChannel;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
import java.nio.file.StandardOpenOption; import java.nio.file.StandardOpenOption;
import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.concurrent.TimeUnit;
import java.util.stream.Stream;
import javax.servlet.MultipartConfigElement;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.client.HttpClient;
import org.eclipse.jetty.client.api.ContentResponse;
import org.eclipse.jetty.client.api.Request;
import org.eclipse.jetty.client.api.Response;
import org.eclipse.jetty.client.util.InputStreamResponseListener;
import org.eclipse.jetty.client.util.MultiPartContentProvider;
import org.eclipse.jetty.client.util.PathContentProvider;
import org.eclipse.jetty.http.HttpHeader;
import org.eclipse.jetty.http.HttpMethod;
import org.eclipse.jetty.server.Server; import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.server.ServerConnector; import org.eclipse.jetty.server.ServerConnector;
import org.eclipse.jetty.server.handler.DefaultHandler; import org.eclipse.jetty.server.handler.DefaultHandler;
import org.eclipse.jetty.server.handler.HandlerList; import org.eclipse.jetty.server.handler.HandlerList;
import org.eclipse.jetty.servlet.ServletHolder;
import org.eclipse.jetty.toolchain.test.FS; import org.eclipse.jetty.toolchain.test.FS;
import org.eclipse.jetty.toolchain.test.MavenTestingUtils; import org.eclipse.jetty.toolchain.test.MavenTestingUtils;
import org.eclipse.jetty.util.IO; import org.eclipse.jetty.util.IO;
@ -42,9 +60,12 @@ import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll; import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test; import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;
import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.is; import static org.hamcrest.Matchers.is;
public class HugeResourceTest public class HugeResourceTest
@ -54,8 +75,10 @@ public class HugeResourceTest
private static final long GB = 1024 * MB; private static final long GB = 1024 * MB;
public static Path staticBase; public static Path staticBase;
public static Path outputDir; public static Path outputDir;
public static Path multipartTempDir;
public Server server; public Server server;
public HttpClient client;
@BeforeAll @BeforeAll
public static void prepareStaticFiles() throws IOException public static void prepareStaticFiles() throws IOException
@ -69,6 +92,20 @@ public class HugeResourceTest
outputDir = MavenTestingUtils.getTargetTestingPath(HugeResourceTest.class.getSimpleName() + "-outputdir"); outputDir = MavenTestingUtils.getTargetTestingPath(HugeResourceTest.class.getSimpleName() + "-outputdir");
FS.ensureEmpty(outputDir); FS.ensureEmpty(outputDir);
multipartTempDir = MavenTestingUtils.getTargetTestingPath(HugeResourceTest.class.getSimpleName() + "-multipart-tmp");
FS.ensureEmpty(multipartTempDir);
}
public static Stream<Arguments> staticFiles()
{
ArrayList<Arguments> ret = new ArrayList<>();
ret.add(Arguments.of("test-1g.dat", 1 * GB));
ret.add(Arguments.of("test-4g.dat", 4 * GB));
ret.add(Arguments.of("test-10g.dat", 10 * GB));
return ret.stream();
} }
@AfterAll @AfterAll
@ -123,6 +160,17 @@ public class HugeResourceTest
context.setContextPath("/"); context.setContextPath("/");
context.setBaseResource(new PathResource(staticBase)); context.setBaseResource(new PathResource(staticBase));
context.addServlet(PostServlet.class, "/post");
String location = multipartTempDir.toString();
long maxFileSize = Long.MAX_VALUE;
long maxRequestSize = Long.MAX_VALUE;
int fileSizeThreshold = (int)(2 * MB);
MultipartConfigElement multipartConfig = new MultipartConfigElement(location, maxFileSize, maxRequestSize, fileSizeThreshold);
ServletHolder holder = context.addServlet(MultipartServlet.class, "/multipart");
holder.getRegistration().setMultipartConfig(multipartConfig);
HandlerList handlers = new HandlerList(); HandlerList handlers = new HandlerList();
handlers.addHandler(context); handlers.addHandler(context);
handlers.addHandler(new DefaultHandler()); handlers.addHandler(new DefaultHandler());
@ -137,67 +185,155 @@ public class HugeResourceTest
server.stop(); server.stop();
} }
@Test @BeforeEach
public void testDownload_1G() throws IOException public void startClient() throws Exception
{ {
download(server.getURI().resolve("/test-1g.dat"), 1 * GB); client = new HttpClient();
client.start();
} }
@Test @AfterEach
public void testDownload_4G() throws IOException public void stopClient() throws Exception
{ {
download(server.getURI().resolve("/test-4g.dat"), 4 * GB); client.stop();
} }
@Test @ParameterizedTest
public void testDownload_10G() throws IOException @MethodSource("staticFiles")
public void testDownload(String filename, long expectedSize) throws Exception
{ {
download(server.getURI().resolve("/test-10g.dat"), 10 * GB); URI destUri = server.getURI().resolve("/" + filename);
} InputStreamResponseListener responseListener = new InputStreamResponseListener();
private void download(URI destUri, long expectedSize) throws IOException Request request = client.newRequest(destUri)
{ .method(HttpMethod.GET);
HttpURLConnection http = (HttpURLConnection)destUri.toURL().openConnection(); request.send(responseListener);
assertThat("HTTP Response Code", http.getResponseCode(), is(200)); Response response = responseListener.get(5, TimeUnit.SECONDS);
dumpResponseHeaders(http); assertThat("HTTP Response Code", response.getStatus(), is(200));
dumpResponse(response);
// if a Content-Length is provided, test it String contentLength = response.getHeaders().get(HttpHeader.CONTENT_LENGTH);
String contentLength = http.getHeaderField("Content-Length"); long contentLengthLong = Long.parseLong(contentLength);
if (contentLength != null) assertThat("Http Response Header: \"Content-Length: " + contentLength + "\"", contentLengthLong, is(expectedSize));
{
long contentLengthLong = Long.parseLong(contentLength);
assertThat("Http Response Header: \"Content-Length: " + contentLength + "\"", contentLengthLong, is(expectedSize));
}
// Download the file
String filename = destUri.getPath();
int idx = filename.lastIndexOf('/');
if (idx >= 0)
{
filename = filename.substring(idx + 1);
}
Path outputFile = outputDir.resolve(filename); Path outputFile = outputDir.resolve(filename);
try (OutputStream out = Files.newOutputStream(outputFile); try (OutputStream out = Files.newOutputStream(outputFile);
InputStream in = http.getInputStream()) InputStream in = responseListener.getInputStream())
{ {
IO.copy(in, out); IO.copy(in, out);
} }
// Verify the file download size
assertThat("Downloaded Files Size: " + filename, Files.size(outputFile), is(expectedSize)); assertThat("Downloaded Files Size: " + filename, Files.size(outputFile), is(expectedSize));
} }
private void dumpResponseHeaders(HttpURLConnection http) @ParameterizedTest
@MethodSource("staticFiles")
public void testUpload(String filename, long expectedSize) throws Exception
{ {
int i = 0; Path inputFile = staticBase.resolve(filename);
String value;
while ((value = http.getHeaderField(i)) != null) PathContentProvider pathContentProvider = new PathContentProvider(inputFile);
URI destUri = server.getURI().resolve("/post");
Request request = client.newRequest(destUri).method(HttpMethod.POST).content(pathContentProvider);
ContentResponse response = request.send();
assertThat("HTTP Response Code", response.getStatus(), is(200));
dumpResponse(response);
String responseBody = response.getContentAsString();
assertThat("Response", responseBody, containsString("bytes-received=" + expectedSize));
}
@ParameterizedTest
@MethodSource("staticFiles")
public void testUpload_Multipart(String filename, long expectedSize) throws Exception
{
MultiPartContentProvider multipart = new MultiPartContentProvider();
Path inputFile = staticBase.resolve(filename);
String name = String.format("file-%d", expectedSize);
multipart.addFilePart(name, filename, new PathContentProvider(inputFile), null);
URI destUri = server.getURI().resolve("/multipart");
Request request = client.newRequest(destUri).method(HttpMethod.POST).content(multipart);
ContentResponse response = request.send();
assertThat("HTTP Response Code", response.getStatus(), is(200));
dumpResponse(response);
String responseBody = response.getContentAsString();
String expectedResponse = String.format("part[%s].size=%d", name, expectedSize);
assertThat("Response", responseBody, containsString(expectedResponse));
}
private void dumpResponse(Response response)
{
System.out.printf(" %s %d %s%n", response.getVersion(), response.getStatus(), response.getReason());
response.getHeaders().forEach((field) -> System.out.printf(" %s%n", field));
}
public static class ByteCountingOutputStream extends OutputStream
{
private long count = 0;
public long getCount()
{ {
String key = http.getHeaderFieldKey(i); return count;
System.err.printf(" %s: %s%n", key, value); }
i++;
@Override
public void write(int b)
{
count++;
}
@Override
public void write(byte[] b)
{
count += b.length;
}
@Override
public void write(byte[] b, int off, int len)
{
count += len;
}
}
public static class PostServlet extends HttpServlet
{
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws IOException
{
ByteCountingOutputStream byteCounting = new ByteCountingOutputStream();
IO.copy(req.getInputStream(), byteCounting);
resp.setContentType("text/plain");
resp.setCharacterEncoding("utf-8");
resp.getWriter().printf("bytes-received=%d%n", byteCounting.getCount());
}
}
public static class MultipartServlet extends HttpServlet
{
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException
{
resp.setContentType("text/plain");
resp.setCharacterEncoding("utf-8");
PrintWriter out = resp.getWriter();
req.getParts().forEach((part) ->
{
out.printf("part[%s].filename=%s%n", part.getName(), part.getSubmittedFileName());
out.printf("part[%s].size=%d%n", part.getName(), part.getSize());
try (InputStream inputStream = part.getInputStream();
ByteCountingOutputStream byteCounting = new ByteCountingOutputStream())
{
IO.copy(inputStream, byteCounting);
out.printf("part[%s].inputStream.length=%d%n", part.getName(), byteCounting.getCount());
}
catch (IOException e)
{
e.printStackTrace(out);
}
});
} }
} }
} }