ChunkingTest.java
/*
* Copyright (c) 2010-2012 Sonatype, Inc. All rights reserved.
*
* This program is licensed to you under the Apache License Version 2.0,
* and you may not use this file except in compliance with the Apache License Version 2.0.
* You may obtain a copy of the Apache License Version 2.0 at http://www.apache.org/licenses/LICENSE-2.0.
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the Apache License Version 2.0 is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the Apache License Version 2.0 for the specific language governing permissions and limitations there under.
*/
package org.asynchttpclient.request.body;
import io.github.artsok.RepeatedIfExceptionsTest;
import io.netty.buffer.Unpooled;
import org.asynchttpclient.AbstractBasicTest;
import org.asynchttpclient.AsyncHttpClient;
import org.asynchttpclient.DefaultAsyncHttpClientConfig;
import org.asynchttpclient.ListenableFuture;
import org.asynchttpclient.Request;
import org.asynchttpclient.Response;
import org.asynchttpclient.request.body.generator.FeedableBodyGenerator;
import org.asynchttpclient.request.body.generator.InputStreamBodyGenerator;
import org.asynchttpclient.request.body.generator.UnboundedQueueFeedableBodyGenerator;
import java.io.BufferedInputStream;
import java.io.InputStream;
import java.nio.file.Files;
import java.time.Duration;
import java.util.concurrent.ExecutionException;
import static org.asynchttpclient.Dsl.asyncHttpClient;
import static org.asynchttpclient.Dsl.config;
import static org.asynchttpclient.Dsl.post;
import static org.asynchttpclient.test.TestUtils.LARGE_IMAGE_BYTES;
import static org.asynchttpclient.test.TestUtils.LARGE_IMAGE_FILE;
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
public class ChunkingTest extends AbstractBasicTest {
// So we can just test the returned data is the image,
// and doesn't contain the chunked delimiters.
@RepeatedIfExceptionsTest(repeats = 5)
public void testBufferLargerThanFileWithStreamBodyGenerator() throws Throwable {
doTestWithInputStreamBodyGenerator(new BufferedInputStream(Files.newInputStream(LARGE_IMAGE_FILE.toPath()), 400000));
}
@RepeatedIfExceptionsTest(repeats = 5)
public void testBufferSmallThanFileWithStreamBodyGenerator() throws Throwable {
doTestWithInputStreamBodyGenerator(new BufferedInputStream(Files.newInputStream(LARGE_IMAGE_FILE.toPath())));
}
@RepeatedIfExceptionsTest(repeats = 5)
public void testDirectFileWithStreamBodyGenerator() throws Throwable {
doTestWithInputStreamBodyGenerator(Files.newInputStream(LARGE_IMAGE_FILE.toPath()));
}
@RepeatedIfExceptionsTest(repeats = 5)
public void testDirectFileWithFeedableBodyGenerator() throws Throwable {
doTestWithFeedableBodyGenerator(Files.newInputStream(LARGE_IMAGE_FILE.toPath()));
}
private void doTestWithInputStreamBodyGenerator(InputStream is) throws Throwable {
try {
try (AsyncHttpClient c = asyncHttpClient(httpClientBuilder())) {
ListenableFuture<Response> responseFuture = c.executeRequest(post(getTargetUrl()).setBody(new InputStreamBodyGenerator(is)));
waitForAndAssertResponse(responseFuture);
}
} finally {
is.close();
}
}
private void doTestWithFeedableBodyGenerator(InputStream is) throws Throwable {
try {
try (AsyncHttpClient c = asyncHttpClient(httpClientBuilder())) {
final FeedableBodyGenerator feedableBodyGenerator = new UnboundedQueueFeedableBodyGenerator();
Request r = post(getTargetUrl()).setBody(feedableBodyGenerator).build();
ListenableFuture<Response> responseFuture = c.executeRequest(r);
feed(feedableBodyGenerator, is);
waitForAndAssertResponse(responseFuture);
}
} finally {
is.close();
}
}
private static void feed(FeedableBodyGenerator feedableBodyGenerator, InputStream is) throws Exception {
try (InputStream inputStream = is) {
byte[] buffer = new byte[512];
for (int i; (i = inputStream.read(buffer)) > -1; ) {
byte[] chunk = new byte[i];
System.arraycopy(buffer, 0, chunk, 0, i);
feedableBodyGenerator.feed(Unpooled.wrappedBuffer(chunk), false);
}
}
feedableBodyGenerator.feed(Unpooled.EMPTY_BUFFER, true);
}
private static DefaultAsyncHttpClientConfig.Builder httpClientBuilder() {
return config()
.setKeepAlive(true)
.setMaxConnectionsPerHost(1)
.setMaxConnections(1)
.setConnectTimeout(Duration.ofSeconds(1))
.setRequestTimeout(Duration.ofSeconds(1))
.setFollowRedirect(true);
}
private static void waitForAndAssertResponse(ListenableFuture<Response> responseFuture) throws InterruptedException, ExecutionException {
Response response = responseFuture.get();
if (500 == response.getStatusCode()) {
logger.debug("==============\n" +
"500 response from call\n" +
"Headers:" + response.getHeaders() + '\n' +
"==============\n");
assertEquals(500, response.getStatusCode(), "Should have 500 status code");
assertTrue(response.getHeader("X-Exception").contains("invalid.chunk.length"), "Should have failed due to chunking");
fail("HARD Failing the test due to provided InputStreamBodyGenerator, chunking incorrectly:" + response.getHeader("X-Exception"));
} else {
assertArrayEquals(LARGE_IMAGE_BYTES, response.getResponseBodyAsBytes());
}
}
}