| /* |
| * Copyright (C) 2013 Square, Inc. |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package com.squareup.okhttp; |
| |
| import com.squareup.okhttp.internal.RecordingHostnameVerifier; |
| import com.squareup.okhttp.internal.SslContextBuilder; |
| import com.squareup.okhttp.mockwebserver.Dispatcher; |
| import com.squareup.okhttp.mockwebserver.MockResponse; |
| import com.squareup.okhttp.mockwebserver.MockWebServer; |
| import com.squareup.okhttp.mockwebserver.RecordedRequest; |
| import com.squareup.okhttp.mockwebserver.SocketPolicy; |
| import java.io.File; |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.net.HttpURLConnection; |
| import java.util.UUID; |
| import java.util.concurrent.CountDownLatch; |
| import java.util.concurrent.atomic.AtomicReference; |
| import javax.net.ssl.SSLContext; |
| import org.junit.After; |
| import org.junit.Before; |
| import org.junit.Test; |
| |
| import static org.junit.Assert.assertEquals; |
| import static org.junit.Assert.assertNull; |
| import static org.junit.Assert.assertTrue; |
| |
| public final class AsyncApiTest { |
| private MockWebServer server = new MockWebServer(); |
| private OkHttpClient client = new OkHttpClient(); |
| private RecordingReceiver receiver = new RecordingReceiver(); |
| |
| private static final SSLContext sslContext = SslContextBuilder.localhost(); |
| private HttpResponseCache cache; |
| |
| @Before public void setUp() throws Exception { |
| String tmp = System.getProperty("java.io.tmpdir"); |
| File cacheDir = new File(tmp, "HttpCache-" + UUID.randomUUID()); |
| cache = new HttpResponseCache(cacheDir, Integer.MAX_VALUE); |
| } |
| |
| @After public void tearDown() throws Exception { |
| server.shutdown(); |
| cache.delete(); |
| } |
| |
| @Test public void get() throws Exception { |
| server.enqueue(new MockResponse() |
| .setBody("abc") |
| .addHeader("Content-Type: text/plain")); |
| server.play(); |
| |
| Request request = new Request.Builder() |
| .url(server.getUrl("/")) |
| .header("User-Agent", "AsyncApiTest") |
| .build(); |
| client.enqueue(request, receiver); |
| |
| receiver.await(request.url()) |
| .assertCode(200) |
| .assertContainsHeaders("Content-Type: text/plain") |
| .assertBody("abc"); |
| |
| assertTrue(server.takeRequest().getHeaders().contains("User-Agent: AsyncApiTest")); |
| } |
| |
| @Test public void connectionPooling() throws Exception { |
| server.enqueue(new MockResponse().setBody("abc")); |
| server.enqueue(new MockResponse().setBody("def")); |
| server.enqueue(new MockResponse().setBody("ghi")); |
| server.play(); |
| |
| client.enqueue(new Request.Builder().url(server.getUrl("/a")).build(), receiver); |
| receiver.await(server.getUrl("/a")).assertBody("abc"); |
| |
| client.enqueue(new Request.Builder().url(server.getUrl("/b")).build(), receiver); |
| receiver.await(server.getUrl("/b")).assertBody("def"); |
| |
| client.enqueue(new Request.Builder().url(server.getUrl("/c")).build(), receiver); |
| receiver.await(server.getUrl("/c")).assertBody("ghi"); |
| |
| assertEquals(0, server.takeRequest().getSequenceNumber()); |
| assertEquals(1, server.takeRequest().getSequenceNumber()); |
| assertEquals(2, server.takeRequest().getSequenceNumber()); |
| } |
| |
| @Test public void tls() throws Exception { |
| server.useHttps(sslContext.getSocketFactory(), false); |
| server.enqueue(new MockResponse() |
| .setBody("abc") |
| .addHeader("Content-Type: text/plain")); |
| server.play(); |
| |
| client.setSslSocketFactory(sslContext.getSocketFactory()); |
| client.setHostnameVerifier(new RecordingHostnameVerifier()); |
| |
| Request request = new Request.Builder() |
| .url(server.getUrl("/")) |
| .build(); |
| client.enqueue(request, receiver); |
| |
| receiver.await(request.url()).assertHandshake(); |
| } |
| |
| @Test public void recoverFromTlsHandshakeFailure() throws Exception { |
| server.useHttps(sslContext.getSocketFactory(), false); |
| server.enqueue(new MockResponse().setSocketPolicy(SocketPolicy.FAIL_HANDSHAKE)); |
| server.enqueue(new MockResponse().setBody("abc")); |
| server.play(); |
| |
| client.setSslSocketFactory(sslContext.getSocketFactory()); |
| client.setHostnameVerifier(new RecordingHostnameVerifier()); |
| |
| Request request = new Request.Builder() |
| .url(server.getUrl("/")) |
| .build(); |
| client.enqueue(request, receiver); |
| |
| receiver.await(request.url()).assertBody("abc"); |
| } |
| |
| @Test public void post() throws Exception { |
| server.enqueue(new MockResponse().setBody("abc")); |
| server.play(); |
| |
| Request request = new Request.Builder() |
| .url(server.getUrl("/")) |
| .post(Request.Body.create(MediaType.parse("text/plain"), "def")) |
| .build(); |
| client.enqueue(request, receiver); |
| |
| receiver.await(request.url()) |
| .assertCode(200) |
| .assertBody("abc"); |
| |
| RecordedRequest recordedRequest = server.takeRequest(); |
| assertEquals("def", recordedRequest.getUtf8Body()); |
| assertEquals("3", recordedRequest.getHeader("Content-Length")); |
| assertEquals("text/plain; charset=utf-8", recordedRequest.getHeader("Content-Type")); |
| } |
| |
| @Test public void conditionalCacheHit() throws Exception { |
| server.enqueue(new MockResponse().setBody("A").addHeader("ETag: v1")); |
| server.enqueue(new MockResponse() |
| .clearHeaders() |
| .setResponseCode(HttpURLConnection.HTTP_NOT_MODIFIED)); |
| server.play(); |
| |
| client.setOkResponseCache(cache); |
| |
| Request request1 = new Request.Builder() |
| .url(server.getUrl("/")) |
| .build(); |
| client.enqueue(request1, receiver); |
| receiver.await(request1.url()).assertCode(200).assertBody("A"); |
| assertNull(server.takeRequest().getHeader("If-None-Match")); |
| |
| Request request2 = new Request.Builder() |
| .url(server.getUrl("/")) |
| .build(); |
| client.enqueue(request2, receiver); |
| receiver.await(request2.url()).assertCode(200).assertBody("A"); |
| assertEquals("v1", server.takeRequest().getHeader("If-None-Match")); |
| } |
| |
| @Test public void conditionalCacheMiss() throws Exception { |
| server.enqueue(new MockResponse().setBody("A").addHeader("ETag: v1")); |
| server.enqueue(new MockResponse().setBody("B")); |
| server.play(); |
| |
| client.setOkResponseCache(cache); |
| |
| Request request1 = new Request.Builder() |
| .url(server.getUrl("/")) |
| .build(); |
| client.enqueue(request1, receiver); |
| receiver.await(request1.url()).assertCode(200).assertBody("A"); |
| assertNull(server.takeRequest().getHeader("If-None-Match")); |
| |
| Request request2 = new Request.Builder() |
| .url(server.getUrl("/")) |
| .build(); |
| client.enqueue(request2, receiver); |
| receiver.await(request2.url()).assertCode(200).assertBody("B"); |
| assertEquals("v1", server.takeRequest().getHeader("If-None-Match")); |
| } |
| |
| @Test public void redirect() throws Exception { |
| server.enqueue(new MockResponse() |
| .setResponseCode(301) |
| .addHeader("Location: /b") |
| .addHeader("Test", "Redirect from /a to /b") |
| .setBody("/a has moved!")); |
| server.enqueue(new MockResponse() |
| .setResponseCode(302) |
| .addHeader("Location: /c") |
| .addHeader("Test", "Redirect from /b to /c") |
| .setBody("/b has moved!")); |
| server.enqueue(new MockResponse().setBody("C")); |
| server.play(); |
| |
| Request request = new Request.Builder().url(server.getUrl("/a")).build(); |
| client.enqueue(request, receiver); |
| |
| receiver.await(server.getUrl("/c")) |
| .assertCode(200) |
| .assertBody("C") |
| .redirectedBy() |
| .assertCode(302) |
| .assertContainsHeaders("Test: Redirect from /b to /c") |
| .redirectedBy() |
| .assertCode(301) |
| .assertContainsHeaders("Test: Redirect from /a to /b"); |
| |
| assertEquals(0, server.takeRequest().getSequenceNumber()); // New connection. |
| assertEquals(1, server.takeRequest().getSequenceNumber()); // Connection reused. |
| assertEquals(2, server.takeRequest().getSequenceNumber()); // Connection reused again! |
| } |
| |
| @Test public void redirectWithRedirectsDisabled() throws Exception { |
| client.setFollowProtocolRedirects(false); |
| server.enqueue(new MockResponse() |
| .setResponseCode(301) |
| .addHeader("Location: /b") |
| .addHeader("Test", "Redirect from /a to /b") |
| .setBody("/a has moved!")); |
| server.play(); |
| |
| Request request = new Request.Builder().url(server.getUrl("/a")).build(); |
| client.enqueue(request, receiver); |
| |
| receiver.await(server.getUrl("/a")) |
| .assertCode(301) |
| .assertBody("/a has moved!") |
| .assertContainsHeaders("Location: /b"); |
| } |
| |
| @Test public void follow20Redirects() throws Exception { |
| for (int i = 0; i < 20; i++) { |
| server.enqueue(new MockResponse() |
| .setResponseCode(301) |
| .addHeader("Location: /" + (i + 1)) |
| .setBody("Redirecting to /" + (i + 1))); |
| } |
| server.enqueue(new MockResponse().setBody("Success!")); |
| server.play(); |
| |
| Request request = new Request.Builder().url(server.getUrl("/0")).build(); |
| client.enqueue(request, receiver); |
| receiver.await(server.getUrl("/20")) |
| .assertCode(200) |
| .assertBody("Success!"); |
| } |
| |
| @Test public void doesNotFollow21Redirects() throws Exception { |
| for (int i = 0; i < 21; i++) { |
| server.enqueue(new MockResponse() |
| .setResponseCode(301) |
| .addHeader("Location: /" + (i + 1)) |
| .setBody("Redirecting to /" + (i + 1))); |
| } |
| server.play(); |
| |
| Request request = new Request.Builder().url(server.getUrl("/0")).build(); |
| client.enqueue(request, receiver); |
| receiver.await(server.getUrl("/20")).assertFailure("Too many redirects: 21"); |
| } |
| |
| @Test public void canceledBeforeResponseReadIsNeverDelivered() throws Exception { |
| client.getDispatcher().setMaxRequests(1); // Force requests to be executed serially. |
| server.setDispatcher(new Dispatcher() { |
| char nextResponse = 'A'; |
| @Override public MockResponse dispatch(RecordedRequest request) { |
| client.cancel("request A"); |
| return new MockResponse().setBody(Character.toString(nextResponse++)); |
| } |
| }); |
| server.play(); |
| |
| // Canceling a request after the server has received a request but before |
| // it has delivered the response. That request will never be received to the |
| // client. |
| Request requestA = new Request.Builder().url(server.getUrl("/a")).tag("request A").build(); |
| client.enqueue(requestA, receiver); |
| assertEquals("/a", server.takeRequest().getPath()); |
| |
| // We then make a second request (not canceled) to make sure the receiver |
| // has nothing left to wait for. |
| Request requestB = new Request.Builder().url(server.getUrl("/b")).tag("request B").build(); |
| client.enqueue(requestB, receiver); |
| assertEquals("/b", server.takeRequest().getPath()); |
| receiver.await(requestB.url()).assertBody("B"); |
| |
| // At this point we know the receiver is ready: if it hasn't received 'A' |
| // yet it never will. |
| receiver.assertNoResponse(requestA.url()); |
| } |
| |
| @Test public void canceledAfterResponseIsDeliveredDoesNothing() throws Exception { |
| server.enqueue(new MockResponse().setBody("A")); |
| server.play(); |
| |
| final CountDownLatch latch = new CountDownLatch(1); |
| final AtomicReference<String> bodyRef = new AtomicReference<String>(); |
| |
| Request request = new Request.Builder().url(server.getUrl("/a")).tag("request A").build(); |
| client.enqueue(request, new Response.Receiver() { |
| @Override public void onFailure(Failure failure) { |
| throw new AssertionError(); |
| } |
| |
| @Override public boolean onResponse(Response response) throws IOException { |
| client.cancel("request A"); |
| bodyRef.set(response.body().string()); |
| latch.countDown(); |
| return true; |
| } |
| }); |
| |
| latch.await(); |
| assertEquals("A", bodyRef.get()); |
| } |
| |
| @Test public void connectionReuseWhenResponseBodyConsumed() throws Exception { |
| server.enqueue(new MockResponse().setBody("abc")); |
| server.enqueue(new MockResponse().setBody("def")); |
| server.play(); |
| |
| Request request = new Request.Builder().url(server.getUrl("/a")).build(); |
| client.enqueue(request, new Response.Receiver() { |
| @Override public void onFailure(Failure failure) { |
| throw new AssertionError(); |
| } |
| @Override public boolean onResponse(Response response) throws IOException { |
| InputStream bytes = response.body().byteStream(); |
| assertEquals('a', bytes.read()); |
| assertEquals('b', bytes.read()); |
| assertEquals('c', bytes.read()); |
| |
| // This request will share a connection with 'A' cause it's all done. |
| client.enqueue(new Request.Builder().url(server.getUrl("/b")).build(), receiver); |
| return true; |
| } |
| }); |
| |
| receiver.await(server.getUrl("/b")).assertCode(200).assertBody("def"); |
| assertEquals(0, server.takeRequest().getSequenceNumber()); // New connection. |
| assertEquals(1, server.takeRequest().getSequenceNumber()); // Connection reuse! |
| } |
| |
| @Test public void postBodyRetransmittedOnRedirect() throws Exception { |
| server.enqueue(new MockResponse() |
| .setResponseCode(302) |
| .addHeader("Location: /b") |
| .setBody("Moved to /b !")); |
| server.enqueue(new MockResponse() |
| .setBody("This is b.")); |
| server.play(); |
| |
| Request request = new Request.Builder() |
| .url(server.getUrl("/")) |
| .post(Request.Body.create(MediaType.parse("text/plain"), "body!")) |
| .build(); |
| client.enqueue(request, receiver); |
| |
| receiver.await(server.getUrl("/b")) |
| .assertCode(200) |
| .assertBody("This is b."); |
| |
| RecordedRequest request1 = server.takeRequest(); |
| assertEquals("body!", request1.getUtf8Body()); |
| assertEquals("5", request1.getHeader("Content-Length")); |
| assertEquals("text/plain; charset=utf-8", request1.getHeader("Content-Type")); |
| assertEquals(0, request1.getSequenceNumber()); |
| |
| RecordedRequest request2 = server.takeRequest(); |
| assertEquals("body!", request2.getUtf8Body()); |
| assertEquals("5", request2.getHeader("Content-Length")); |
| assertEquals("text/plain; charset=utf-8", request2.getHeader("Content-Type")); |
| assertEquals(1, request2.getSequenceNumber()); |
| } |
| } |