blob: 82b1075e90fd1ebaef1ee68c71807be60a83c6c4 [file] [log] [blame]
[email protected]61a527782013-02-21 03:58:001// Copyright (c) 2012 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "base/basictypes.h"
6#include "base/compiler_specific.h"
7#include "base/memory/scoped_ptr.h"
[email protected]61a527782013-02-21 03:58:008#include "net/base/test_completion_callback.h"
[email protected]6e7845ae2013-03-29 21:48:119#include "net/cert/mock_cert_verifier.h"
[email protected]f2cb3cf2013-03-21 01:40:5310#include "net/dns/mock_host_resolver.h"
[email protected]61a527782013-02-21 03:58:0011#include "net/http/http_auth_handler_factory.h"
12#include "net/http/http_network_session.h"
13#include "net/http/http_network_transaction.h"
14#include "net/http/http_server_properties_impl.h"
15#include "net/http/http_stream.h"
16#include "net/http/http_stream_factory.h"
17#include "net/http/http_transaction_unittest.h"
18#include "net/proxy/proxy_config_service_fixed.h"
19#include "net/proxy/proxy_resolver.h"
20#include "net/proxy/proxy_service.h"
21#include "net/quic/crypto/quic_decrypter.h"
22#include "net/quic/crypto/quic_encrypter.h"
23#include "net/quic/quic_framer.h"
[email protected]ed3fc15d2013-03-08 18:37:4424#include "net/quic/test_tools/crypto_test_utils.h"
[email protected]61a527782013-02-21 03:58:0025#include "net/quic/test_tools/mock_clock.h"
[email protected]e8ff26842013-03-22 21:02:0526#include "net/quic/test_tools/mock_crypto_client_stream_factory.h"
[email protected]61a527782013-02-21 03:58:0027#include "net/quic/test_tools/mock_random.h"
28#include "net/quic/test_tools/quic_test_utils.h"
29#include "net/socket/client_socket_factory.h"
30#include "net/socket/mock_client_socket_pool_manager.h"
31#include "net/socket/socket_test_util.h"
32#include "net/socket/ssl_client_socket.h"
33#include "net/spdy/spdy_frame_builder.h"
34#include "net/spdy/spdy_framer.h"
[email protected]536fd0b2013-03-14 17:41:5735#include "net/ssl/ssl_config_service_defaults.h"
[email protected]61a527782013-02-21 03:58:0036#include "testing/gtest/include/gtest/gtest.h"
37#include "testing/platform_test.h"
38
39//-----------------------------------------------------------------------------
40
41namespace {
42
43// This is the expected return from a current server advertising QUIC.
44static const char kQuicAlternateProtocolHttpHeader[] =
45 "Alternate-Protocol: 443:quic/1\r\n\r\n";
46
47// Returns a vector of NPN protocol strings for negotiating QUIC.
48std::vector<std::string> QuicNextProtos() {
49 std::vector<std::string> protos;
50 protos.push_back("http/1.1");
51 protos.push_back("quic/1");
52 return protos;
53}
54
55} // namespace
56
57namespace net {
58namespace test {
59
60class QuicNetworkTransactionTest : public PlatformTest {
61 protected:
[email protected]1c04f9522013-02-21 20:32:4362 QuicNetworkTransactionTest()
63 : clock_(new MockClock),
64 ssl_config_service_(new SSLConfigServiceDefaults),
65 proxy_service_(ProxyService::CreateDirect()),
66 auth_handler_factory_(
67 HttpAuthHandlerFactory::CreateDefault(&host_resolver_)) {
68 }
[email protected]61a527782013-02-21 03:58:0069
70 virtual void SetUp() {
71 NetworkChangeNotifier::NotifyObserversOfIPAddressChangeForTests();
72 MessageLoop::current()->RunUntilIdle();
73 }
74
75 virtual void TearDown() {
76 NetworkChangeNotifier::NotifyObserversOfIPAddressChangeForTests();
77 // Empty the current queue.
78 MessageLoop::current()->RunUntilIdle();
79 PlatformTest::TearDown();
80 NetworkChangeNotifier::NotifyObserversOfIPAddressChangeForTests();
81 MessageLoop::current()->RunUntilIdle();
82 HttpStreamFactory::set_use_alternate_protocols(false);
83 HttpStreamFactory::SetNextProtos(std::vector<std::string>());
84 }
85
[email protected]61a527782013-02-21 03:58:0086 scoped_ptr<QuicEncryptedPacket> ConstructRstPacket(
87 QuicPacketSequenceNumber num,
88 QuicStreamId stream_id) {
89 QuicPacketHeader header;
90 header.public_header.guid = 0xDEADBEEF;
[email protected]9db443912013-02-25 05:27:0391 header.public_header.reset_flag = false;
92 header.public_header.version_flag = false;
[email protected]61a527782013-02-21 03:58:0093 header.packet_sequence_number = num;
[email protected]9db443912013-02-25 05:27:0394 header.entropy_flag = false;
95 header.fec_flag = false;
96 header.fec_entropy_flag = false;
[email protected]61a527782013-02-21 03:58:0097 header.fec_group = 0;
98
[email protected]74bda142013-03-31 02:49:1199 QuicRstStreamFrame rst(stream_id, QUIC_STREAM_NO_ERROR);
[email protected]61a527782013-02-21 03:58:00100 return scoped_ptr<QuicEncryptedPacket>(
101 ConstructPacket(header, QuicFrame(&rst)));
102 }
103
104 scoped_ptr<QuicEncryptedPacket> ConstructAckPacket(
105 QuicPacketSequenceNumber largest_received,
106 QuicPacketSequenceNumber least_unacked) {
107 QuicPacketHeader header;
108 header.public_header.guid = 0xDEADBEEF;
[email protected]9db443912013-02-25 05:27:03109 header.public_header.reset_flag = false;
110 header.public_header.version_flag = false;
[email protected]e8ff26842013-03-22 21:02:05111 header.packet_sequence_number = 2;
[email protected]9db443912013-02-25 05:27:03112 header.entropy_flag = false;
113 header.fec_flag = false;
114 header.fec_entropy_flag = false;
[email protected]61a527782013-02-21 03:58:00115 header.fec_group = 0;
116
[email protected]14e8106c2013-03-14 16:25:33117 QuicAckFrame ack(largest_received, QuicTime::Zero(), least_unacked);
[email protected]61a527782013-02-21 03:58:00118
119 QuicCongestionFeedbackFrame feedback;
120 feedback.type = kTCP;
121 feedback.tcp.accumulated_number_of_lost_packets = 0;
122 feedback.tcp.receive_window = 256000;
123
[email protected]5351cc4b2013-03-03 07:22:41124 QuicFramer framer(kQuicVersion1,
125 QuicDecrypter::Create(kNULL),
[email protected]14e8106c2013-03-14 16:25:33126 QuicEncrypter::Create(kNULL),
[email protected]74bda142013-03-31 02:49:11127 QuicTime::Zero(),
[email protected]14e8106c2013-03-14 16:25:33128 false);
[email protected]61a527782013-02-21 03:58:00129 QuicFrames frames;
130 frames.push_back(QuicFrame(&ack));
131 frames.push_back(QuicFrame(&feedback));
132 scoped_ptr<QuicPacket> packet(
[email protected]9db443912013-02-25 05:27:03133 framer.ConstructFrameDataPacket(header, frames).packet);
134 return scoped_ptr<QuicEncryptedPacket>(
135 framer.EncryptPacket(header.packet_sequence_number, *packet));
[email protected]61a527782013-02-21 03:58:00136 }
137
138 std::string GetRequestString(const std::string& method,
139 const std::string& path) {
140 SpdyHeaderBlock headers;
141 headers[":method"] = method;
142 headers[":host"] = "www.google.com";
143 headers[":path"] = path;
144 headers[":scheme"] = "http";
145 headers[":version"] = "HTTP/1.1";
146 return SerializeHeaderBlock(headers);
147 }
148
149 std::string GetResponseString(const std::string& status,
150 const std::string& body) {
151 SpdyHeaderBlock headers;
152 headers[":status"] = status;
153 headers[":version"] = "HTTP/1.1";
154 headers["content-type"] = "text/plain";
155 return SerializeHeaderBlock(headers) + body;
156 }
157
158 std::string SerializeHeaderBlock(const SpdyHeaderBlock& headers) {
159 size_t len = SpdyFramer::GetSerializedLength(3, &headers);
160 SpdyFrameBuilder builder(len);
161 SpdyFramer::WriteHeaderBlock(&builder, 3, &headers);
162 scoped_ptr<SpdyFrame> frame(builder.take());
163 return std::string(frame->data(), len);
164 }
165
166 // Returns a newly created packet to send kData on stream 1.
167 QuicEncryptedPacket* ConstructDataPacket(
168 QuicPacketSequenceNumber sequence_number,
[email protected]e8ff26842013-03-22 21:02:05169 bool should_include_version,
[email protected]61a527782013-02-21 03:58:00170 bool fin,
171 QuicStreamOffset offset,
172 base::StringPiece data) {
[email protected]e8ff26842013-03-22 21:02:05173 InitializeHeader(sequence_number, should_include_version);
[email protected]61a527782013-02-21 03:58:00174 QuicStreamFrame frame(3, fin, offset, data);
175 return ConstructPacket(header_, QuicFrame(&frame)).release();
176 }
177
178 scoped_ptr<QuicEncryptedPacket> ConstructPacket(
179 const QuicPacketHeader& header,
180 const QuicFrame& frame) {
[email protected]5351cc4b2013-03-03 07:22:41181 QuicFramer framer(kQuicVersion1,
182 QuicDecrypter::Create(kNULL),
[email protected]14e8106c2013-03-14 16:25:33183 QuicEncrypter::Create(kNULL),
[email protected]74bda142013-03-31 02:49:11184 QuicTime::Zero(),
[email protected]14e8106c2013-03-14 16:25:33185 false);
[email protected]61a527782013-02-21 03:58:00186 QuicFrames frames;
187 frames.push_back(frame);
188 scoped_ptr<QuicPacket> packet(
[email protected]9db443912013-02-25 05:27:03189 framer.ConstructFrameDataPacket(header, frames).packet);
190 return scoped_ptr<QuicEncryptedPacket>(
191 framer.EncryptPacket(header.packet_sequence_number, *packet));
[email protected]61a527782013-02-21 03:58:00192 }
193
[email protected]e8ff26842013-03-22 21:02:05194 void InitializeHeader(QuicPacketSequenceNumber sequence_number,
195 bool should_include_version) {
[email protected]61a527782013-02-21 03:58:00196 header_.public_header.guid = random_generator_.RandUint64();
[email protected]9db443912013-02-25 05:27:03197 header_.public_header.reset_flag = false;
[email protected]e8ff26842013-03-22 21:02:05198 header_.public_header.version_flag = should_include_version;
[email protected]61a527782013-02-21 03:58:00199 header_.packet_sequence_number = sequence_number;
200 header_.fec_group = 0;
[email protected]9db443912013-02-25 05:27:03201 header_.entropy_flag = false;
202 header_.fec_flag = false;
203 header_.fec_entropy_flag = false;
[email protected]61a527782013-02-21 03:58:00204 }
205
206 void CreateSession() {
[email protected]4dca587c2013-03-07 16:54:47207 params_.enable_quic = true;
208 params_.quic_clock = clock_;
209 params_.quic_random = &random_generator_;
[email protected]61a527782013-02-21 03:58:00210 params_.client_socket_factory = &socket_factory_;
[email protected]e8ff26842013-03-22 21:02:05211 params_.quic_crypto_client_stream_factory = &crypto_client_stream_factory_;
[email protected]1c04f9522013-02-21 20:32:43212 params_.host_resolver = &host_resolver_;
213 params_.cert_verifier = &cert_verifier_;
214 params_.proxy_service = proxy_service_.get();
215 params_.ssl_config_service = ssl_config_service_.get();
216 params_.http_auth_handler_factory = auth_handler_factory_.get();
[email protected]61a527782013-02-21 03:58:00217 params_.http_server_properties = &http_server_properties;
218
219 session_ = new HttpNetworkSession(params_);
220 }
221
222 QuicPacketHeader header_;
223 scoped_refptr<HttpNetworkSession> session_;
224 MockClientSocketFactory socket_factory_;
[email protected]e8ff26842013-03-22 21:02:05225 MockCryptoClientStreamFactory crypto_client_stream_factory_;
[email protected]1c04f9522013-02-21 20:32:43226 MockClock* clock_; // Owned by QuicStreamFactory after CreateSession.
227 MockHostResolver host_resolver_;
228 MockCertVerifier cert_verifier_;
229 scoped_refptr<SSLConfigServiceDefaults> ssl_config_service_;
230 scoped_ptr<ProxyService> proxy_service_;
231 scoped_ptr<HttpAuthHandlerFactory> auth_handler_factory_;
[email protected]61a527782013-02-21 03:58:00232 MockRandom random_generator_;
233 HttpServerPropertiesImpl http_server_properties;
234 HttpNetworkSession::Params params_;
235};
236
[email protected]4dca587c2013-03-07 16:54:47237TEST_F(QuicNetworkTransactionTest, ForceQuic) {
238 params_.origin_port_to_force_quic_on = 80;
239
240 HttpRequestInfo request;
241 request.method = "GET";
242 request.url = GURL("http://www.google.com/");
243 request.load_flags = 0;
244
[email protected]4dca587c2013-03-07 16:54:47245 scoped_ptr<QuicEncryptedPacket> data(
[email protected]e8ff26842013-03-22 21:02:05246 ConstructDataPacket(1, true, true, 0, GetRequestString("GET", "/")));
247 scoped_ptr<QuicEncryptedPacket> ack(ConstructAckPacket(1, 0));
[email protected]4dca587c2013-03-07 16:54:47248
249 MockWrite quic_writes[] = {
[email protected]4dca587c2013-03-07 16:54:47250 MockWrite(SYNCHRONOUS, data->data(), data->length()),
251 MockWrite(SYNCHRONOUS, ack->data(), ack->length()),
252 };
253
[email protected]4dca587c2013-03-07 16:54:47254 scoped_ptr<QuicEncryptedPacket> resp(
[email protected]e8ff26842013-03-22 21:02:05255 ConstructDataPacket(
256 1, false, true, 0, GetResponseString("200 OK", "hello!")));
[email protected]4dca587c2013-03-07 16:54:47257 MockRead quic_reads[] = {
[email protected]4dca587c2013-03-07 16:54:47258 MockRead(SYNCHRONOUS, resp->data(), resp->length()),
259 MockRead(ASYNC, OK), // EOF
260 };
261
262 DelayedSocketData quic_data(
263 1, // wait for one write to finish before reading.
264 quic_reads, arraysize(quic_reads),
265 quic_writes, arraysize(quic_writes));
266
267 socket_factory_.AddSocketDataProvider(&quic_data);
268
269 // The non-alternate protocol job needs to hang in order to guarantee that the
270 // alternate-protocol job will "win".
271 MockConnect never_finishing_connect(SYNCHRONOUS, ERR_IO_PENDING);
272 StaticSocketDataProvider hanging_non_alternate_protocol_socket(
273 NULL, 0, NULL, 0);
274 hanging_non_alternate_protocol_socket.set_connect_data(
275 never_finishing_connect);
276 socket_factory_.AddSocketDataProvider(
277 &hanging_non_alternate_protocol_socket);
278
279 TestCompletionCallback callback;
280
281 CreateSession();
282 scoped_ptr<HttpNetworkTransaction> trans(
[email protected]262eec82013-03-19 21:01:36283 new HttpNetworkTransaction(DEFAULT_PRIORITY, session_));
[email protected]4dca587c2013-03-07 16:54:47284
285 int rv = trans->Start(&request, callback.callback(), BoundNetLog());
286 EXPECT_EQ(ERR_IO_PENDING, rv);
287 EXPECT_EQ(OK, callback.WaitForResult());
288
289 const HttpResponseInfo* response = trans->GetResponseInfo();
290 ASSERT_TRUE(response != NULL);
291 ASSERT_TRUE(response->headers != NULL);
292 EXPECT_EQ("HTTP/1.1 200 OK", response->headers->GetStatusLine());
293 EXPECT_TRUE(response->was_fetched_via_spdy);
[email protected]808957c2013-03-13 03:51:27294 EXPECT_EQ(HttpResponseInfo::CONNECTION_INFO_QUIC1_SPDY3,
295 response->connection_info);
[email protected]4dca587c2013-03-07 16:54:47296
297 std::string response_data;
298 ASSERT_EQ(OK, ReadTransaction(trans.get(), &response_data));
299 EXPECT_EQ("hello!", response_data);
300}
301
302TEST_F(QuicNetworkTransactionTest, DoNotForceQuicForHttps) {
303 // Attempt to "force" quic on 443, which will not be honored.
304 params_.origin_port_to_force_quic_on = 443;
305
306 HttpRequestInfo request;
307 request.method = "GET";
308 request.url = GURL("https://www.google.com/");
309 request.load_flags = 0;
310
311 MockRead data_reads[] = {
312 MockRead("HTTP/1.1 200 OK\r\n\r\n"),
313 MockRead("hello world"),
314 MockRead(SYNCHRONOUS, ERR_TEST_PEER_CLOSE_AFTER_NEXT_MOCK_READ),
315 MockRead(ASYNC, OK)
316 };
317
318 StaticSocketDataProvider data(
319 data_reads, arraysize(data_reads), NULL, 0);
320 socket_factory_.AddSocketDataProvider(&data);
321 SSLSocketDataProvider ssl(ASYNC, OK);
322 socket_factory_.AddSSLSocketDataProvider(&ssl);
323
324 TestCompletionCallback callback;
325
326 CreateSession();
327 scoped_ptr<HttpNetworkTransaction> trans(
[email protected]262eec82013-03-19 21:01:36328 new HttpNetworkTransaction(DEFAULT_PRIORITY, session_));
[email protected]4dca587c2013-03-07 16:54:47329
330 int rv = trans->Start(&request, callback.callback(), BoundNetLog());
331 EXPECT_EQ(ERR_IO_PENDING, rv);
332 EXPECT_EQ(OK, callback.WaitForResult());
333
334 const HttpResponseInfo* response = trans->GetResponseInfo();
335 ASSERT_TRUE(response != NULL);
336 ASSERT_TRUE(response->headers != NULL);
337 EXPECT_EQ("HTTP/1.1 200 OK", response->headers->GetStatusLine());
338
339 std::string response_data;
340 ASSERT_EQ(OK, ReadTransaction(trans.get(), &response_data));
341 EXPECT_EQ("hello world", response_data);
342}
343
[email protected]61a527782013-02-21 03:58:00344TEST_F(QuicNetworkTransactionTest, UseAlternateProtocolForQuic) {
345 HttpStreamFactory::set_use_alternate_protocols(true);
346 HttpStreamFactory::SetNextProtos(QuicNextProtos());
[email protected]61a527782013-02-21 03:58:00347
348 HttpRequestInfo request;
349 request.method = "GET";
350 request.url = GURL("http://www.google.com/");
351 request.load_flags = 0;
352
353 MockRead data_reads[] = {
354 MockRead("HTTP/1.1 200 OK\r\n"),
355 MockRead(kQuicAlternateProtocolHttpHeader),
356 MockRead("hello world"),
357 MockRead(SYNCHRONOUS, ERR_TEST_PEER_CLOSE_AFTER_NEXT_MOCK_READ),
358 MockRead(ASYNC, OK)
359 };
360
361 StaticSocketDataProvider first_transaction(
362 data_reads, arraysize(data_reads), NULL, 0);
363 socket_factory_.AddSocketDataProvider(&first_transaction);
364
365
[email protected]61a527782013-02-21 03:58:00366 scoped_ptr<QuicEncryptedPacket> data(
[email protected]e8ff26842013-03-22 21:02:05367 ConstructDataPacket(1, true, true, 0, GetRequestString("GET", "/")));
368 scoped_ptr<QuicEncryptedPacket> ack(ConstructAckPacket(1, 0));
[email protected]61a527782013-02-21 03:58:00369
370 MockWrite quic_writes[] = {
[email protected]61a527782013-02-21 03:58:00371 MockWrite(SYNCHRONOUS, data->data(), data->length()),
372 MockWrite(SYNCHRONOUS, ack->data(), ack->length()),
373 };
374
[email protected]61a527782013-02-21 03:58:00375 scoped_ptr<QuicEncryptedPacket> resp(
[email protected]e8ff26842013-03-22 21:02:05376 ConstructDataPacket(
377 1, false, true, 0, GetResponseString("200 OK", "hello!")));
[email protected]61a527782013-02-21 03:58:00378 MockRead quic_reads[] = {
[email protected]61a527782013-02-21 03:58:00379 MockRead(SYNCHRONOUS, resp->data(), resp->length()),
380 MockRead(ASYNC, OK), // EOF
381 };
382
383 DelayedSocketData quic_data(
384 1, // wait for one write to finish before reading.
385 quic_reads, arraysize(quic_reads),
386 quic_writes, arraysize(quic_writes));
387
388 socket_factory_.AddSocketDataProvider(&quic_data);
389
390 // The non-alternate protocol job needs to hang in order to guarantee that the
391 // alternate-protocol job will "win".
392 MockConnect never_finishing_connect(SYNCHRONOUS, ERR_IO_PENDING);
393 StaticSocketDataProvider hanging_non_alternate_protocol_socket(
394 NULL, 0, NULL, 0);
395 hanging_non_alternate_protocol_socket.set_connect_data(
396 never_finishing_connect);
397 socket_factory_.AddSocketDataProvider(
398 &hanging_non_alternate_protocol_socket);
399
400 TestCompletionCallback callback;
401
402 CreateSession();
403 scoped_ptr<HttpNetworkTransaction> trans(
[email protected]262eec82013-03-19 21:01:36404 new HttpNetworkTransaction(DEFAULT_PRIORITY, session_));
[email protected]61a527782013-02-21 03:58:00405
406 int rv = trans->Start(&request, callback.callback(), BoundNetLog());
407 EXPECT_EQ(ERR_IO_PENDING, rv);
408 EXPECT_EQ(OK, callback.WaitForResult());
409
410 const HttpResponseInfo* response = trans->GetResponseInfo();
411 ASSERT_TRUE(response != NULL);
412 ASSERT_TRUE(response->headers != NULL);
413 EXPECT_EQ("HTTP/1.1 200 OK", response->headers->GetStatusLine());
414
415 std::string response_data;
416 ASSERT_EQ(OK, ReadTransaction(trans.get(), &response_data));
417 EXPECT_EQ("hello world", response_data);
418
[email protected]262eec82013-03-19 21:01:36419 trans.reset(new HttpNetworkTransaction(DEFAULT_PRIORITY, session_));
[email protected]61a527782013-02-21 03:58:00420
421 rv = trans->Start(&request, callback.callback(), BoundNetLog());
422 EXPECT_EQ(ERR_IO_PENDING, rv);
423 EXPECT_EQ(OK, callback.WaitForResult());
424
425 response = trans->GetResponseInfo();
426 ASSERT_TRUE(response != NULL);
427 ASSERT_TRUE(response->headers != NULL);
428 EXPECT_EQ("HTTP/1.1 200 OK", response->headers->GetStatusLine());
429 EXPECT_TRUE(response->was_fetched_via_spdy);
[email protected]b09e0132013-03-11 20:39:33430 EXPECT_TRUE(response->was_npn_negotiated);
431 EXPECT_EQ("quic/1+spdy/3", response->npn_negotiated_protocol);
[email protected]61a527782013-02-21 03:58:00432
433 ASSERT_EQ(OK, ReadTransaction(trans.get(), &response_data));
434 EXPECT_EQ("hello!", response_data);
435}
436
[email protected]ff899c82013-03-12 23:10:50437TEST_F(QuicNetworkTransactionTest, DontUseAlternateProtocolForQuicHttps) {
438 HttpStreamFactory::set_use_alternate_protocols(true);
439 HttpStreamFactory::SetNextProtos(QuicNextProtos());
440
441 HttpRequestInfo request;
442 request.method = "GET";
443 request.url = GURL("https://www.google.com/");
444 request.load_flags = 0;
445
446 MockRead data_reads[] = {
447 MockRead("HTTP/1.1 200 OK\r\n"),
448 MockRead("Content-length: 11\r\n"),
449 MockRead(kQuicAlternateProtocolHttpHeader),
450 MockRead("hello world"),
451
452 MockRead("HTTP/1.1 200 OK\r\n"),
453 MockRead("Content-length: 6\r\n"),
454 MockRead(kQuicAlternateProtocolHttpHeader),
455 MockRead("hello!"),
456 };
457
458 StaticSocketDataProvider first_transaction(
459 data_reads, arraysize(data_reads), NULL, 0);
460 socket_factory_.AddSocketDataProvider(&first_transaction);
461 SSLSocketDataProvider ssl(ASYNC, OK);
462 socket_factory_.AddSSLSocketDataProvider(&ssl);
463
464 TestCompletionCallback callback;
465
466 CreateSession();
467 scoped_ptr<HttpNetworkTransaction> trans(
[email protected]262eec82013-03-19 21:01:36468 new HttpNetworkTransaction(DEFAULT_PRIORITY, session_));
[email protected]ff899c82013-03-12 23:10:50469
470 int rv = trans->Start(&request, callback.callback(), BoundNetLog());
471 EXPECT_EQ(ERR_IO_PENDING, rv);
472 EXPECT_EQ(OK, callback.WaitForResult());
473
474 const HttpResponseInfo* response = trans->GetResponseInfo();
475 ASSERT_TRUE(response != NULL);
476 ASSERT_TRUE(response->headers != NULL);
477 EXPECT_EQ("HTTP/1.1 200 OK", response->headers->GetStatusLine());
478
479 std::string response_data;
480 ASSERT_EQ(OK, ReadTransaction(trans.get(), &response_data));
481 EXPECT_EQ("hello world", response_data);
482
[email protected]262eec82013-03-19 21:01:36483 trans.reset(new HttpNetworkTransaction(DEFAULT_PRIORITY, session_));
[email protected]ff899c82013-03-12 23:10:50484
485 rv = trans->Start(&request, callback.callback(), BoundNetLog());
486 EXPECT_EQ(ERR_IO_PENDING, rv);
487 EXPECT_EQ(OK, callback.WaitForResult());
488
489 response = trans->GetResponseInfo();
490 ASSERT_TRUE(response != NULL);
491 ASSERT_TRUE(response->headers != NULL);
492 EXPECT_EQ("HTTP/1.1 200 OK", response->headers->GetStatusLine());
493 EXPECT_FALSE(response->was_fetched_via_spdy);
494
495 ASSERT_EQ(OK, ReadTransaction(trans.get(), &response_data));
496 EXPECT_EQ("hello!", response_data);
497}
498
[email protected]61a527782013-02-21 03:58:00499} // namespace test
500} // namespace net