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 { |
mmenke
2015/03/25 18:34:57
Maybe a zero length upload without a content lengt
xunjieli
2015/03/27 17:46:45
Done.
|
+ 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); |
mmenke
2015/03/25 18:34:57
Suggest a test like this that does two writes.
xunjieli
2015/03/27 17:46:45
Done.
|
+ 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())); |
mmenke
2015/03/25 18:34:57
Rather than doing all this manually, can we just u
xunjieli
2015/03/27 17:46:45
Done. Good idea! But the default implementation re
|
+ } |
+ |
+ 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))); |
+ } |
+ } |
+} |