Issue #3888 - Adding HttpClient tests
+ Also applying changes from review Signed-off-by: Joakim Erdfelt <joakim.erdfelt@gmail.com>
This commit is contained in:
parent
122f994aaf
commit
6fd8aeefde
|
@ -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
|
||||||
{
|
{
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue