| Index: components/cronet/android/test/javatests/src/org/chromium/net/urlconnection/CronetBufferedOutputStreamTest.java
|
| diff --git a/components/cronet/android/test/javatests/src/org/chromium/net/urlconnection/CronetBufferedOutputStreamTest.java b/components/cronet/android/test/javatests/src/org/chromium/net/urlconnection/CronetBufferedOutputStreamTest.java
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..f95045a4cf0f79e2493b411b2f9449c430620af6
|
| --- /dev/null
|
| +++ b/components/cronet/android/test/javatests/src/org/chromium/net/urlconnection/CronetBufferedOutputStreamTest.java
|
| @@ -0,0 +1,289 @@
|
| +// Copyright 2015 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +package org.chromium.net.urlconnection;
|
| +
|
| +import android.test.suitebuilder.annotation.SmallTest;
|
| +
|
| +import org.chromium.base.test.util.Feature;
|
| +
|
| +import org.chromium.net.CronetTestActivity;
|
| +import org.chromium.net.CronetTestBase;
|
| +import org.chromium.net.NativeTestServer;
|
| +import org.chromium.net.UploadDataSink;
|
| +import org.chromium.net.UrlRequestContext;
|
| +
|
| +import java.io.ByteArrayOutputStream;
|
| +import java.io.InputStream;
|
| +import java.io.OutputStream;
|
| +import java.net.HttpURLConnection;
|
| +import java.net.URL;
|
| +import java.nio.ByteBuffer;
|
| +import java.util.Arrays;
|
| +
|
| +/**
|
| + * Tests the CronetBufferedOutputStream implementation.
|
| + */
|
| +public class CronetBufferedOutputStreamTest extends CronetTestBase {
|
| + private static final String UPLOAD_DATA_STRING = "Nifty upload data!";
|
| + private static final byte[] UPLOAD_DATA = UPLOAD_DATA_STRING.getBytes();
|
| + private static final int REPEAT_COUNT = 100000;
|
| + private CronetTestActivity mActivity;
|
| + private UrlRequestContext mUrlRequestContext;
|
| +
|
| + @Override
|
| + protected void setUp() throws Exception {
|
| + super.setUp();
|
| + mActivity = launchCronetTestApp();
|
| + mUrlRequestContext = mActivity.mUrlRequestContext;
|
| + assertTrue(NativeTestServer.startNativeTestServer(
|
| + getInstrumentation().getTargetContext()));
|
| + }
|
| +
|
| + @Override
|
| + protected void tearDown() throws Exception {
|
| + NativeTestServer.shutdownNativeTestServer();
|
| + super.tearDown();
|
| + }
|
| +
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @CompareDefaultWithCronet
|
| + public void testPostAfterConnectionMade() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + HttpURLConnection connection =
|
| + (HttpURLConnection) url.openConnection();
|
| + connection.setDoOutput(true);
|
| + connection.setRequestMethod("POST");
|
| + assertEquals(200, connection.getResponseCode());
|
| + try {
|
| + connection.getOutputStream();
|
| + fail();
|
| + } catch (java.net.ProtocolException e) {
|
| + // Expected.
|
| + }
|
| + }
|
| +
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @CompareDefaultWithCronet
|
| + public void testPostWithContentLength() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + HttpURLConnection connection =
|
| + (HttpURLConnection) url.openConnection();
|
| + connection.setDoOutput(true);
|
| + connection.setRequestMethod("POST");
|
| + connection.setRequestProperty("Content-Length",
|
| + Integer.toString(UPLOAD_DATA.length * REPEAT_COUNT));
|
| + byte[] largeData = new byte[REPEAT_COUNT * UPLOAD_DATA.length];
|
| + for (int i = 0; i < REPEAT_COUNT; i++) {
|
| + for (int j = 0; j < UPLOAD_DATA.length; j++) {
|
| + largeData[i * UPLOAD_DATA.length + j] = UPLOAD_DATA[j];
|
| + }
|
| + }
|
| + OutputStream out = connection.getOutputStream();
|
| + int totalBytesWritten = 0;
|
| + // Number of bytes to write each time. It is incremented by one from 0.
|
| + int bytesToWrite = 0;
|
| + while (totalBytesWritten < largeData.length) {
|
| + if (bytesToWrite > largeData.length - totalBytesWritten) {
|
| + // Do not write out of bound.
|
| + bytesToWrite = largeData.length - totalBytesWritten;
|
| + }
|
| + out.write(largeData, totalBytesWritten, bytesToWrite);
|
| + totalBytesWritten += bytesToWrite;
|
| + bytesToWrite++;
|
| + }
|
| + assertEquals(200, connection.getResponseCode());
|
| + assertEquals("OK", connection.getResponseMessage());
|
| + checkLargeData(getResponseAsString(connection));
|
| + connection.disconnect();
|
| + }
|
| +
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @CompareDefaultWithCronet
|
| + public void testPostWithContentLengthWriteOneByte() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + HttpURLConnection connection =
|
| + (HttpURLConnection) url.openConnection();
|
| + connection.setDoOutput(true);
|
| + connection.setRequestMethod("POST");
|
| + connection.setRequestProperty("Content-Length",
|
| + Integer.toString(UPLOAD_DATA.length * REPEAT_COUNT));
|
| + OutputStream out = connection.getOutputStream();
|
| + for (int i = 0; i < REPEAT_COUNT; i++) {
|
| + for (int j = 0; j < UPLOAD_DATA.length; j++) {
|
| + // Write one byte at a time.
|
| + out.write(UPLOAD_DATA[j]);
|
| + }
|
| + }
|
| + assertEquals(200, connection.getResponseCode());
|
| + assertEquals("OK", connection.getResponseMessage());
|
| + checkLargeData(getResponseAsString(connection));
|
| + connection.disconnect();
|
| + }
|
| +
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @CompareDefaultWithCronet
|
| + public void testPostWithZeroContentLength() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + HttpURLConnection connection =
|
| + (HttpURLConnection) url.openConnection();
|
| + connection.setDoOutput(true);
|
| + connection.setRequestMethod("POST");
|
| + connection.setRequestProperty("Content-Length", "0");
|
| + assertEquals(200, connection.getResponseCode());
|
| + assertEquals("OK", connection.getResponseMessage());
|
| + assertEquals("", getResponseAsString(connection));
|
| + connection.disconnect();
|
| + }
|
| +
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @CompareDefaultWithCronet
|
| + public void testPostWithoutContentLength() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + HttpURLConnection connection =
|
| + (HttpURLConnection) url.openConnection();
|
| + connection.setDoOutput(true);
|
| + connection.setRequestMethod("POST");
|
| + OutputStream out = connection.getOutputStream();
|
| + for (int i = 0; i < REPEAT_COUNT; i++) {
|
| + out.write(UPLOAD_DATA);
|
| + }
|
| + assertEquals(200, connection.getResponseCode());
|
| + assertEquals("OK", connection.getResponseMessage());
|
| + checkLargeData(getResponseAsString(connection));
|
| + connection.disconnect();
|
| + }
|
| +
|
| + /**
|
| + * Tests that if caller writes more than the content length provided,
|
| + * an exception should occur.
|
| + */
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @CompareDefaultWithCronet
|
| + public void testWriteBufferOutOfBound() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + HttpURLConnection connection =
|
| + (HttpURLConnection) url.openConnection();
|
| + connection.setDoOutput(true);
|
| + connection.setRequestMethod("POST");
|
| + // Use a content length that is 1 byte shorter than actual data.
|
| + connection.setRequestProperty("Content-Length",
|
| + Integer.toString(UPLOAD_DATA.length - 1));
|
| + OutputStream out = connection.getOutputStream();
|
| + try {
|
| + out.write(UPLOAD_DATA);
|
| + fail();
|
| + } catch (java.net.ProtocolException e) {
|
| + assertEquals("exceeded content-length limit of "
|
| + + (UPLOAD_DATA.length - 1) + " bytes", e.getMessage());
|
| + }
|
| + }
|
| +
|
| + /**
|
| + * Same as {@code testWriteBufferOutOfBound()}, but it only writes one byte
|
| + * at a time.
|
| + */
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @CompareDefaultWithCronet
|
| + public void testWriteBufferOutOfBoundWriteOneByte() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + HttpURLConnection connection =
|
| + (HttpURLConnection) url.openConnection();
|
| + connection.setDoOutput(true);
|
| + connection.setRequestMethod("POST");
|
| + // Use a content length that is 1 byte shorter than actual data.
|
| + connection.setRequestProperty("Content-Length",
|
| + Integer.toString(UPLOAD_DATA.length - 1));
|
| + OutputStream out = connection.getOutputStream();
|
| + try {
|
| + for (int i = 0; i < UPLOAD_DATA.length; i++) {
|
| + out.write(UPLOAD_DATA[i]);
|
| + }
|
| + fail();
|
| + } catch (java.net.ProtocolException e) {
|
| + assertEquals("exceeded content-length limit of "
|
| + + (UPLOAD_DATA.length - 1) + " bytes", e.getMessage());
|
| + }
|
| + }
|
| +
|
| + /**
|
| + * Tests that {@link CronetBufferedOutputStream} supports rewind.
|
| + */
|
| + @SmallTest
|
| + @Feature({"Cronet"})
|
| + @OnlyRunCronetHttpURLConnection
|
| + public void testRewind() throws Exception {
|
| + URL url = new URL(NativeTestServer.getEchoBodyURL());
|
| + CronetHttpURLConnection conn = new CronetHttpURLConnection(url,
|
| + mUrlRequestContext);
|
| + CronetBufferedOutputStream out =
|
| + new CronetBufferedOutputStream(conn, UPLOAD_DATA.length);
|
| + out.write(UPLOAD_DATA);
|
| + TestUploadDataSink sink = new TestUploadDataSink();
|
| + out.rewind(sink);
|
| + assertTrue(sink.mRewindSucceededCalled);
|
| + ByteBuffer byteBuffer = ByteBuffer.allocate(UPLOAD_DATA.length);
|
| + out.read(sink, byteBuffer);
|
| + assertTrue(sink.mReadSucceededCalled);
|
| + assertTrue(Arrays.equals(UPLOAD_DATA, byteBuffer.array()));
|
| + }
|
| +
|
| + private static class TestUploadDataSink implements UploadDataSink {
|
| + boolean mReadSucceededCalled = false;
|
| + boolean mRewindSucceededCalled = false;
|
| +
|
| + @Override
|
| + public void onReadSucceeded(boolean finalChunk) {
|
| + mReadSucceededCalled = true;
|
| + }
|
| +
|
| + @Override
|
| + public void onReadError(Exception exception) {
|
| + // Left blank.
|
| + }
|
| +
|
| + @Override
|
| + public void onRewindSucceeded() {
|
| + mRewindSucceededCalled = true;
|
| + }
|
| +
|
| + @Override
|
| + public void onRewindError(Exception e) {
|
| + // Left blank.
|
| + }
|
| + }
|
| +
|
| + /**
|
| + * Helper method to extract response body as a string for testing.
|
| + */
|
| + private String getResponseAsString(HttpURLConnection connection)
|
| + throws Exception {
|
| + InputStream in = connection.getInputStream();
|
| + ByteArrayOutputStream out = new ByteArrayOutputStream();
|
| + int b;
|
| + while ((b = in.read()) != -1) {
|
| + out.write(b);
|
| + }
|
| + return out.toString();
|
| + }
|
| +
|
| + /**
|
| + * Helper function to check whether {@code data} is a concatenation of
|
| + * {@code REPEAT_COUNT} {@code UPLOAD_DATA_STRING} strings.
|
| + */
|
| + private void checkLargeData(String data) {
|
| + for (int i = 0; i < REPEAT_COUNT; i++) {
|
| + assertEquals(UPLOAD_DATA_STRING, data.substring(
|
| + UPLOAD_DATA_STRING.length() * i,
|
| + UPLOAD_DATA_STRING.length() * (i + 1)));
|
| + }
|
| + }
|
| +}
|
|
|