blob: 0cb728b28b36cfe57519b1052dadc6ee544b217d [file] [log] [blame]
Yixin Wang0d2c6b7e12017-08-16 21:12:551// Copyright (c) 2017 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
Ryan Hamiltona3ee93a72018-08-01 22:03:085#include "net/quic/quic_proxy_client_socket.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:556
Bence Béky8f9d7d3952017-10-09 19:58:047#include <memory>
Bence Békyd8a21fc32018-06-27 18:29:588#include <utility>
Bence Béky8f9d7d3952017-10-09 19:58:049
Sebastien Marchand6d0558fd2019-01-25 16:49:3710#include "base/bind.h"
Bence Béky8f9d7d3952017-10-09 19:58:0411#include "base/memory/ptr_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5512#include "base/run_loop.h"
13#include "base/strings/utf_string_conversions.h"
Lei Zhang52637ed2019-02-20 01:38:3714#include "base/threading/thread_task_runner_handle.h"
Zhongyi Shic16b4102019-02-12 00:37:4015#include "base/time/default_tick_clock.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5516#include "net/dns/mock_host_resolver.h"
17#include "net/http/http_auth_cache.h"
18#include "net/http/http_auth_handler_factory.h"
19#include "net/http/http_response_headers.h"
20#include "net/http/transport_security_state.h"
21#include "net/log/test_net_log.h"
22#include "net/log/test_net_log_util.h"
Ryan Hamiltona3ee93a72018-08-01 22:03:0823#include "net/quic/crypto/proof_verifier_chromium.h"
24#include "net/quic/mock_crypto_client_stream_factory.h"
25#include "net/quic/mock_quic_data.h"
26#include "net/quic/quic_chromium_alarm_factory.h"
27#include "net/quic/quic_chromium_client_session.h"
28#include "net/quic/quic_chromium_connection_helper.h"
29#include "net/quic/quic_chromium_packet_writer.h"
30#include "net/quic/quic_http_utils.h"
31#include "net/quic/quic_server_info.h"
32#include "net/quic/quic_stream_factory.h"
33#include "net/quic/quic_test_packet_maker.h"
34#include "net/quic/test_task_runner.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5535#include "net/socket/socket_test_util.h"
36#include "net/test/cert_test_util.h"
Victor Vasilievdcdb6192019-02-25 19:49:5637#include "net/test/gtest_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5538#include "net/test/test_data_directory.h"
Bence Béky98447b12018-05-08 03:14:0139#include "net/test/test_with_scoped_task_environment.h"
Ryan Hamilton56b10c5d2018-05-11 13:40:1640#include "net/third_party/quic/core/crypto/null_encrypter.h"
Ryan Hamilton47cf9d12018-10-17 04:33:0941#include "net/third_party/quic/core/quic_utils.h"
Ryan Hamilton56b10c5d2018-05-11 13:40:1642#include "net/third_party/quic/core/tls_client_handshaker.h"
43#include "net/third_party/quic/test_tools/crypto_test_utils.h"
44#include "net/third_party/quic/test_tools/mock_clock.h"
45#include "net/third_party/quic/test_tools/mock_random.h"
46#include "net/third_party/quic/test_tools/quic_connection_peer.h"
47#include "net/third_party/quic/test_tools/quic_test_utils.h"
[email protected]578968d42017-12-13 15:39:3248#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5549#include "testing/gmock/include/gmock/gmock.h"
50#include "testing/gtest/include/gtest/gtest.h"
51
52using testing::_;
53using testing::AnyNumber;
54using testing::Return;
55
56namespace {
57
58static const char kOriginHost[] = "www.google.com";
59static const int kOriginPort = 443;
60static const char kProxyUrl[] = "https://myproxy:6121/";
61static const char kProxyHost[] = "myproxy";
62static const int kProxyPort = 6121;
63static const char kUserAgent[] = "Mozilla/1.0";
64static const char kRedirectUrl[] = "https://example.com/";
65
66static const char kMsg1[] = "\0hello!\xff";
67static const int kLen1 = 8;
68static const char kMsg2[] = "\0a2345678\0";
69static const int kLen2 = 10;
70static const char kMsg3[] = "bye!";
71static const int kLen3 = 4;
72static const char kMsg33[] = "bye!bye!";
73static const int kLen33 = kLen3 + kLen3;
74static const char kMsg333[] = "bye!bye!bye!";
75static const int kLen333 = kLen3 + kLen3 + kLen3;
76
77} // anonymous namespace
78
79namespace net {
80namespace test {
81
Michael Warres74ee3ce2017-10-09 15:26:3782class QuicProxyClientSocketTest
Ryan Hamilton8d9ee76e2018-05-29 23:52:5283 : public ::testing::TestWithParam<
84 std::tuple<quic::QuicTransportVersion, bool>>,
Bence Béky98447b12018-05-08 03:14:0185 public WithScopedTaskEnvironment {
Yixin Wang0d2c6b7e12017-08-16 21:12:5586 protected:
87 static const bool kFin = true;
88 static const bool kIncludeVersion = true;
89 static const bool kIncludeDiversificationNonce = true;
90 static const bool kIncludeCongestionFeedback = true;
91 static const bool kSendFeedback = true;
92
93 static size_t GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:5294 quic::QuicByteCount packet_length,
95 quic::QuicTransportVersion version,
Yixin Wang0d2c6b7e12017-08-16 21:12:5596 bool include_version,
97 bool include_diversification_nonce,
Ryan Hamilton8d9ee76e2018-05-29 23:52:5298 quic::QuicConnectionIdLength connection_id_length,
99 quic::QuicPacketNumberLength packet_number_length,
100 quic::QuicStreamOffset offset) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55101 size_t min_data_length = 1;
102 size_t min_packet_length =
Ryan Hamilton8d9ee76e2018-05-29 23:52:52103 quic::NullEncrypter(quic::Perspective::IS_CLIENT)
Yixin Wang0d2c6b7e12017-08-16 21:12:55104 .GetCiphertextSize(min_data_length) +
Ryan Hamilton8d9ee76e2018-05-29 23:52:52105 quic::QuicPacketCreator::StreamFramePacketOverhead(
Michael Warres60637ae2018-06-05 20:03:11106 version, quic::PACKET_8BYTE_CONNECTION_ID,
107 quic::PACKET_0BYTE_CONNECTION_ID, include_version,
Yixin Wang0d2c6b7e12017-08-16 21:12:55108 include_diversification_nonce, packet_number_length, offset);
109
110 DCHECK(packet_length >= min_packet_length);
111 return min_data_length + packet_length - min_packet_length;
112 }
113
114 QuicProxyClientSocketTest()
Yixin Wang079ad542018-01-11 04:06:05115 : version_(std::get<0>(GetParam())),
Ryan Hamilton47cf9d12018-10-17 04:33:09116 client_data_stream_id1_(quic::QuicUtils::GetHeadersStreamId(version_) +
Fan Yang32c5a112018-12-10 20:06:33117 quic::QuicUtils::StreamIdDelta(version_)),
Yixin Wang079ad542018-01-11 04:06:05118 client_headers_include_h2_stream_dependency_(std::get<1>(GetParam())),
Ryan Hamilton8d9ee76e2018-05-29 23:52:52119 crypto_config_(quic::test::crypto_test_utils::ProofVerifierForTesting(),
120 quic::TlsClientHandshaker::CreateSslCtx()),
David Schinazic8281052019-01-24 06:14:17121 connection_id_(quic::test::TestConnectionId(2)),
Yixin Wang0d2c6b7e12017-08-16 21:12:55122 client_maker_(version_,
123 connection_id_,
124 &clock_,
125 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52126 quic::Perspective::IS_CLIENT,
Yixin Wang079ad542018-01-11 04:06:05127 client_headers_include_h2_stream_dependency_),
Yixin Wang0d2c6b7e12017-08-16 21:12:55128 server_maker_(version_,
129 connection_id_,
130 &clock_,
131 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52132 quic::Perspective::IS_SERVER,
Yixin Wang079ad542018-01-11 04:06:05133 false),
Yixin Wang0d2c6b7e12017-08-16 21:12:55134 random_generator_(0),
135 header_stream_offset_(0),
136 response_offset_(0),
137 user_agent_(kUserAgent),
138 proxy_host_port_(kProxyHost, kProxyPort),
139 endpoint_host_port_(kOriginHost, kOriginPort),
140 host_resolver_(new MockCachingHostResolver()),
141 http_auth_handler_factory_(
142 HttpAuthHandlerFactory::CreateDefault(host_resolver_.get())) {
143 IPAddress ip(192, 0, 2, 33);
144 peer_addr_ = IPEndPoint(ip, 443);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52145 clock_.AdvanceTime(quic::QuicTime::Delta::FromMilliseconds(20));
Yixin Wang0d2c6b7e12017-08-16 21:12:55146 }
147
148 void SetUp() override {}
149
Yixin Wang2bea3cf2017-11-09 18:11:03150 void TearDown() override {
151 sock_.reset();
152 EXPECT_TRUE(mock_quic_data_.AllReadDataConsumed());
153 EXPECT_TRUE(mock_quic_data_.AllWriteDataConsumed());
154 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55155
156 void Initialize() {
157 std::unique_ptr<MockUDPClientSocket> socket(new MockUDPClientSocket(
158 mock_quic_data_.InitializeAndGetSequencedSocketData(),
159 net_log_.bound().net_log()));
160 socket->Connect(peer_addr_);
161 runner_ = new TestTaskRunner(&clock_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52162 send_algorithm_ = new quic::test::MockSendAlgorithm();
Yixin Wang0d2c6b7e12017-08-16 21:12:55163 EXPECT_CALL(*send_algorithm_, InRecovery()).WillRepeatedly(Return(false));
164 EXPECT_CALL(*send_algorithm_, InSlowStart()).WillRepeatedly(Return(false));
165 EXPECT_CALL(*send_algorithm_, OnPacketSent(_, _, _, _, _))
Dan Zhangf11470172017-09-18 22:02:09166 .Times(testing::AtLeast(1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55167 EXPECT_CALL(*send_algorithm_, GetCongestionWindow())
Ryan Hamilton8d9ee76e2018-05-29 23:52:52168 .WillRepeatedly(Return(quic::kMaxPacketSize));
Yixin Wang0d2c6b7e12017-08-16 21:12:55169 EXPECT_CALL(*send_algorithm_, PacingRate(_))
Ryan Hamilton8d9ee76e2018-05-29 23:52:52170 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Michael Warres74ee3ce2017-10-09 15:26:37171 EXPECT_CALL(*send_algorithm_, CanSend(_)).WillRepeatedly(Return(true));
Yixin Wang0d2c6b7e12017-08-16 21:12:55172 EXPECT_CALL(*send_algorithm_, BandwidthEstimate())
Ryan Hamilton8d9ee76e2018-05-29 23:52:52173 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Yixin Wang0d2c6b7e12017-08-16 21:12:55174 EXPECT_CALL(*send_algorithm_, SetFromConfig(_, _)).Times(AnyNumber());
175 EXPECT_CALL(*send_algorithm_, OnApplicationLimited(_)).Times(AnyNumber());
176 EXPECT_CALL(*send_algorithm_, GetCongestionControlType())
177 .Times(AnyNumber());
178 helper_.reset(
179 new QuicChromiumConnectionHelper(&clock_, &random_generator_));
180 alarm_factory_.reset(new QuicChromiumAlarmFactory(runner_.get(), &clock_));
181
Ryan Hamilton9edcf1a2017-11-22 05:55:17182 QuicChromiumPacketWriter* writer = new QuicChromiumPacketWriter(
183 socket.get(), base::ThreadTaskRunnerHandle::Get().get());
Ryan Hamilton8d9ee76e2018-05-29 23:52:52184 quic::QuicConnection* connection = new quic::QuicConnection(
185 connection_id_,
186 quic::QuicSocketAddress(quic::QuicSocketAddressImpl(peer_addr_)),
Yixin Wang0d2c6b7e12017-08-16 21:12:55187 helper_.get(), alarm_factory_.get(), writer, true /* owns_writer */,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52188 quic::Perspective::IS_CLIENT,
189 quic::test::SupportedVersions(
190 quic::ParsedQuicVersion(quic::PROTOCOL_QUIC_CRYPTO, version_)));
Yixin Wang0d2c6b7e12017-08-16 21:12:55191 connection->set_visitor(&visitor_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52192 quic::test::QuicConnectionPeer::SetSendAlgorithm(connection,
193 send_algorithm_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55194
195 // Load a certificate that is valid for *.example.org
196 scoped_refptr<X509Certificate> test_cert(
197 ImportCertFromFile(GetTestCertsDirectory(), "wildcard.pem"));
198 EXPECT_TRUE(test_cert.get());
199
200 verify_details_.cert_verify_result.verified_cert = test_cert;
201 verify_details_.cert_verify_result.is_issued_by_known_root = true;
202 crypto_client_stream_factory_.AddProofVerifyDetails(&verify_details_);
203
204 base::TimeTicks dns_end = base::TimeTicks::Now();
205 base::TimeTicks dns_start = dns_end - base::TimeDelta::FromMilliseconds(1);
206
207 session_.reset(new QuicChromiumClientSession(
208 connection, std::move(socket),
209 /*stream_factory=*/nullptr, &crypto_client_stream_factory_, &clock_,
Nick Harper89bc7212018-07-31 19:07:57210 &transport_security_state_, /*ssl_config_service=*/nullptr,
Yixin Wang0d2c6b7e12017-08-16 21:12:55211 base::WrapUnique(static_cast<QuicServerInfo*>(nullptr)),
Paul Jensen8e3c5d32018-02-19 17:06:33212 QuicSessionKey("mail.example.org", 80, PRIVACY_MODE_DISABLED,
213 SocketTag()),
Zhongyi Shi757fcce2018-06-27 05:41:27214 /*require_confirmation=*/false, /*migrate_session_early_v2=*/false,
215 /*migrate_session_on_network_change_v2=*/false,
216 /*default_network=*/NetworkChangeNotifier::kInvalidNetworkHandle,
Zhongyi Shie01f2db2019-02-22 19:53:23217 quic::QuicTime::Delta::FromMilliseconds(
218 kDefaultRetransmittableOnWireTimeoutMillisecs),
Zhongyi Shic16b4102019-02-12 00:37:40219 base::TimeDelta::FromSeconds(kDefaultIdleSessionMigrationPeriodSeconds),
Zhongyi Shi73f23ca872017-12-13 18:37:13220 base::TimeDelta::FromSeconds(kMaxTimeOnNonDefaultNetworkSecs),
Zhongyi Shiee760762018-08-01 00:54:29221 kMaxMigrationsToNonDefaultNetworkOnWriteError,
Zhongyi Shi8b1e43f2017-12-13 20:46:30222 kMaxMigrationsToNonDefaultNetworkOnPathDegrading,
Zhongyi Shi5f587cc2017-11-21 23:24:17223 kQuicYieldAfterPacketsRead,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52224 quic::QuicTime::Delta::FromMilliseconds(
225 kQuicYieldAfterDurationMilliseconds),
Zhongyi Shidbce7f412019-02-01 23:16:29226 /*go_away_on_path_degrading*/ false,
Yixin Wang079ad542018-01-11 04:06:05227 client_headers_include_h2_stream_dependency_, /*cert_verify_flags=*/0,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52228 quic::test::DefaultQuicConfig(), &crypto_config_, "CONNECTION_UNKNOWN",
229 dns_start, dns_end, &push_promise_index_, nullptr,
Zhongyi Shic16b4102019-02-12 00:37:40230 base::DefaultTickClock::GetInstance(),
Yixin Wang0d2c6b7e12017-08-16 21:12:55231 base::ThreadTaskRunnerHandle::Get().get(),
232 /*socket_performance_watcher=*/nullptr, net_log_.bound().net_log()));
233
234 writer->set_delegate(session_.get());
235
Ryan Hamilton6c2a2a82017-12-15 02:06:28236 session_handle_ =
237 session_->CreateHandle(HostPortPair("mail.example.org", 80));
Yixin Wang0d2c6b7e12017-08-16 21:12:55238
239 session_->Initialize();
240 TestCompletionCallback callback;
241 EXPECT_THAT(session_->CryptoConnect(callback.callback()), IsOk());
242 EXPECT_TRUE(session_->IsCryptoHandshakeConfirmed());
243
Ramin Halavati683bcaa92018-02-14 08:42:39244 EXPECT_THAT(session_handle_->RequestStream(true, callback.callback(),
245 TRAFFIC_ANNOTATION_FOR_TESTS),
Yixin Wang0d2c6b7e12017-08-16 21:12:55246 IsOk());
247 std::unique_ptr<QuicChromiumClientStream::Handle> stream_handle =
248 session_handle_->ReleaseStream();
249 EXPECT_TRUE(stream_handle->IsOpen());
250
251 sock_.reset(new QuicProxyClientSocket(
252 std::move(stream_handle), std::move(session_handle_), user_agent_,
253 endpoint_host_port_, net_log_.bound(),
254 new HttpAuthController(HttpAuth::AUTH_PROXY,
255 GURL("https://" + proxy_host_port_.ToString()),
256 &http_auth_cache_,
257 http_auth_handler_factory_.get())));
258
259 session_->StartReading();
260 }
261
Ryan Hamilton0239aac2018-05-19 00:03:13262 void PopulateConnectRequestIR(spdy::SpdyHeaderBlock* block) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55263 (*block)[":method"] = "CONNECT";
264 (*block)[":authority"] = endpoint_host_port_.ToString();
265 (*block)["user-agent"] = kUserAgent;
266 }
267
268 // Helper functions for constructing packets sent by the client
269
Ryan Hamilton8d9ee76e2018-05-29 23:52:52270 std::unique_ptr<quic::QuicReceivedPacket> ConstructSettingsPacket(
Fan Yangac867502019-01-28 21:10:23271 uint64_t packet_number) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55272 return client_maker_.MakeInitialSettingsPacket(packet_number,
273 &header_stream_offset_);
274 }
275
Ryan Hamilton8d9ee76e2018-05-29 23:52:52276 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23277 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52278 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23279 uint64_t largest_received,
280 uint64_t smallest_received,
281 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55282 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09283 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55284 largest_received, smallest_received, least_unacked, kSendFeedback);
285 }
286
Frank Kastenholz684ea412019-02-13 18:48:18287 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstOnlyPacket(
288 uint64_t packet_number,
289 quic::QuicRstStreamErrorCode error_code,
290 uint64_t largest_received,
291 uint64_t smallest_received,
292 uint64_t least_unacked,
293 size_t bytes_written) {
294 return client_maker_.MakeAckAndRstPacket(
295 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
296 largest_received, smallest_received, least_unacked, kSendFeedback,
297 bytes_written,
298 /*include_stop_sending=*/false);
299 }
300
Ryan Hamilton8d9ee76e2018-05-29 23:52:52301 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23302 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52303 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23304 uint64_t largest_received,
305 uint64_t smallest_received,
306 uint64_t least_unacked,
Yixin Wang0d2c6b7e12017-08-16 21:12:55307 size_t bytes_written) {
308 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09309 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55310 largest_received, smallest_received, least_unacked, kSendFeedback,
Frank Kastenholz684ea412019-02-13 18:48:18311 bytes_written,
312 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55313 }
314
Ryan Hamilton8d9ee76e2018-05-29 23:52:52315 std::unique_ptr<quic::QuicReceivedPacket> ConstructRstPacket(
Fan Yangac867502019-01-28 21:10:23316 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52317 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55318 size_t bytes_written) {
319 return client_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09320 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18321 bytes_written,
322 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55323 }
324
Ryan Hamilton8d9ee76e2018-05-29 23:52:52325 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectRequestPacket(
Fan Yangac867502019-01-28 21:10:23326 uint64_t packet_number,
Lily Chenf11e1292018-11-29 16:42:09327 RequestPriority request_priority = LOWEST) {
Ryan Hamilton0239aac2018-05-19 00:03:13328 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55329 PopulateConnectRequestIR(&block);
330 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09331 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Lily Chenf11e1292018-11-29 16:42:09332 ConvertRequestPriorityToQuicPriority(request_priority),
333 std::move(block), 0, nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55334 }
335
Ryan Hamilton8d9ee76e2018-05-29 23:52:52336 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectAuthRequestPacket(
Fan Yangac867502019-01-28 21:10:23337 uint64_t packet_number) {
Ryan Hamilton0239aac2018-05-19 00:03:13338 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55339 PopulateConnectRequestIR(&block);
340 block["proxy-authorization"] = "Basic Zm9vOmJhcg==";
341 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09342 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Yixin Wang7a3f1b8d2018-01-17 21:40:48343 ConvertRequestPriorityToQuicPriority(LOWEST), std::move(block), 0,
344 nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55345 }
346
Ryan Hamilton8d9ee76e2018-05-29 23:52:52347 std::unique_ptr<quic::QuicReceivedPacket> ConstructDataPacket(
Fan Yangac867502019-01-28 21:10:23348 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52349 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41350 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09351 return client_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41352 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55353 }
354
Renjied172e812019-01-16 05:12:35355 std::unique_ptr<quic::QuicReceivedPacket> ConstructMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23356 uint64_t packet_number,
Renjied172e812019-01-16 05:12:35357 quic::QuicStreamOffset offset,
358 const std::vector<std::string> data_writes) {
359 return client_maker_.MakeMultipleDataFramesPacket(
360 packet_number, client_data_stream_id1_, !kIncludeVersion, !kFin, offset,
361 data_writes);
362 }
363
Ryan Hamilton8d9ee76e2018-05-29 23:52:52364 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndDataPacket(
Fan Yangac867502019-01-28 21:10:23365 uint64_t packet_number,
366 uint64_t largest_received,
367 uint64_t smallest_received,
368 uint64_t least_unacked,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52369 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41370 quic::QuicStringPiece data) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55371 return client_maker_.MakeAckAndDataPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09372 packet_number, !kIncludeVersion, client_data_stream_id1_,
373 largest_received, smallest_received, least_unacked, !kFin, offset,
Renjief49758b2019-01-11 23:32:41374 data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55375 }
376
Renjied172e812019-01-16 05:12:35377 std::unique_ptr<quic::QuicReceivedPacket>
378 ConstructAckAndMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23379 uint64_t packet_number,
380 uint64_t largest_received,
381 uint64_t smallest_received,
382 uint64_t least_unacked,
Renjied172e812019-01-16 05:12:35383 quic::QuicStreamOffset offset,
384 const std::vector<std::string> data_writes) {
385 return client_maker_.MakeAckAndMultipleDataFramesPacket(
386 packet_number, !kIncludeVersion, client_data_stream_id1_,
387 largest_received, smallest_received, least_unacked, !kFin, offset,
388 data_writes);
389 }
390
Ryan Hamilton8d9ee76e2018-05-29 23:52:52391 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckPacket(
Fan Yangac867502019-01-28 21:10:23392 uint64_t packet_number,
393 uint64_t largest_received,
394 uint64_t smallest_received,
395 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55396 return client_maker_.MakeAckPacket(packet_number, largest_received,
397 smallest_received, least_unacked,
398 kSendFeedback);
399 }
400
401 // Helper functions for constructing packets sent by the server
402
Ryan Hamilton8d9ee76e2018-05-29 23:52:52403 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerRstPacket(
Fan Yangac867502019-01-28 21:10:23404 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52405 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55406 size_t bytes_written) {
407 return server_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09408 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18409 bytes_written,
410 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55411 }
412
Ryan Hamilton8d9ee76e2018-05-29 23:52:52413 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataPacket(
Fan Yangac867502019-01-28 21:10:23414 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52415 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41416 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09417 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41418 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55419 }
420
Ryan Hamilton8d9ee76e2018-05-29 23:52:52421 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataFinPacket(
Fan Yangac867502019-01-28 21:10:23422 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52423 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41424 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09425 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41426 !kIncludeVersion, kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55427 }
428
Ryan Hamilton8d9ee76e2018-05-29 23:52:52429 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerConnectReplyPacket(
Fan Yangac867502019-01-28 21:10:23430 uint64_t packet_number,
Yixin Wang0d2c6b7e12017-08-16 21:12:55431 bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13432 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55433 block[":status"] = "200";
434
435 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09436 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55437 std::move(block), nullptr, &response_offset_);
438 }
439
Ryan Hamilton8d9ee76e2018-05-29 23:52:52440 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23441 ConstructServerConnectAuthReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13442 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55443 block[":status"] = "407";
444 block["proxy-authenticate"] = "Basic realm=\"MyRealm1\"";
445 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09446 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55447 std::move(block), nullptr, &response_offset_);
448 }
449
Ryan Hamilton8d9ee76e2018-05-29 23:52:52450 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23451 ConstructServerConnectRedirectReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13452 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55453 block[":status"] = "302";
454 block["location"] = kRedirectUrl;
455 block["set-cookie"] = "foo=bar";
456 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09457 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55458 std::move(block), nullptr, &response_offset_);
459 }
460
Ryan Hamilton8d9ee76e2018-05-29 23:52:52461 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23462 ConstructServerConnectErrorReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13463 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55464 block[":status"] = "500";
465
466 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09467 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55468 std::move(block), nullptr, &response_offset_);
469 }
470
471 void AssertConnectSucceeds() {
472 TestCompletionCallback callback;
473 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
474 ASSERT_THAT(callback.WaitForResult(), IsOk());
475 }
476
477 void AssertConnectFails(int result) {
478 TestCompletionCallback callback;
479 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
480 ASSERT_EQ(result, callback.WaitForResult());
481 }
482
483 void ResumeAndRun() {
484 // Run until the pause, if the provider isn't paused yet.
485 SequencedSocketData* data = mock_quic_data_.GetSequencedSocketData();
486 data->RunUntilPaused();
487 data->Resume();
488 base::RunLoop().RunUntilIdle();
489 }
490
491 void AssertWriteReturns(const char* data, int len, int rv) {
Victor Costan9c7302b2018-08-27 16:39:44492 scoped_refptr<IOBufferWithSize> buf =
493 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55494 memcpy(buf->data(), data, len);
495 EXPECT_EQ(rv,
[email protected]578968d42017-12-13 15:39:32496 sock_->Write(buf.get(), buf->size(), write_callback_.callback(),
497 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55498 }
499
500 void AssertSyncWriteSucceeds(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44501 scoped_refptr<IOBufferWithSize> buf =
502 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55503 memcpy(buf->data(), data, len);
Bence Békyd8a21fc32018-06-27 18:29:58504 EXPECT_EQ(len,
505 sock_->Write(buf.get(), buf->size(), CompletionOnceCallback(),
506 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55507 }
508
509 void AssertSyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44510 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Bence Békyd8a21fc32018-06-27 18:29:58511 ASSERT_EQ(len, sock_->Read(buf.get(), len, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:13512 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55513 ASSERT_TRUE(sock_->IsConnected());
514 }
515
516 void AssertAsyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44517 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55518 ASSERT_EQ(ERR_IO_PENDING,
519 sock_->Read(buf.get(), len, read_callback_.callback()));
520 EXPECT_TRUE(sock_->IsConnected());
521
522 ResumeAndRun();
523
524 EXPECT_EQ(len, read_callback_.WaitForResult());
525 EXPECT_TRUE(sock_->IsConnected());
Ryan Hamilton0239aac2018-05-19 00:03:13526 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55527 }
528
529 void AssertReadStarts(const char* data, int len) {
530 // Issue the read, which will be completed asynchronously.
Victor Costan9c7302b2018-08-27 16:39:44531 read_buf_ = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55532 ASSERT_EQ(ERR_IO_PENDING,
533 sock_->Read(read_buf_.get(), len, read_callback_.callback()));
534 EXPECT_TRUE(sock_->IsConnected());
535 }
536
537 void AssertReadReturns(const char* data, int len) {
538 EXPECT_TRUE(sock_->IsConnected());
539
540 // Now the read will return.
541 EXPECT_EQ(len, read_callback_.WaitForResult());
Ryan Hamilton0239aac2018-05-19 00:03:13542 ASSERT_EQ(spdy::SpdyString(data, len),
543 spdy::SpdyString(read_buf_->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55544 }
545
Renjief49758b2019-01-11 23:32:41546 quic::QuicString ConstructDataHeader(size_t body_len) {
547 if (version_ != quic::QUIC_VERSION_99) {
548 return "";
549 }
550 quic::HttpEncoder encoder;
551 std::unique_ptr<char[]> buffer;
552 auto header_length = encoder.SerializeDataFrameHeader(body_len, &buffer);
553 return quic::QuicString(buffer.get(), header_length);
554 }
555
Ryan Hamilton8d9ee76e2018-05-29 23:52:52556 const quic::QuicTransportVersion version_;
Ryan Hamilton47cf9d12018-10-17 04:33:09557 const quic::QuicStreamId client_data_stream_id1_;
Yixin Wang079ad542018-01-11 04:06:05558 const bool client_headers_include_h2_stream_dependency_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55559
560 // order of destruction of these members matter
Ryan Hamilton8d9ee76e2018-05-29 23:52:52561 quic::MockClock clock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55562 MockQuicData mock_quic_data_;
563 std::unique_ptr<QuicChromiumConnectionHelper> helper_;
564 std::unique_ptr<QuicChromiumClientSession> session_;
565 std::unique_ptr<QuicChromiumClientSession::Handle> session_handle_;
566 std::unique_ptr<QuicProxyClientSocket> sock_;
567
568 BoundTestNetLog net_log_;
569
Ryan Hamilton8d9ee76e2018-05-29 23:52:52570 quic::test::MockSendAlgorithm* send_algorithm_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55571 scoped_refptr<TestTaskRunner> runner_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55572
573 std::unique_ptr<QuicChromiumAlarmFactory> alarm_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52574 testing::StrictMock<quic::test::MockQuicConnectionVisitor> visitor_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55575 TransportSecurityState transport_security_state_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52576 quic::QuicCryptoClientConfig crypto_config_;
577 quic::QuicClientPushPromiseIndex push_promise_index_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55578
Ryan Hamilton8d9ee76e2018-05-29 23:52:52579 const quic::QuicConnectionId connection_id_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55580 QuicTestPacketMaker client_maker_;
581 QuicTestPacketMaker server_maker_;
582 IPEndPoint peer_addr_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52583 quic::test::MockRandom random_generator_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55584 ProofVerifyDetailsChromium verify_details_;
585 MockCryptoClientStreamFactory crypto_client_stream_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52586 quic::QuicStreamOffset header_stream_offset_;
587 quic::QuicStreamOffset response_offset_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55588
589 std::string user_agent_;
590 HostPortPair proxy_host_port_;
591 HostPortPair endpoint_host_port_;
592 HttpAuthCache http_auth_cache_;
593 std::unique_ptr<MockHostResolverBase> host_resolver_;
594 std::unique_ptr<HttpAuthHandlerRegistryFactory> http_auth_handler_factory_;
595
596 TestCompletionCallback read_callback_;
597 scoped_refptr<IOBuffer> read_buf_;
598
599 TestCompletionCallback write_callback_;
600
601 DISALLOW_COPY_AND_ASSIGN(QuicProxyClientSocketTest);
602};
603
604TEST_P(QuicProxyClientSocketTest, ConnectSendsCorrectRequest) {
Zhongyi Shi32f2fd02018-04-16 18:23:43605 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
606 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
607 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55608 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
609 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52610 SYNCHRONOUS,
611 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55612
613 Initialize();
614
615 ASSERT_FALSE(sock_->IsConnected());
616
617 AssertConnectSucceeds();
618
619 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
620 ASSERT_TRUE(response != nullptr);
621 ASSERT_EQ(200, response->headers->response_code());
622}
623
624TEST_P(QuicProxyClientSocketTest, ConnectWithAuthRequested) {
Zhongyi Shi32f2fd02018-04-16 18:23:43625 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
626 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
627 mock_quic_data_.AddRead(ASYNC,
628 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55629 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
630 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52631 SYNCHRONOUS,
632 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55633
634 Initialize();
635
636 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
637
638 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
639 ASSERT_TRUE(response != nullptr);
640 ASSERT_EQ(407, response->headers->response_code());
641}
642
643TEST_P(QuicProxyClientSocketTest, ConnectWithAuthCredentials) {
Zhongyi Shi32f2fd02018-04-16 18:23:43644 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
645 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectAuthRequestPacket(2));
646 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55647 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
648 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52649 SYNCHRONOUS,
650 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55651
652 Initialize();
653
654 // Add auth to cache
655 const base::string16 kFoo(base::ASCIIToUTF16("foo"));
656 const base::string16 kBar(base::ASCIIToUTF16("bar"));
657 http_auth_cache_.Add(GURL(kProxyUrl), "MyRealm1", HttpAuth::AUTH_SCHEME_BASIC,
658 "Basic realm=MyRealm1", AuthCredentials(kFoo, kBar),
659 "/");
660
661 AssertConnectSucceeds();
662
663 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
664 ASSERT_TRUE(response != nullptr);
665 ASSERT_EQ(200, response->headers->response_code());
666}
667
668TEST_P(QuicProxyClientSocketTest, ConnectRedirects) {
Zhongyi Shi32f2fd02018-04-16 18:23:43669 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
670 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
671 mock_quic_data_.AddRead(ASYNC,
672 ConstructServerConnectRedirectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55673 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
674 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52675 SYNCHRONOUS,
676 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55677
678 Initialize();
679
Eric Roman74103c72019-02-21 00:23:12680 AssertConnectFails(ERR_HTTPS_PROXY_TUNNEL_RESPONSE_REDIRECT);
Yixin Wang0d2c6b7e12017-08-16 21:12:55681
682 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
683 ASSERT_TRUE(response != nullptr);
684
685 const HttpResponseHeaders* headers = response->headers.get();
686 ASSERT_EQ(302, headers->response_code());
687 ASSERT_FALSE(headers->HasHeader("set-cookie"));
688 ASSERT_TRUE(headers->HasHeaderValue("content-length", "0"));
689
690 std::string location;
691 ASSERT_TRUE(headers->IsRedirect(&location));
692 ASSERT_EQ(location, kRedirectUrl);
693}
694
695TEST_P(QuicProxyClientSocketTest, ConnectFails) {
Zhongyi Shi32f2fd02018-04-16 18:23:43696 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
697 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Yixin Wang0d2c6b7e12017-08-16 21:12:55698 mock_quic_data_.AddRead(ASYNC, 0); // EOF
699
700 Initialize();
701
702 ASSERT_FALSE(sock_->IsConnected());
703
704 AssertConnectFails(ERR_QUIC_PROTOCOL_ERROR);
705
706 ASSERT_FALSE(sock_->IsConnected());
707}
708
709TEST_P(QuicProxyClientSocketTest, WasEverUsedReturnsCorrectValue) {
Zhongyi Shi32f2fd02018-04-16 18:23:43710 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
711 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
712 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55713 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
714 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52715 SYNCHRONOUS,
716 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55717
718 Initialize();
719
720 EXPECT_TRUE(sock_->WasEverUsed()); // Used due to crypto handshake
721 AssertConnectSucceeds();
722 EXPECT_TRUE(sock_->WasEverUsed());
723 sock_->Disconnect();
724 EXPECT_TRUE(sock_->WasEverUsed());
725}
726
727TEST_P(QuicProxyClientSocketTest, GetPeerAddressReturnsCorrectValues) {
Zhongyi Shi32f2fd02018-04-16 18:23:43728 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
729 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
730 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55731 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
732 mock_quic_data_.AddRead(ASYNC, 0); // EOF
733
734 Initialize();
735
736 IPEndPoint addr;
737 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
738
739 AssertConnectSucceeds();
740 EXPECT_TRUE(sock_->IsConnected());
741 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsOk());
742
743 ResumeAndRun();
744
745 EXPECT_FALSE(sock_->IsConnected());
746 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
747
748 sock_->Disconnect();
749
750 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
751}
752
753TEST_P(QuicProxyClientSocketTest, IsConnectedAndIdle) {
Zhongyi Shi32f2fd02018-04-16 18:23:43754 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
755 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
756 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55757 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
758
Renjief49758b2019-01-11 23:32:41759 quic::QuicString header = ConstructDataHeader(kLen1);
760 mock_quic_data_.AddRead(
761 ASYNC,
762 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43763 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55764 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52765 mock_quic_data_.AddWrite(
766 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55767
768 Initialize();
769
770 EXPECT_FALSE(sock_->IsConnectedAndIdle());
771
772 AssertConnectSucceeds();
773
774 EXPECT_TRUE(sock_->IsConnectedAndIdle());
775
776 // The next read is consumed and buffered.
777 ResumeAndRun();
778
779 EXPECT_FALSE(sock_->IsConnectedAndIdle());
780
781 AssertSyncReadEquals(kMsg1, kLen1);
782
783 EXPECT_TRUE(sock_->IsConnectedAndIdle());
784}
785
786TEST_P(QuicProxyClientSocketTest, GetTotalReceivedBytes) {
Zhongyi Shi32f2fd02018-04-16 18:23:43787 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
788 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
789 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55790 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
791
Renjief49758b2019-01-11 23:32:41792 quic::QuicString header = ConstructDataHeader(kLen333);
793 mock_quic_data_.AddRead(
794 ASYNC, ConstructServerDataPacket(
795 2, 0, header + quic::QuicString(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:43796 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55797 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52798 mock_quic_data_.AddWrite(
799 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55800
801 Initialize();
802
803 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
804
805 AssertConnectSucceeds();
806
807 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
808
809 // The next read is consumed and buffered.
810 ResumeAndRun();
811
812 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
813
814 // The payload from the single large data frame will be read across
815 // two different reads.
816 AssertSyncReadEquals(kMsg33, kLen33);
817
Renjief49758b2019-01-11 23:32:41818 EXPECT_EQ((int64_t)(kLen33 + header.length()),
819 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55820
821 AssertSyncReadEquals(kMsg3, kLen3);
822
Renjief49758b2019-01-11 23:32:41823 EXPECT_EQ((int64_t)(kLen333 + header.length()),
824 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55825}
826
Lily Chenf11e1292018-11-29 16:42:09827TEST_P(QuicProxyClientSocketTest, SetStreamPriority) {
828 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
829 mock_quic_data_.AddWrite(SYNCHRONOUS,
830 ConstructConnectRequestPacket(2, HIGHEST));
831 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
832 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
833 mock_quic_data_.AddWrite(
834 SYNCHRONOUS,
835 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
836
837 Initialize();
838
839 sock_->SetStreamPriority(HIGHEST);
840 AssertConnectSucceeds();
841}
Yixin Wang0d2c6b7e12017-08-16 21:12:55842
843TEST_P(QuicProxyClientSocketTest, WriteSendsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43844 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
845 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
846 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55847 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:41848 if (version_ == quic::QUIC_VERSION_99) {
849 quic::QuicString header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:41850 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:35851 SYNCHRONOUS,
852 ConstructAckAndMultipleDataFramesPacket(
853 3, 1, 1, 1, 0, {header, quic::QuicString(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41854 quic::QuicString header2 = ConstructDataHeader(kLen2);
Renjied172e812019-01-16 05:12:35855 mock_quic_data_.AddWrite(SYNCHRONOUS,
856 ConstructMultipleDataFramesPacket(
857 4, kLen1 + header.length(),
858 {header2, quic::QuicString(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:41859 mock_quic_data_.AddWrite(
860 SYNCHRONOUS,
Renjied172e812019-01-16 05:12:35861 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED,
Renjief49758b2019-01-11 23:32:41862 kLen1 + kLen2 + header.length() + header2.length()));
863 } else {
864 mock_quic_data_.AddWrite(
865 SYNCHRONOUS, ConstructAckAndDataPacket(3, 1, 1, 1, 0,
866 quic::QuicString(kMsg1, kLen1)));
867 mock_quic_data_.AddWrite(
868 SYNCHRONOUS,
869 ConstructDataPacket(4, kLen1, quic::QuicString(kMsg2, kLen2)));
870 mock_quic_data_.AddWrite(
871 SYNCHRONOUS,
872 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED, kLen1 + kLen2));
873 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55874
875 Initialize();
876
877 AssertConnectSucceeds();
878
879 AssertSyncWriteSucceeds(kMsg1, kLen1);
880 AssertSyncWriteSucceeds(kMsg2, kLen2);
881}
882
883TEST_P(QuicProxyClientSocketTest, WriteSplitsLargeDataIntoMultiplePackets) {
Renjief49758b2019-01-11 23:32:41884 int write_packet_index = 1;
885 mock_quic_data_.AddWrite(SYNCHRONOUS,
886 ConstructSettingsPacket(write_packet_index++));
887 mock_quic_data_.AddWrite(SYNCHRONOUS,
888 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:43889 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55890 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:41891 quic::QuicString header = ConstructDataHeader(kLen1);
892 if (version_ != quic::QUIC_VERSION_99) {
893 mock_quic_data_.AddWrite(
894 SYNCHRONOUS, ConstructAckAndDataPacket(write_packet_index++, 1, 1, 1, 0,
895 quic::QuicString(kMsg1, kLen1)));
896 } else {
Renjied172e812019-01-16 05:12:35897 mock_quic_data_.AddWrite(SYNCHRONOUS,
898 ConstructAckAndMultipleDataFramesPacket(
899 write_packet_index++, 1, 1, 1, 0,
900 {header, quic::QuicString(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41901 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55902
903 // Expect |kNumDataPackets| data packets, each containing the max possible
904 // amount of data.
Renjied172e812019-01-16 05:12:35905 int numDataPackets = 3;
906 std::string data(numDataPackets * quic::kDefaultMaxPacketSize, 'x');
Renjief49758b2019-01-11 23:32:41907 quic::QuicStreamOffset offset = kLen1 + header.length();
Renjied172e812019-01-16 05:12:35908
Renjief49758b2019-01-11 23:32:41909 if (version_ == quic::QUIC_VERSION_99) {
Renjied172e812019-01-16 05:12:35910 numDataPackets++;
Renjief49758b2019-01-11 23:32:41911 }
Renjied172e812019-01-16 05:12:35912 size_t total_data_length = 0;
913 for (int i = 0; i < numDataPackets; ++i) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55914 size_t max_packet_data_length = GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52915 quic::kDefaultMaxPacketSize, version_, !kIncludeVersion,
916 !kIncludeDiversificationNonce, quic::PACKET_8BYTE_CONNECTION_ID,
917 quic::PACKET_1BYTE_PACKET_NUMBER, offset);
Renjied172e812019-01-16 05:12:35918 if (version_ == quic::QUIC_VERSION_99 && i == 0) {
919 // 3973 is the data frame length from packet length.
920 quic::QuicString header2 = ConstructDataHeader(3973);
921 mock_quic_data_.AddWrite(
922 SYNCHRONOUS,
923 ConstructMultipleDataFramesPacket(
924 write_packet_index++, offset,
925 {header2,
926 quic::QuicString(data.c_str(), max_packet_data_length - 7)}));
927 offset += max_packet_data_length - header2.length() - 1;
928 } else if (version_ == quic::QUIC_VERSION_99 && i == numDataPackets - 1) {
929 mock_quic_data_.AddWrite(
930 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, offset,
931 quic::QuicString(data.c_str(), 7)));
932 offset += 7;
933 } else {
934 mock_quic_data_.AddWrite(
935 SYNCHRONOUS,
936 ConstructDataPacket(
937 write_packet_index++, offset,
938 quic::QuicString(data.c_str(), max_packet_data_length)));
939 offset += max_packet_data_length;
940 }
941 if (i != 3) {
942 total_data_length += max_packet_data_length;
943 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55944 }
Renjied172e812019-01-16 05:12:35945
Yixin Wang0d2c6b7e12017-08-16 21:12:55946 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41947 SYNCHRONOUS, ConstructRstPacket(write_packet_index++,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52948 quic::QUIC_STREAM_CANCELLED, offset));
Yixin Wang0d2c6b7e12017-08-16 21:12:55949
950 Initialize();
951
952 AssertConnectSucceeds();
953
954 // Make a small write. An ACK and STOP_WAITING will be bundled. This prevents
955 // ACK and STOP_WAITING from being bundled with the subsequent large write.
956 // This allows the test code for computing the size of data sent in each
957 // packet to not become too complicated.
958 AssertSyncWriteSucceeds(kMsg1, kLen1);
959
960 // Make large write that should be split up
961 AssertSyncWriteSucceeds(data.c_str(), total_data_length);
962}
963
964// ----------- Read
965
966TEST_P(QuicProxyClientSocketTest, ReadReadsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43967 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
968 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
969 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55970 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
971
Renjief49758b2019-01-11 23:32:41972 quic::QuicString header = ConstructDataHeader(kLen1);
973 mock_quic_data_.AddRead(
974 ASYNC,
975 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43976 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55977 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52978 mock_quic_data_.AddWrite(
979 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55980
981 Initialize();
982
983 AssertConnectSucceeds();
984
985 ResumeAndRun();
986 AssertSyncReadEquals(kMsg1, kLen1);
987}
988
989TEST_P(QuicProxyClientSocketTest, ReadDataFromBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:43990 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
991 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
992 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55993 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
994
Renjief49758b2019-01-11 23:32:41995 quic::QuicString header = ConstructDataHeader(kLen1);
996 mock_quic_data_.AddRead(
997 ASYNC,
998 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43999 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551000 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1001
Renjief49758b2019-01-11 23:32:411002 quic::QuicString header2 = ConstructDataHeader(kLen2);
1003 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1004 3, kLen1 + header.length(),
1005 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551006 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1007
1008 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521009 SYNCHRONOUS,
1010 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551011
1012 Initialize();
1013
1014 AssertConnectSucceeds();
1015
1016 ResumeAndRun();
1017 AssertSyncReadEquals(kMsg1, kLen1);
1018
1019 ResumeAndRun();
1020 AssertSyncReadEquals(kMsg2, kLen2);
1021}
1022
1023TEST_P(QuicProxyClientSocketTest, ReadDataMultipleBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431024 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1025 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1026 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551027 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1028
Renjief49758b2019-01-11 23:32:411029 quic::QuicString header = ConstructDataHeader(kLen1);
1030 mock_quic_data_.AddRead(
1031 ASYNC,
1032 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431033 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411034 quic::QuicString header2 = ConstructDataHeader(kLen2);
1035 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1036 3, kLen1 + header.length(),
1037 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551038 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1039
1040 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521041 SYNCHRONOUS,
1042 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551043
1044 Initialize();
1045
1046 AssertConnectSucceeds();
1047
1048 // The next two reads are consumed and buffered.
1049 ResumeAndRun();
1050
1051 AssertSyncReadEquals(kMsg1, kLen1);
1052 AssertSyncReadEquals(kMsg2, kLen2);
1053}
1054
1055TEST_P(QuicProxyClientSocketTest, LargeReadWillMergeDataFromDifferentFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431056 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1057 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1058 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551059 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1060
Renjief49758b2019-01-11 23:32:411061 quic::QuicString header = ConstructDataHeader(kLen3);
1062 mock_quic_data_.AddRead(
1063 ASYNC,
1064 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg3, kLen3)));
Zhongyi Shi32f2fd02018-04-16 18:23:431065 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411066 quic::QuicString header2 = ConstructDataHeader(kLen3);
1067 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1068 3, kLen3 + header.length(),
1069 header2 + quic::QuicString(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551070 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1071
1072 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521073 SYNCHRONOUS,
1074 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551075
1076 Initialize();
1077
1078 AssertConnectSucceeds();
1079
1080 // The next two reads are consumed and buffered.
1081 ResumeAndRun();
1082 // The payload from two data frames, each with kMsg3 will be combined
1083 // together into a single read().
1084 AssertSyncReadEquals(kMsg33, kLen33);
1085}
1086
1087TEST_P(QuicProxyClientSocketTest, MultipleShortReadsThenMoreRead) {
Zhongyi Shi32f2fd02018-04-16 18:23:431088 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1089 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1090 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551091 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1092
1093 int offset = 0;
1094
Renjief49758b2019-01-11 23:32:411095 quic::QuicString header = ConstructDataHeader(kLen1);
1096 mock_quic_data_.AddRead(
1097 ASYNC, ConstructServerDataPacket(
1098 2, offset, header + quic::QuicString(kMsg1, kLen1)));
1099 offset += kLen1 + header.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431100 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551101
Renjief49758b2019-01-11 23:32:411102 quic::QuicString header2 = ConstructDataHeader(kLen3);
1103 mock_quic_data_.AddRead(
1104 ASYNC, ConstructServerDataPacket(
1105 3, offset, header2 + quic::QuicString(kMsg3, kLen3)));
1106 offset += kLen3 + header2.length();
1107 mock_quic_data_.AddRead(
1108 ASYNC, ConstructServerDataPacket(
1109 4, offset, header2 + quic::QuicString(kMsg3, kLen3)));
1110 offset += kLen3 + header2.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431111 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(4, 4, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551112
Renjief49758b2019-01-11 23:32:411113 quic::QuicString header3 = ConstructDataHeader(kLen2);
1114 mock_quic_data_.AddRead(
1115 ASYNC, ConstructServerDataPacket(
1116 5, offset, header3 + quic::QuicString(kMsg2, kLen2)));
1117 offset += kLen2 + header3.length();
Yixin Wang0d2c6b7e12017-08-16 21:12:551118 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1119
1120 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521121 SYNCHRONOUS,
1122 ConstructAckAndRstPacket(5, quic::QUIC_STREAM_CANCELLED, 5, 5, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551123
1124 Initialize();
1125
1126 AssertConnectSucceeds();
1127
1128 // The next 4 reads are consumed and buffered.
1129 ResumeAndRun();
1130
1131 AssertSyncReadEquals(kMsg1, kLen1);
1132 // The payload from two data frames, each with kMsg3 will be combined
1133 // together into a single read().
1134 AssertSyncReadEquals(kMsg33, kLen33);
1135 AssertSyncReadEquals(kMsg2, kLen2);
1136}
1137
1138TEST_P(QuicProxyClientSocketTest, ReadWillSplitDataFromLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431139 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1140 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1141 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551142 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1143
Renjief49758b2019-01-11 23:32:411144 quic::QuicString header = ConstructDataHeader(kLen1);
1145 mock_quic_data_.AddRead(
1146 ASYNC,
1147 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431148 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411149 quic::QuicString header2 = ConstructDataHeader(kLen33);
1150 mock_quic_data_.AddRead(
1151 ASYNC,
1152 ConstructServerDataPacket(3, kLen1 + header.length(),
1153 header2 + quic::QuicString(kMsg33, kLen33)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551154 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1155
1156 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521157 SYNCHRONOUS,
1158 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551159
1160 Initialize();
1161
1162 AssertConnectSucceeds();
1163
1164 // The next 2 reads are consumed and buffered.
1165 ResumeAndRun();
1166
1167 AssertSyncReadEquals(kMsg1, kLen1);
1168 // The payload from the single large data frame will be read across
1169 // two different reads.
1170 AssertSyncReadEquals(kMsg3, kLen3);
1171 AssertSyncReadEquals(kMsg3, kLen3);
1172}
1173
1174TEST_P(QuicProxyClientSocketTest, MultipleReadsFromSameLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431175 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1176 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1177 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551178 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1179
Renjief49758b2019-01-11 23:32:411180 quic::QuicString header = ConstructDataHeader(kLen333);
1181 mock_quic_data_.AddRead(
1182 ASYNC, ConstructServerDataPacket(
1183 2, 0, header + quic::QuicString(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:431184 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551185 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1186
Ryan Hamilton8d9ee76e2018-05-29 23:52:521187 mock_quic_data_.AddWrite(
1188 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551189
1190 Initialize();
1191
1192 AssertConnectSucceeds();
1193
1194 // The next read is consumed and buffered.
1195 ResumeAndRun();
1196
1197 // The payload from the single large data frame will be read across
1198 // two different reads.
1199 AssertSyncReadEquals(kMsg33, kLen33);
1200
1201 // Now attempt to do a read of more data than remains buffered
Victor Costan9c7302b2018-08-27 16:39:441202 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kLen33);
Bence Békyd8a21fc32018-06-27 18:29:581203 ASSERT_EQ(kLen3, sock_->Read(buf.get(), kLen33, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:131204 ASSERT_EQ(spdy::SpdyString(kMsg3, kLen3),
1205 spdy::SpdyString(buf->data(), kLen3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551206 ASSERT_TRUE(sock_->IsConnected());
1207}
1208
1209TEST_P(QuicProxyClientSocketTest, ReadAuthResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431210 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1211 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1212 mock_quic_data_.AddRead(ASYNC,
1213 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551214 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1215
Renjief49758b2019-01-11 23:32:411216 quic::QuicString header = ConstructDataHeader(kLen1);
1217 mock_quic_data_.AddRead(
1218 ASYNC,
1219 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431220 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411221 quic::QuicString header2 = ConstructDataHeader(kLen2);
1222 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1223 3, kLen1 + header.length(),
1224 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551225 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1226
1227 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521228 SYNCHRONOUS,
1229 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551230
1231 Initialize();
1232
1233 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
1234
1235 // The next two reads are consumed and buffered.
1236 ResumeAndRun();
1237
1238 AssertSyncReadEquals(kMsg1, kLen1);
1239 AssertSyncReadEquals(kMsg2, kLen2);
1240}
1241
1242TEST_P(QuicProxyClientSocketTest, ReadErrorResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431243 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1244 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1245 mock_quic_data_.AddRead(ASYNC,
1246 ConstructServerConnectErrorReplyPacket(1, !kFin));
Renjief49758b2019-01-11 23:32:411247 quic::QuicString header = ConstructDataHeader(kLen1);
1248 mock_quic_data_.AddRead(
1249 SYNCHRONOUS,
1250 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431251 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411252 quic::QuicString header2 = ConstructDataHeader(kLen2);
1253 mock_quic_data_.AddRead(
1254 SYNCHRONOUS,
1255 ConstructServerDataPacket(3, kLen1 + header.length(),
1256 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551257 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1258
1259 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521260 SYNCHRONOUS,
1261 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551262 Initialize();
1263
1264 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
1265}
1266
1267// ----------- Reads and Writes
1268
1269TEST_P(QuicProxyClientSocketTest, AsyncReadAroundWrite) {
Renjief49758b2019-01-11 23:32:411270 int write_packet_index = 1;
1271 mock_quic_data_.AddWrite(SYNCHRONOUS,
1272 ConstructSettingsPacket(write_packet_index++));
1273 mock_quic_data_.AddWrite(SYNCHRONOUS,
1274 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431275 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551276 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1277
Renjief49758b2019-01-11 23:32:411278 quic::QuicString header = ConstructDataHeader(kLen1);
1279 mock_quic_data_.AddRead(
1280 ASYNC,
1281 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431282 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjief49758b2019-01-11 23:32:411283 ConstructAckPacket(write_packet_index++, 2, 1, 1));
1284
1285 quic::QuicString header2 = ConstructDataHeader(kLen2);
1286 if (version_ == quic::QUIC_VERSION_99) {
Renjied172e812019-01-16 05:12:351287 mock_quic_data_.AddWrite(SYNCHRONOUS,
1288 ConstructMultipleDataFramesPacket(
1289 write_packet_index++, 0,
1290 {header2, quic::QuicString(kMsg2, kLen2)}));
1291 } else {
Renjief49758b2019-01-11 23:32:411292 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:411293 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, header2.length(),
1294 quic::QuicString(kMsg2, kLen2)));
Renjied172e812019-01-16 05:12:351295 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551296
1297 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1298
Renjief49758b2019-01-11 23:32:411299 quic::QuicString header3 = ConstructDataHeader(kLen3);
1300 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1301 3, kLen1 + header.length(),
1302 header3 + quic::QuicString(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551303 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1304
1305 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:411306 SYNCHRONOUS, ConstructAckAndRstPacket(write_packet_index++,
1307 quic::QUIC_STREAM_CANCELLED, 3, 3,
1308 1, kLen2 + header2.length()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551309
1310 Initialize();
1311
1312 AssertConnectSucceeds();
1313
1314 ResumeAndRun();
1315
1316 AssertSyncReadEquals(kMsg1, kLen1);
1317
1318 AssertReadStarts(kMsg3, kLen3);
1319 // Read should block until after the write succeeds.
1320
1321 AssertSyncWriteSucceeds(kMsg2, kLen2);
1322
1323 ASSERT_FALSE(read_callback_.have_result());
1324 ResumeAndRun();
1325
1326 // Now the read will return.
1327 AssertReadReturns(kMsg3, kLen3);
1328}
1329
1330TEST_P(QuicProxyClientSocketTest, AsyncWriteAroundReads) {
Renjied172e812019-01-16 05:12:351331 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1332 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Zhongyi Shi32f2fd02018-04-16 18:23:431333 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551334 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1335
Renjief49758b2019-01-11 23:32:411336 quic::QuicString header = ConstructDataHeader(kLen1);
1337 mock_quic_data_.AddRead(
1338 ASYNC,
1339 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Renjied172e812019-01-16 05:12:351340 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551341 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1342
Renjief49758b2019-01-11 23:32:411343 quic::QuicString header2 = ConstructDataHeader(kLen3);
1344 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1345 3, kLen1 + header.length(),
1346 header2 + quic::QuicString(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551347 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1348
1349 mock_quic_data_.AddWrite(ASYNC, ERR_IO_PENDING); // Pause
1350
Renjief49758b2019-01-11 23:32:411351 quic::QuicString header3 = ConstructDataHeader(kLen2);
1352 if (version_ != quic::QUIC_VERSION_99) {
1353 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351354 ASYNC, ConstructDataPacket(4, 0, quic::QuicString(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411355 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351356 SYNCHRONOUS, ConstructAckAndDataPacket(5, 3, 3, 1, kLen2,
1357 quic::QuicString(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411358 } else {
1359 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351360 ASYNC, ConstructMultipleDataFramesPacket(
1361 4, 0, {header3, quic::QuicString(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:411362 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351363 ASYNC,
1364 ConstructAckAndDataPacket(5, 3, 3, 1, header3.length() + kLen2,
1365 header3 + quic::QuicString(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411366 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551367
Renjied172e812019-01-16 05:12:351368 mock_quic_data_.AddWrite(
1369 SYNCHRONOUS, ConstructRstPacket(6, quic::QUIC_STREAM_CANCELLED,
1370 kLen2 + kLen2 + 2 * header3.length()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551371
1372 Initialize();
1373
1374 AssertConnectSucceeds();
1375
1376 ResumeAndRun();
1377 AssertSyncReadEquals(kMsg1, kLen1);
1378
1379 // Write should block until the next read completes.
1380 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1381 // asynchronous starting with the second time it's called while the UDP socket
1382 // is write-blocked. Therefore, at least two writes need to be called on
1383 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351384 AssertWriteReturns(kMsg2, kLen2, kLen2);
Yixin Wang0d2c6b7e12017-08-16 21:12:551385 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1386
1387 AssertAsyncReadEquals(kMsg3, kLen3);
1388
1389 ASSERT_FALSE(write_callback_.have_result());
1390
1391 // Now the write will complete
1392 ResumeAndRun();
Yixin Wangdbbd8752018-01-17 21:50:021393 EXPECT_EQ(kLen2, write_callback_.WaitForResult());
Yixin Wang0d2c6b7e12017-08-16 21:12:551394}
1395
1396// ----------- Reading/Writing on Closed socket
1397
1398// Reading from an already closed socket should return 0
1399TEST_P(QuicProxyClientSocketTest, ReadOnClosedSocketReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431400 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1401 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1402 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551403 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1404
1405 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1406
1407 Initialize();
1408
1409 AssertConnectSucceeds();
1410
1411 ResumeAndRun();
1412
1413 ASSERT_FALSE(sock_->IsConnected());
Bence Békyd8a21fc32018-06-27 18:29:581414 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
1415 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
1416 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551417 ASSERT_FALSE(sock_->IsConnectedAndIdle());
1418}
1419
1420// Read pending when socket is closed should return 0
1421TEST_P(QuicProxyClientSocketTest, PendingReadOnCloseReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431422 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1423 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1424 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551425 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1426
1427 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1428
1429 Initialize();
1430
1431 AssertConnectSucceeds();
1432
1433 AssertReadStarts(kMsg1, kLen1);
1434
1435 ResumeAndRun();
1436
1437 ASSERT_EQ(0, read_callback_.WaitForResult());
1438}
1439
1440// Reading from a disconnected socket is an error
1441TEST_P(QuicProxyClientSocketTest, ReadOnDisconnectSocketReturnsNotConnected) {
Zhongyi Shi32f2fd02018-04-16 18:23:431442 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1443 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1444 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551445 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1446 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521447 SYNCHRONOUS,
1448 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551449
1450 Initialize();
1451
1452 AssertConnectSucceeds();
1453
1454 sock_->Disconnect();
1455
1456 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581457 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551458}
1459
1460// Reading data after receiving FIN should return buffered data received before
1461// FIN, then 0.
1462TEST_P(QuicProxyClientSocketTest, ReadAfterFinReceivedReturnsBufferedData) {
Zhongyi Shi32f2fd02018-04-16 18:23:431463 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1464 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1465 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551466 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1467
Renjief49758b2019-01-11 23:32:411468 quic::QuicString header = ConstructDataHeader(kLen1);
Zhongyi Shi32f2fd02018-04-16 18:23:431469 mock_quic_data_.AddRead(ASYNC,
Renjief49758b2019-01-11 23:32:411470 ConstructServerDataFinPacket(
1471 2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431472 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551473 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521474 mock_quic_data_.AddWrite(
1475 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551476
1477 Initialize();
1478
1479 AssertConnectSucceeds();
1480
1481 ResumeAndRun();
1482
1483 AssertSyncReadEquals(kMsg1, kLen1);
Bence Békyd8a21fc32018-06-27 18:29:581484 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
1485 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551486
1487 sock_->Disconnect();
1488 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581489 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551490}
1491
1492// Calling Write() on a closed socket is an error.
1493TEST_P(QuicProxyClientSocketTest, WriteOnClosedStream) {
Zhongyi Shi32f2fd02018-04-16 18:23:431494 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1495 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1496 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551497 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1498
1499 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1500
1501 Initialize();
1502
1503 AssertConnectSucceeds();
1504
1505 ResumeAndRun();
1506
1507 AssertWriteReturns(kMsg1, kLen1, ERR_QUIC_PROTOCOL_ERROR);
1508}
1509
1510// Calling Write() on a disconnected socket is an error.
1511TEST_P(QuicProxyClientSocketTest, WriteOnDisconnectedSocket) {
Zhongyi Shi32f2fd02018-04-16 18:23:431512 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1513 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1514 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551515 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1516 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521517 SYNCHRONOUS,
1518 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551519
1520 Initialize();
1521
1522 AssertConnectSucceeds();
1523
1524 sock_->Disconnect();
1525
1526 AssertWriteReturns(kMsg1, kLen1, ERR_SOCKET_NOT_CONNECTED);
1527}
1528
1529// If the socket is closed with a pending Write(), the callback should be called
1530// with the same error the session was closed with.
1531TEST_P(QuicProxyClientSocketTest, WritePendingOnClose) {
Zhongyi Shi32f2fd02018-04-16 18:23:431532 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1533 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1534 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551535 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1536 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1537
1538 Initialize();
1539
1540 AssertConnectSucceeds();
1541
1542 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1543 // asynchronous starting with the second time it's called while the UDP socket
1544 // is write-blocked. Therefore, at least two writes need to be called on
1545 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351546 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551547
1548 // This second write will be async. This is the pending write that's being
1549 // tested.
1550 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1551
1552 // Make sure the write actually starts.
1553 base::RunLoop().RunUntilIdle();
1554
Ryan Hamilton8d9ee76e2018-05-29 23:52:521555 session_->CloseSessionOnError(ERR_CONNECTION_CLOSED,
Renjieba55fae2018-09-20 03:05:161556 quic::QUIC_INTERNAL_ERROR,
1557 quic::ConnectionCloseBehavior::SILENT_CLOSE);
Yixin Wang0d2c6b7e12017-08-16 21:12:551558
1559 EXPECT_THAT(write_callback_.WaitForResult(), IsError(ERR_CONNECTION_CLOSED));
1560}
1561
1562TEST_P(QuicProxyClientSocketTest, DisconnectWithWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431563 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1564 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1565 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551566 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1567 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1568
1569 Initialize();
1570
1571 AssertConnectSucceeds();
1572
1573 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1574 // asynchronous starting with the second time it's called while the UDP socket
1575 // is write-blocked. Therefore, at least two writes need to be called on
1576 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351577 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551578
1579 // This second write will be async. This is the pending write that's being
1580 // tested.
1581 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1582
1583 // Make sure the write actually starts.
1584 base::RunLoop().RunUntilIdle();
1585
1586 sock_->Disconnect();
1587 EXPECT_FALSE(sock_->IsConnected());
1588
1589 base::RunLoop().RunUntilIdle();
1590
1591 EXPECT_FALSE(sock_->IsConnected());
1592 EXPECT_FALSE(write_callback_.have_result());
1593}
1594
1595// If the socket is Disconnected with a pending Read(), the callback
1596// should not be called.
1597TEST_P(QuicProxyClientSocketTest, DisconnectWithReadPending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431598 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1599 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1600 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551601 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1602 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521603 SYNCHRONOUS,
1604 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551605
1606 Initialize();
1607
1608 AssertConnectSucceeds();
1609
1610 EXPECT_TRUE(sock_->IsConnected());
1611
1612 AssertReadStarts(kMsg1, kLen1);
1613
1614 sock_->Disconnect();
1615 EXPECT_FALSE(sock_->IsConnected());
1616
1617 base::RunLoop().RunUntilIdle();
1618
1619 EXPECT_FALSE(sock_->IsConnected());
1620 EXPECT_FALSE(read_callback_.have_result());
1621}
1622
1623// If the socket is Reset when both a read and write are pending,
1624// both should be called back.
1625TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431626 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1627 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1628 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551629 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1630
1631 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521632 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang2bea3cf2017-11-09 18:11:031633 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:411634 quic::QuicString header = ConstructDataHeader(kLen2);
1635 if (version_ != quic::QUIC_VERSION_99) {
1636 mock_quic_data_.AddWrite(
1637 ASYNC, ConstructAckAndDataPacket(3, 1, 1, 1, 0,
1638 quic::QuicString(kMsg2, kLen2)));
1639 mock_quic_data_.AddWrite(
1640 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1641 2, 2, 1, kLen2));
1642 } else {
Renjief49758b2019-01-11 23:32:411643 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351644 ASYNC, ConstructAckAndMultipleDataFramesPacket(
1645 3, 1, 1, 1, 0, {header, quic::QuicString(kMsg2, kLen2)}));
Frank Kastenholz684ea412019-02-13 18:48:181646 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
1647 4, quic::QUIC_STREAM_CANCELLED, 2,
1648 2, 1, header.length() + kLen2));
Renjief49758b2019-01-11 23:32:411649 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551650
1651 Initialize();
1652
1653 AssertConnectSucceeds();
1654
1655 EXPECT_TRUE(sock_->IsConnected());
1656
1657 AssertReadStarts(kMsg1, kLen1);
1658
1659 // Write should block until the next read completes.
1660 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1661 // asynchronous starting with the second time it's called while the UDP socket
1662 // is write-blocked. Therefore, at least two writes need to be called on
1663 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351664 AssertWriteReturns(kMsg2, kLen2, kLen2);
Renjief49758b2019-01-11 23:32:411665
Yixin Wang0d2c6b7e12017-08-16 21:12:551666 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1667
1668 ResumeAndRun();
1669
1670 EXPECT_TRUE(read_callback_.have_result());
1671 EXPECT_TRUE(write_callback_.have_result());
1672}
1673
1674// Makes sure the proxy client socket's source gets the expected NetLog events
1675// and only the expected NetLog events (No SpdySession events).
1676TEST_P(QuicProxyClientSocketTest, NetLog) {
Zhongyi Shi32f2fd02018-04-16 18:23:431677 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1678 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1679 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551680 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1681
Renjief49758b2019-01-11 23:32:411682 quic::QuicString header = ConstructDataHeader(kLen1);
1683 mock_quic_data_.AddRead(
1684 ASYNC,
1685 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431686 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang2bea3cf2017-11-09 18:11:031687 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521688 mock_quic_data_.AddWrite(
1689 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551690
1691 Initialize();
1692
1693 AssertConnectSucceeds();
1694
1695 ResumeAndRun();
1696 AssertSyncReadEquals(kMsg1, kLen1);
1697
1698 NetLogSource sock_source = sock_->NetLog().source();
1699 sock_.reset();
1700
1701 TestNetLogEntry::List entry_list;
1702 net_log_.GetEntriesForSource(sock_source, &entry_list);
1703
1704 ASSERT_EQ(entry_list.size(), 10u);
1705 EXPECT_TRUE(
1706 LogContainsBeginEvent(entry_list, 0, NetLogEventType::SOCKET_ALIVE));
1707 EXPECT_TRUE(LogContainsEvent(entry_list, 1,
1708 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
1709 NetLogEventPhase::NONE));
1710 EXPECT_TRUE(LogContainsBeginEvent(
1711 entry_list, 2, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1712 EXPECT_TRUE(LogContainsEvent(
1713 entry_list, 3, NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
1714 NetLogEventPhase::NONE));
1715 EXPECT_TRUE(LogContainsEndEvent(
1716 entry_list, 4, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1717 EXPECT_TRUE(LogContainsBeginEvent(
1718 entry_list, 5, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1719 EXPECT_TRUE(LogContainsEvent(
1720 entry_list, 6,
1721 NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
1722 NetLogEventPhase::NONE));
1723 EXPECT_TRUE(LogContainsEndEvent(
1724 entry_list, 7, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1725 EXPECT_TRUE(LogContainsEvent(entry_list, 8,
1726 NetLogEventType::SOCKET_BYTES_RECEIVED,
1727 NetLogEventPhase::NONE));
1728 EXPECT_TRUE(
1729 LogContainsEndEvent(entry_list, 9, NetLogEventType::SOCKET_ALIVE));
1730}
1731
Bence Béky8ddc2492018-06-13 01:02:041732// A helper class that will delete |sock| when the callback is invoked.
Yixin Wang0d2c6b7e12017-08-16 21:12:551733class DeleteSockCallback : public TestCompletionCallbackBase {
1734 public:
1735 explicit DeleteSockCallback(std::unique_ptr<QuicProxyClientSocket>* sock)
Bence Béky8ddc2492018-06-13 01:02:041736 : sock_(sock) {}
Yixin Wang0d2c6b7e12017-08-16 21:12:551737
1738 ~DeleteSockCallback() override {}
1739
Bence Béky8ddc2492018-06-13 01:02:041740 CompletionOnceCallback callback() {
1741 return base::BindOnce(&DeleteSockCallback::OnComplete,
1742 base::Unretained(this));
1743 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551744
1745 private:
1746 void OnComplete(int result) {
1747 sock_->reset(NULL);
1748 SetResult(result);
1749 }
1750
1751 std::unique_ptr<QuicProxyClientSocket>* sock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:551752
1753 DISALLOW_COPY_AND_ASSIGN(DeleteSockCallback);
1754};
1755
1756// If the socket is reset when both a read and write are pending, and the
1757// read callback causes the socket to be deleted, the write callback should
1758// not be called.
1759TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePendingDelete) {
Zhongyi Shi32f2fd02018-04-16 18:23:431760 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1761 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1762 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551763 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1764
1765 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521766 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551767 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:411768 if (version_ != quic::QUIC_VERSION_99) {
1769 mock_quic_data_.AddWrite(
1770 ASYNC, ConstructAckAndDataPacket(3, 1, 1, 1, 0,
1771 quic::QuicString(kMsg1, kLen1)));
1772 mock_quic_data_.AddWrite(
1773 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1774 2, 2, 1, kLen1));
1775 } else {
1776 quic::QuicString header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411777 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351778 ASYNC, ConstructAckAndMultipleDataFramesPacket(
1779 3, 1, 1, 1, 0, {header, quic::QuicString(kMsg1, kLen1)}));
Frank Kastenholz684ea412019-02-13 18:48:181780 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
1781 4, quic::QUIC_STREAM_CANCELLED, 2,
1782 2, 1, header.length() + kLen1));
Renjief49758b2019-01-11 23:32:411783 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551784
1785 Initialize();
1786
1787 AssertConnectSucceeds();
1788
1789 EXPECT_TRUE(sock_->IsConnected());
1790
1791 DeleteSockCallback read_callback(&sock_);
Victor Costan9c7302b2018-08-27 16:39:441792 scoped_refptr<IOBuffer> read_buf = base::MakeRefCounted<IOBuffer>(kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551793 ASSERT_EQ(ERR_IO_PENDING,
1794 sock_->Read(read_buf.get(), kLen1, read_callback.callback()));
1795
1796 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1797 // asynchronous starting with the second time it's called while the UDP socket
1798 // is write-blocked. Therefore, at least two writes need to be called on
1799 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351800 AssertWriteReturns(kMsg1, kLen1, kLen1);
Renjief49758b2019-01-11 23:32:411801
Yixin Wang0d2c6b7e12017-08-16 21:12:551802 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1803
1804 ResumeAndRun();
1805
1806 EXPECT_FALSE(sock_.get());
1807
1808 EXPECT_EQ(0, read_callback.WaitForResult());
1809 EXPECT_FALSE(write_callback_.have_result());
1810}
1811
Victor Costane635086f2019-01-27 05:20:301812INSTANTIATE_TEST_SUITE_P(
Bence Békyce380cb2018-04-26 23:39:551813 VersionIncludeStreamDependencySequence,
Yixin Wang079ad542018-01-11 04:06:051814 QuicProxyClientSocketTest,
Ryan Hamilton8d9ee76e2018-05-29 23:52:521815 ::testing::Combine(
1816 ::testing::ValuesIn(quic::AllSupportedTransportVersions()),
1817 ::testing::Bool()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551818
1819} // namespace test
Dan Zhangf11470172017-09-18 22:02:091820} // namespace net