blob: 633d2d63c8cddf58369451067e522e2c1fdbe4d1 [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>
Eric Orthbe2efac2019-03-06 01:11:118#include <tuple>
Bence Békyd8a21fc32018-06-27 18:29:589#include <utility>
Eric Orthbe2efac2019-03-06 01:11:1110#include <vector>
Bence Béky8f9d7d3952017-10-09 19:58:0411
Sebastien Marchand6d0558fd2019-01-25 16:49:3712#include "base/bind.h"
Bence Béky8f9d7d3952017-10-09 19:58:0413#include "base/memory/ptr_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5514#include "base/run_loop.h"
15#include "base/strings/utf_string_conversions.h"
Lei Zhang52637ed2019-02-20 01:38:3716#include "base/threading/thread_task_runner_handle.h"
Zhongyi Shic16b4102019-02-12 00:37:4017#include "base/time/default_tick_clock.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5518#include "net/dns/mock_host_resolver.h"
19#include "net/http/http_auth_cache.h"
20#include "net/http/http_auth_handler_factory.h"
21#include "net/http/http_response_headers.h"
22#include "net/http/transport_security_state.h"
23#include "net/log/test_net_log.h"
24#include "net/log/test_net_log_util.h"
Victor Vasiliev4f6fb892019-05-31 16:58:3125#include "net/quic/address_utils.h"
Ryan Hamiltona3ee93a72018-08-01 22:03:0826#include "net/quic/crypto/proof_verifier_chromium.h"
27#include "net/quic/mock_crypto_client_stream_factory.h"
28#include "net/quic/mock_quic_data.h"
29#include "net/quic/quic_chromium_alarm_factory.h"
30#include "net/quic/quic_chromium_client_session.h"
31#include "net/quic/quic_chromium_connection_helper.h"
32#include "net/quic/quic_chromium_packet_writer.h"
33#include "net/quic/quic_http_utils.h"
34#include "net/quic/quic_server_info.h"
35#include "net/quic/quic_stream_factory.h"
36#include "net/quic/quic_test_packet_maker.h"
37#include "net/quic/test_task_runner.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5538#include "net/socket/socket_test_util.h"
39#include "net/test/cert_test_util.h"
Victor Vasilievdcdb6192019-02-25 19:49:5640#include "net/test/gtest_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5541#include "net/test/test_data_directory.h"
Bence Béky98447b12018-05-08 03:14:0142#include "net/test/test_with_scoped_task_environment.h"
Victor Vasiliev6bb59d22019-03-08 21:34:5143#include "net/third_party/quiche/src/quic/core/crypto/null_encrypter.h"
44#include "net/third_party/quiche/src/quic/core/quic_utils.h"
45#include "net/third_party/quiche/src/quic/core/tls_client_handshaker.h"
46#include "net/third_party/quiche/src/quic/test_tools/crypto_test_utils.h"
47#include "net/third_party/quiche/src/quic/test_tools/mock_clock.h"
48#include "net/third_party/quiche/src/quic/test_tools/mock_random.h"
49#include "net/third_party/quiche/src/quic/test_tools/quic_connection_peer.h"
50#include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h"
[email protected]578968d42017-12-13 15:39:3251#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5552#include "testing/gmock/include/gmock/gmock.h"
53#include "testing/gtest/include/gtest/gtest.h"
54
55using testing::_;
56using testing::AnyNumber;
57using testing::Return;
58
59namespace {
60
61static const char kOriginHost[] = "www.google.com";
62static const int kOriginPort = 443;
63static const char kProxyUrl[] = "https://myproxy:6121/";
64static const char kProxyHost[] = "myproxy";
65static const int kProxyPort = 6121;
66static const char kUserAgent[] = "Mozilla/1.0";
67static const char kRedirectUrl[] = "https://example.com/";
68
69static const char kMsg1[] = "\0hello!\xff";
70static const int kLen1 = 8;
71static const char kMsg2[] = "\0a2345678\0";
72static const int kLen2 = 10;
73static const char kMsg3[] = "bye!";
74static const int kLen3 = 4;
75static const char kMsg33[] = "bye!bye!";
76static const int kLen33 = kLen3 + kLen3;
77static const char kMsg333[] = "bye!bye!bye!";
78static const int kLen333 = kLen3 + kLen3 + kLen3;
79
80} // anonymous namespace
81
82namespace net {
83namespace test {
84
Michael Warres74ee3ce2017-10-09 15:26:3785class QuicProxyClientSocketTest
Ryan Hamilton8d9ee76e2018-05-29 23:52:5286 : public ::testing::TestWithParam<
Nick Harper23290b82019-05-02 00:02:5687 std::tuple<quic::ParsedQuicVersion, bool>>,
Bence Béky98447b12018-05-08 03:14:0188 public WithScopedTaskEnvironment {
Yixin Wang0d2c6b7e12017-08-16 21:12:5589 protected:
90 static const bool kFin = true;
91 static const bool kIncludeVersion = true;
92 static const bool kIncludeDiversificationNonce = true;
93 static const bool kIncludeCongestionFeedback = true;
94 static const bool kSendFeedback = true;
95
96 static size_t GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:5297 quic::QuicByteCount packet_length,
Nick Harper23290b82019-05-02 00:02:5698 quic::ParsedQuicVersion version,
Yixin Wang0d2c6b7e12017-08-16 21:12:5599 bool include_version,
100 bool include_diversification_nonce,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52101 quic::QuicConnectionIdLength connection_id_length,
102 quic::QuicPacketNumberLength packet_number_length,
103 quic::QuicStreamOffset offset) {
Michael Warres167db3e2019-03-01 21:38:03104 quic::QuicVariableLengthIntegerLength retry_token_length_length =
105 quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
106 quic::QuicVariableLengthIntegerLength length_length =
Nick Harper23290b82019-05-02 00:02:56107 quic::QuicVersionHasLongHeaderLengths(version.transport_version) &&
108 include_version
Michael Warres167db3e2019-03-01 21:38:03109 ? quic::VARIABLE_LENGTH_INTEGER_LENGTH_2
110 : quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
Yixin Wang0d2c6b7e12017-08-16 21:12:55111 size_t min_data_length = 1;
112 size_t min_packet_length =
Ryan Hamilton8d9ee76e2018-05-29 23:52:52113 quic::NullEncrypter(quic::Perspective::IS_CLIENT)
Yixin Wang0d2c6b7e12017-08-16 21:12:55114 .GetCiphertextSize(min_data_length) +
Ryan Hamilton8d9ee76e2018-05-29 23:52:52115 quic::QuicPacketCreator::StreamFramePacketOverhead(
Nick Harper23290b82019-05-02 00:02:56116 version.transport_version, quic::PACKET_8BYTE_CONNECTION_ID,
Michael Warres60637ae2018-06-05 20:03:11117 quic::PACKET_0BYTE_CONNECTION_ID, include_version,
Michael Warres167db3e2019-03-01 21:38:03118 include_diversification_nonce, packet_number_length,
119 retry_token_length_length, length_length, offset);
Yixin Wang0d2c6b7e12017-08-16 21:12:55120
121 DCHECK(packet_length >= min_packet_length);
122 return min_data_length + packet_length - min_packet_length;
123 }
124
125 QuicProxyClientSocketTest()
Yixin Wang079ad542018-01-11 04:06:05126 : version_(std::get<0>(GetParam())),
Nick Harper23290b82019-05-02 00:02:56127 client_data_stream_id1_(
128 quic::QuicUtils::GetHeadersStreamId(version_.transport_version) +
129 quic::QuicUtils::StreamIdDelta(version_.transport_version)),
Yixin Wang079ad542018-01-11 04:06:05130 client_headers_include_h2_stream_dependency_(std::get<1>(GetParam())),
Ryan Hamiltonabad59e2019-06-06 04:02:59131 mock_quic_data_(version_),
Ryan Hamilton8d9ee76e2018-05-29 23:52:52132 crypto_config_(quic::test::crypto_test_utils::ProofVerifierForTesting(),
133 quic::TlsClientHandshaker::CreateSslCtx()),
David Schinazic8281052019-01-24 06:14:17134 connection_id_(quic::test::TestConnectionId(2)),
Yixin Wang0d2c6b7e12017-08-16 21:12:55135 client_maker_(version_,
136 connection_id_,
137 &clock_,
138 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52139 quic::Perspective::IS_CLIENT,
Yixin Wang079ad542018-01-11 04:06:05140 client_headers_include_h2_stream_dependency_),
Yixin Wang0d2c6b7e12017-08-16 21:12:55141 server_maker_(version_,
142 connection_id_,
143 &clock_,
144 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52145 quic::Perspective::IS_SERVER,
Yixin Wang079ad542018-01-11 04:06:05146 false),
Yixin Wang0d2c6b7e12017-08-16 21:12:55147 random_generator_(0),
148 header_stream_offset_(0),
149 response_offset_(0),
150 user_agent_(kUserAgent),
151 proxy_host_port_(kProxyHost, kProxyPort),
152 endpoint_host_port_(kOriginHost, kOriginPort),
153 host_resolver_(new MockCachingHostResolver()),
Eric Orthbe2efac2019-03-06 01:11:11154 http_auth_handler_factory_(HttpAuthHandlerFactory::CreateDefault()) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55155 IPAddress ip(192, 0, 2, 33);
156 peer_addr_ = IPEndPoint(ip, 443);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52157 clock_.AdvanceTime(quic::QuicTime::Delta::FromMilliseconds(20));
Yixin Wang0d2c6b7e12017-08-16 21:12:55158 }
159
160 void SetUp() override {}
161
Yixin Wang2bea3cf2017-11-09 18:11:03162 void TearDown() override {
163 sock_.reset();
164 EXPECT_TRUE(mock_quic_data_.AllReadDataConsumed());
165 EXPECT_TRUE(mock_quic_data_.AllWriteDataConsumed());
166 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55167
168 void Initialize() {
169 std::unique_ptr<MockUDPClientSocket> socket(new MockUDPClientSocket(
170 mock_quic_data_.InitializeAndGetSequencedSocketData(),
171 net_log_.bound().net_log()));
172 socket->Connect(peer_addr_);
173 runner_ = new TestTaskRunner(&clock_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52174 send_algorithm_ = new quic::test::MockSendAlgorithm();
Yixin Wang0d2c6b7e12017-08-16 21:12:55175 EXPECT_CALL(*send_algorithm_, InRecovery()).WillRepeatedly(Return(false));
176 EXPECT_CALL(*send_algorithm_, InSlowStart()).WillRepeatedly(Return(false));
177 EXPECT_CALL(*send_algorithm_, OnPacketSent(_, _, _, _, _))
Dan Zhangf11470172017-09-18 22:02:09178 .Times(testing::AtLeast(1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55179 EXPECT_CALL(*send_algorithm_, GetCongestionWindow())
Zhongyi Shica576df2019-04-12 17:43:40180 .WillRepeatedly(Return(quic::kMaxOutgoingPacketSize));
Yixin Wang0d2c6b7e12017-08-16 21:12:55181 EXPECT_CALL(*send_algorithm_, PacingRate(_))
Ryan Hamilton8d9ee76e2018-05-29 23:52:52182 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Michael Warres74ee3ce2017-10-09 15:26:37183 EXPECT_CALL(*send_algorithm_, CanSend(_)).WillRepeatedly(Return(true));
Yixin Wang0d2c6b7e12017-08-16 21:12:55184 EXPECT_CALL(*send_algorithm_, BandwidthEstimate())
Ryan Hamilton8d9ee76e2018-05-29 23:52:52185 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Yixin Wang0d2c6b7e12017-08-16 21:12:55186 EXPECT_CALL(*send_algorithm_, SetFromConfig(_, _)).Times(AnyNumber());
187 EXPECT_CALL(*send_algorithm_, OnApplicationLimited(_)).Times(AnyNumber());
188 EXPECT_CALL(*send_algorithm_, GetCongestionControlType())
189 .Times(AnyNumber());
190 helper_.reset(
191 new QuicChromiumConnectionHelper(&clock_, &random_generator_));
192 alarm_factory_.reset(new QuicChromiumAlarmFactory(runner_.get(), &clock_));
193
Ryan Hamilton9edcf1a2017-11-22 05:55:17194 QuicChromiumPacketWriter* writer = new QuicChromiumPacketWriter(
195 socket.get(), base::ThreadTaskRunnerHandle::Get().get());
Ryan Hamilton8d9ee76e2018-05-29 23:52:52196 quic::QuicConnection* connection = new quic::QuicConnection(
Victor Vasiliev4f6fb892019-05-31 16:58:31197 connection_id_, net::ToQuicSocketAddress(peer_addr_), helper_.get(),
198 alarm_factory_.get(), writer, true /* owns_writer */,
Nick Harper23290b82019-05-02 00:02:56199 quic::Perspective::IS_CLIENT, quic::test::SupportedVersions(version_));
Yixin Wang0d2c6b7e12017-08-16 21:12:55200 connection->set_visitor(&visitor_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52201 quic::test::QuicConnectionPeer::SetSendAlgorithm(connection,
202 send_algorithm_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55203
204 // Load a certificate that is valid for *.example.org
205 scoped_refptr<X509Certificate> test_cert(
206 ImportCertFromFile(GetTestCertsDirectory(), "wildcard.pem"));
207 EXPECT_TRUE(test_cert.get());
208
209 verify_details_.cert_verify_result.verified_cert = test_cert;
210 verify_details_.cert_verify_result.is_issued_by_known_root = true;
211 crypto_client_stream_factory_.AddProofVerifyDetails(&verify_details_);
212
213 base::TimeTicks dns_end = base::TimeTicks::Now();
214 base::TimeTicks dns_start = dns_end - base::TimeDelta::FromMilliseconds(1);
215
216 session_.reset(new QuicChromiumClientSession(
217 connection, std::move(socket),
218 /*stream_factory=*/nullptr, &crypto_client_stream_factory_, &clock_,
Nick Harper89bc7212018-07-31 19:07:57219 &transport_security_state_, /*ssl_config_service=*/nullptr,
Yixin Wang0d2c6b7e12017-08-16 21:12:55220 base::WrapUnique(static_cast<QuicServerInfo*>(nullptr)),
Paul Jensen8e3c5d32018-02-19 17:06:33221 QuicSessionKey("mail.example.org", 80, PRIVACY_MODE_DISABLED,
222 SocketTag()),
Zhongyi Shi757fcce2018-06-27 05:41:27223 /*require_confirmation=*/false, /*migrate_session_early_v2=*/false,
224 /*migrate_session_on_network_change_v2=*/false,
225 /*default_network=*/NetworkChangeNotifier::kInvalidNetworkHandle,
Zhongyi Shie01f2db2019-02-22 19:53:23226 quic::QuicTime::Delta::FromMilliseconds(
227 kDefaultRetransmittableOnWireTimeoutMillisecs),
Zhongyi Shi32fe14d42019-02-28 00:25:36228 /*migrate_idle_session=*/true,
Zhongyi Shic16b4102019-02-12 00:37:40229 base::TimeDelta::FromSeconds(kDefaultIdleSessionMigrationPeriodSeconds),
Zhongyi Shi73f23ca872017-12-13 18:37:13230 base::TimeDelta::FromSeconds(kMaxTimeOnNonDefaultNetworkSecs),
Zhongyi Shiee760762018-08-01 00:54:29231 kMaxMigrationsToNonDefaultNetworkOnWriteError,
Zhongyi Shi8b1e43f2017-12-13 20:46:30232 kMaxMigrationsToNonDefaultNetworkOnPathDegrading,
Zhongyi Shi5f587cc2017-11-21 23:24:17233 kQuicYieldAfterPacketsRead,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52234 quic::QuicTime::Delta::FromMilliseconds(
235 kQuicYieldAfterDurationMilliseconds),
Zhongyi Shidbce7f412019-02-01 23:16:29236 /*go_away_on_path_degrading*/ false,
Yixin Wang079ad542018-01-11 04:06:05237 client_headers_include_h2_stream_dependency_, /*cert_verify_flags=*/0,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52238 quic::test::DefaultQuicConfig(), &crypto_config_, "CONNECTION_UNKNOWN",
239 dns_start, dns_end, &push_promise_index_, nullptr,
Zhongyi Shic16b4102019-02-12 00:37:40240 base::DefaultTickClock::GetInstance(),
Yixin Wang0d2c6b7e12017-08-16 21:12:55241 base::ThreadTaskRunnerHandle::Get().get(),
242 /*socket_performance_watcher=*/nullptr, net_log_.bound().net_log()));
243
244 writer->set_delegate(session_.get());
245
Ryan Hamilton6c2a2a82017-12-15 02:06:28246 session_handle_ =
247 session_->CreateHandle(HostPortPair("mail.example.org", 80));
Yixin Wang0d2c6b7e12017-08-16 21:12:55248
249 session_->Initialize();
250 TestCompletionCallback callback;
251 EXPECT_THAT(session_->CryptoConnect(callback.callback()), IsOk());
252 EXPECT_TRUE(session_->IsCryptoHandshakeConfirmed());
253
Ramin Halavati683bcaa92018-02-14 08:42:39254 EXPECT_THAT(session_handle_->RequestStream(true, callback.callback(),
255 TRAFFIC_ANNOTATION_FOR_TESTS),
Yixin Wang0d2c6b7e12017-08-16 21:12:55256 IsOk());
257 std::unique_ptr<QuicChromiumClientStream::Handle> stream_handle =
258 session_handle_->ReleaseStream();
259 EXPECT_TRUE(stream_handle->IsOpen());
260
261 sock_.reset(new QuicProxyClientSocket(
262 std::move(stream_handle), std::move(session_handle_), user_agent_,
263 endpoint_host_port_, net_log_.bound(),
Eric Orthbe2efac2019-03-06 01:11:11264 new HttpAuthController(
265 HttpAuth::AUTH_PROXY,
266 GURL("https://" + proxy_host_port_.ToString()), &http_auth_cache_,
267 http_auth_handler_factory_.get(), host_resolver_.get())));
Yixin Wang0d2c6b7e12017-08-16 21:12:55268
269 session_->StartReading();
270 }
271
Ryan Hamilton0239aac2018-05-19 00:03:13272 void PopulateConnectRequestIR(spdy::SpdyHeaderBlock* block) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55273 (*block)[":method"] = "CONNECT";
274 (*block)[":authority"] = endpoint_host_port_.ToString();
275 (*block)["user-agent"] = kUserAgent;
276 }
277
278 // Helper functions for constructing packets sent by the client
279
Ryan Hamilton8d9ee76e2018-05-29 23:52:52280 std::unique_ptr<quic::QuicReceivedPacket> ConstructSettingsPacket(
Fan Yangac867502019-01-28 21:10:23281 uint64_t packet_number) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55282 return client_maker_.MakeInitialSettingsPacket(packet_number,
283 &header_stream_offset_);
284 }
285
Ryan Hamilton8d9ee76e2018-05-29 23:52:52286 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23287 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52288 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23289 uint64_t largest_received,
290 uint64_t smallest_received,
291 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55292 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09293 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55294 largest_received, smallest_received, least_unacked, kSendFeedback);
295 }
296
Frank Kastenholz684ea412019-02-13 18:48:18297 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstOnlyPacket(
298 uint64_t packet_number,
299 quic::QuicRstStreamErrorCode error_code,
300 uint64_t largest_received,
301 uint64_t smallest_received,
302 uint64_t least_unacked,
303 size_t bytes_written) {
304 return client_maker_.MakeAckAndRstPacket(
305 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
306 largest_received, smallest_received, least_unacked, kSendFeedback,
307 bytes_written,
308 /*include_stop_sending=*/false);
309 }
310
Ryan Hamilton8d9ee76e2018-05-29 23:52:52311 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23312 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52313 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23314 uint64_t largest_received,
315 uint64_t smallest_received,
316 uint64_t least_unacked,
Yixin Wang0d2c6b7e12017-08-16 21:12:55317 size_t bytes_written) {
318 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09319 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55320 largest_received, smallest_received, least_unacked, kSendFeedback,
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> ConstructRstPacket(
Fan Yangac867502019-01-28 21:10:23326 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52327 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55328 size_t bytes_written) {
329 return client_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09330 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18331 bytes_written,
332 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55333 }
334
Ryan Hamilton8d9ee76e2018-05-29 23:52:52335 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectRequestPacket(
Fan Yangac867502019-01-28 21:10:23336 uint64_t packet_number,
Lily Chenf11e1292018-11-29 16:42:09337 RequestPriority request_priority = LOWEST) {
Ryan Hamilton0239aac2018-05-19 00:03:13338 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55339 PopulateConnectRequestIR(&block);
340 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09341 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Lily Chenf11e1292018-11-29 16:42:09342 ConvertRequestPriorityToQuicPriority(request_priority),
343 std::move(block), 0, nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55344 }
345
Ryan Hamilton8d9ee76e2018-05-29 23:52:52346 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectAuthRequestPacket(
Fan Yangac867502019-01-28 21:10:23347 uint64_t packet_number) {
Ryan Hamilton0239aac2018-05-19 00:03:13348 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55349 PopulateConnectRequestIR(&block);
350 block["proxy-authorization"] = "Basic Zm9vOmJhcg==";
351 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09352 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Yixin Wang7a3f1b8d2018-01-17 21:40:48353 ConvertRequestPriorityToQuicPriority(LOWEST), std::move(block), 0,
354 nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55355 }
356
Ryan Hamilton8d9ee76e2018-05-29 23:52:52357 std::unique_ptr<quic::QuicReceivedPacket> ConstructDataPacket(
Fan Yangac867502019-01-28 21:10:23358 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52359 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41360 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09361 return client_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41362 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55363 }
364
Renjied172e812019-01-16 05:12:35365 std::unique_ptr<quic::QuicReceivedPacket> ConstructMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23366 uint64_t packet_number,
Renjied172e812019-01-16 05:12:35367 quic::QuicStreamOffset offset,
368 const std::vector<std::string> data_writes) {
369 return client_maker_.MakeMultipleDataFramesPacket(
370 packet_number, client_data_stream_id1_, !kIncludeVersion, !kFin, offset,
371 data_writes);
372 }
373
Ryan Hamilton8d9ee76e2018-05-29 23:52:52374 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndDataPacket(
Fan Yangac867502019-01-28 21:10:23375 uint64_t packet_number,
376 uint64_t largest_received,
377 uint64_t smallest_received,
378 uint64_t least_unacked,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52379 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41380 quic::QuicStringPiece data) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55381 return client_maker_.MakeAckAndDataPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09382 packet_number, !kIncludeVersion, client_data_stream_id1_,
383 largest_received, smallest_received, least_unacked, !kFin, offset,
Renjief49758b2019-01-11 23:32:41384 data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55385 }
386
Renjied172e812019-01-16 05:12:35387 std::unique_ptr<quic::QuicReceivedPacket>
388 ConstructAckAndMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23389 uint64_t packet_number,
390 uint64_t largest_received,
391 uint64_t smallest_received,
392 uint64_t least_unacked,
Renjied172e812019-01-16 05:12:35393 quic::QuicStreamOffset offset,
394 const std::vector<std::string> data_writes) {
395 return client_maker_.MakeAckAndMultipleDataFramesPacket(
396 packet_number, !kIncludeVersion, client_data_stream_id1_,
397 largest_received, smallest_received, least_unacked, !kFin, offset,
398 data_writes);
399 }
400
Ryan Hamilton8d9ee76e2018-05-29 23:52:52401 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckPacket(
Fan Yangac867502019-01-28 21:10:23402 uint64_t packet_number,
403 uint64_t largest_received,
404 uint64_t smallest_received,
405 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55406 return client_maker_.MakeAckPacket(packet_number, largest_received,
407 smallest_received, least_unacked,
408 kSendFeedback);
409 }
410
411 // Helper functions for constructing packets sent by the server
412
Ryan Hamilton8d9ee76e2018-05-29 23:52:52413 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerRstPacket(
Fan Yangac867502019-01-28 21:10:23414 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52415 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55416 size_t bytes_written) {
417 return server_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09418 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18419 bytes_written,
420 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55421 }
422
Ryan Hamilton8d9ee76e2018-05-29 23:52:52423 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataPacket(
Fan Yangac867502019-01-28 21:10:23424 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52425 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41426 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09427 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41428 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55429 }
430
Ryan Hamilton8d9ee76e2018-05-29 23:52:52431 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataFinPacket(
Fan Yangac867502019-01-28 21:10:23432 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52433 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41434 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09435 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41436 !kIncludeVersion, kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55437 }
438
Ryan Hamilton8d9ee76e2018-05-29 23:52:52439 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerConnectReplyPacket(
Fan Yangac867502019-01-28 21:10:23440 uint64_t packet_number,
Yixin Wang0d2c6b7e12017-08-16 21:12:55441 bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13442 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55443 block[":status"] = "200";
444
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 ConstructServerConnectAuthReplyPacket(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"] = "407";
454 block["proxy-authenticate"] = "Basic realm=\"MyRealm1\"";
455 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09456 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55457 std::move(block), nullptr, &response_offset_);
458 }
459
Ryan Hamilton8d9ee76e2018-05-29 23:52:52460 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23461 ConstructServerConnectRedirectReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13462 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55463 block[":status"] = "302";
464 block["location"] = kRedirectUrl;
465 block["set-cookie"] = "foo=bar";
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
Ryan Hamilton8d9ee76e2018-05-29 23:52:52471 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23472 ConstructServerConnectErrorReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13473 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55474 block[":status"] = "500";
475
476 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09477 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55478 std::move(block), nullptr, &response_offset_);
479 }
480
481 void AssertConnectSucceeds() {
482 TestCompletionCallback callback;
483 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
484 ASSERT_THAT(callback.WaitForResult(), IsOk());
485 }
486
487 void AssertConnectFails(int result) {
488 TestCompletionCallback callback;
489 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
490 ASSERT_EQ(result, callback.WaitForResult());
491 }
492
493 void ResumeAndRun() {
494 // Run until the pause, if the provider isn't paused yet.
495 SequencedSocketData* data = mock_quic_data_.GetSequencedSocketData();
496 data->RunUntilPaused();
497 data->Resume();
498 base::RunLoop().RunUntilIdle();
499 }
500
501 void AssertWriteReturns(const char* data, int len, int rv) {
Victor Costan9c7302b2018-08-27 16:39:44502 scoped_refptr<IOBufferWithSize> buf =
503 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55504 memcpy(buf->data(), data, len);
505 EXPECT_EQ(rv,
[email protected]578968d42017-12-13 15:39:32506 sock_->Write(buf.get(), buf->size(), write_callback_.callback(),
507 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55508 }
509
510 void AssertSyncWriteSucceeds(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44511 scoped_refptr<IOBufferWithSize> buf =
512 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55513 memcpy(buf->data(), data, len);
Bence Békyd8a21fc32018-06-27 18:29:58514 EXPECT_EQ(len,
515 sock_->Write(buf.get(), buf->size(), CompletionOnceCallback(),
516 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55517 }
518
519 void AssertSyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44520 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Bence Békyd8a21fc32018-06-27 18:29:58521 ASSERT_EQ(len, sock_->Read(buf.get(), len, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:13522 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55523 ASSERT_TRUE(sock_->IsConnected());
524 }
525
526 void AssertAsyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44527 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55528 ASSERT_EQ(ERR_IO_PENDING,
529 sock_->Read(buf.get(), len, read_callback_.callback()));
530 EXPECT_TRUE(sock_->IsConnected());
531
532 ResumeAndRun();
533
534 EXPECT_EQ(len, read_callback_.WaitForResult());
535 EXPECT_TRUE(sock_->IsConnected());
Ryan Hamilton0239aac2018-05-19 00:03:13536 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55537 }
538
539 void AssertReadStarts(const char* data, int len) {
540 // Issue the read, which will be completed asynchronously.
Victor Costan9c7302b2018-08-27 16:39:44541 read_buf_ = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55542 ASSERT_EQ(ERR_IO_PENDING,
543 sock_->Read(read_buf_.get(), len, read_callback_.callback()));
544 EXPECT_TRUE(sock_->IsConnected());
545 }
546
547 void AssertReadReturns(const char* data, int len) {
548 EXPECT_TRUE(sock_->IsConnected());
549
550 // Now the read will return.
551 EXPECT_EQ(len, read_callback_.WaitForResult());
Ryan Hamilton0239aac2018-05-19 00:03:13552 ASSERT_EQ(spdy::SpdyString(data, len),
553 spdy::SpdyString(read_buf_->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55554 }
555
Victor Vasiliev076657c2019-03-12 02:46:43556 std::string ConstructDataHeader(size_t body_len) {
Nick Harper23290b82019-05-02 00:02:56557 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41558 return "";
559 }
560 quic::HttpEncoder encoder;
561 std::unique_ptr<char[]> buffer;
562 auto header_length = encoder.SerializeDataFrameHeader(body_len, &buffer);
Victor Vasiliev076657c2019-03-12 02:46:43563 return std::string(buffer.get(), header_length);
Renjief49758b2019-01-11 23:32:41564 }
565
Nick Harper23290b82019-05-02 00:02:56566 const quic::ParsedQuicVersion version_;
Ryan Hamilton47cf9d12018-10-17 04:33:09567 const quic::QuicStreamId client_data_stream_id1_;
Yixin Wang079ad542018-01-11 04:06:05568 const bool client_headers_include_h2_stream_dependency_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55569
570 // order of destruction of these members matter
Ryan Hamilton8d9ee76e2018-05-29 23:52:52571 quic::MockClock clock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55572 MockQuicData mock_quic_data_;
573 std::unique_ptr<QuicChromiumConnectionHelper> helper_;
574 std::unique_ptr<QuicChromiumClientSession> session_;
575 std::unique_ptr<QuicChromiumClientSession::Handle> session_handle_;
576 std::unique_ptr<QuicProxyClientSocket> sock_;
577
578 BoundTestNetLog net_log_;
579
Ryan Hamilton8d9ee76e2018-05-29 23:52:52580 quic::test::MockSendAlgorithm* send_algorithm_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55581 scoped_refptr<TestTaskRunner> runner_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55582
583 std::unique_ptr<QuicChromiumAlarmFactory> alarm_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52584 testing::StrictMock<quic::test::MockQuicConnectionVisitor> visitor_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55585 TransportSecurityState transport_security_state_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52586 quic::QuicCryptoClientConfig crypto_config_;
587 quic::QuicClientPushPromiseIndex push_promise_index_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55588
Ryan Hamilton8d9ee76e2018-05-29 23:52:52589 const quic::QuicConnectionId connection_id_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55590 QuicTestPacketMaker client_maker_;
591 QuicTestPacketMaker server_maker_;
592 IPEndPoint peer_addr_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52593 quic::test::MockRandom random_generator_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55594 ProofVerifyDetailsChromium verify_details_;
595 MockCryptoClientStreamFactory crypto_client_stream_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52596 quic::QuicStreamOffset header_stream_offset_;
597 quic::QuicStreamOffset response_offset_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55598
599 std::string user_agent_;
600 HostPortPair proxy_host_port_;
601 HostPortPair endpoint_host_port_;
602 HttpAuthCache http_auth_cache_;
603 std::unique_ptr<MockHostResolverBase> host_resolver_;
604 std::unique_ptr<HttpAuthHandlerRegistryFactory> http_auth_handler_factory_;
605
606 TestCompletionCallback read_callback_;
607 scoped_refptr<IOBuffer> read_buf_;
608
609 TestCompletionCallback write_callback_;
610
611 DISALLOW_COPY_AND_ASSIGN(QuicProxyClientSocketTest);
612};
613
614TEST_P(QuicProxyClientSocketTest, ConnectSendsCorrectRequest) {
Zhongyi Shi32f2fd02018-04-16 18:23:43615 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
616 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
617 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55618 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
619 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52620 SYNCHRONOUS,
621 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55622
623 Initialize();
624
625 ASSERT_FALSE(sock_->IsConnected());
626
627 AssertConnectSucceeds();
628
629 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
630 ASSERT_TRUE(response != nullptr);
631 ASSERT_EQ(200, response->headers->response_code());
632}
633
634TEST_P(QuicProxyClientSocketTest, ConnectWithAuthRequested) {
Zhongyi Shi32f2fd02018-04-16 18:23:43635 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
636 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
637 mock_quic_data_.AddRead(ASYNC,
638 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55639 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
640 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52641 SYNCHRONOUS,
642 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55643
644 Initialize();
645
646 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
647
648 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
649 ASSERT_TRUE(response != nullptr);
650 ASSERT_EQ(407, response->headers->response_code());
651}
652
653TEST_P(QuicProxyClientSocketTest, ConnectWithAuthCredentials) {
Zhongyi Shi32f2fd02018-04-16 18:23:43654 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
655 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectAuthRequestPacket(2));
656 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55657 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
658 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52659 SYNCHRONOUS,
660 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55661
662 Initialize();
663
664 // Add auth to cache
665 const base::string16 kFoo(base::ASCIIToUTF16("foo"));
666 const base::string16 kBar(base::ASCIIToUTF16("bar"));
667 http_auth_cache_.Add(GURL(kProxyUrl), "MyRealm1", HttpAuth::AUTH_SCHEME_BASIC,
668 "Basic realm=MyRealm1", AuthCredentials(kFoo, kBar),
669 "/");
670
671 AssertConnectSucceeds();
672
673 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
674 ASSERT_TRUE(response != nullptr);
675 ASSERT_EQ(200, response->headers->response_code());
676}
677
Eric Roman96c5b292019-04-23 18:04:59678// Tests that a redirect response from a CONNECT fails.
Yixin Wang0d2c6b7e12017-08-16 21:12:55679TEST_P(QuicProxyClientSocketTest, ConnectRedirects) {
Zhongyi Shi32f2fd02018-04-16 18:23:43680 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
681 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
682 mock_quic_data_.AddRead(ASYNC,
683 ConstructServerConnectRedirectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55684 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
685 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52686 SYNCHRONOUS,
687 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55688
689 Initialize();
690
Eric Roman96c5b292019-04-23 18:04:59691 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
Yixin Wang0d2c6b7e12017-08-16 21:12:55692
693 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
694 ASSERT_TRUE(response != nullptr);
695
696 const HttpResponseHeaders* headers = response->headers.get();
697 ASSERT_EQ(302, headers->response_code());
Eric Roman96c5b292019-04-23 18:04:59698 ASSERT_TRUE(headers->HasHeader("set-cookie"));
Yixin Wang0d2c6b7e12017-08-16 21:12:55699
700 std::string location;
701 ASSERT_TRUE(headers->IsRedirect(&location));
702 ASSERT_EQ(location, kRedirectUrl);
703}
704
705TEST_P(QuicProxyClientSocketTest, ConnectFails) {
Zhongyi Shi32f2fd02018-04-16 18:23:43706 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
707 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Yixin Wang0d2c6b7e12017-08-16 21:12:55708 mock_quic_data_.AddRead(ASYNC, 0); // EOF
709
710 Initialize();
711
712 ASSERT_FALSE(sock_->IsConnected());
713
714 AssertConnectFails(ERR_QUIC_PROTOCOL_ERROR);
715
716 ASSERT_FALSE(sock_->IsConnected());
717}
718
719TEST_P(QuicProxyClientSocketTest, WasEverUsedReturnsCorrectValue) {
Zhongyi Shi32f2fd02018-04-16 18:23:43720 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
721 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
722 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55723 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
724 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52725 SYNCHRONOUS,
726 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55727
728 Initialize();
729
730 EXPECT_TRUE(sock_->WasEverUsed()); // Used due to crypto handshake
731 AssertConnectSucceeds();
732 EXPECT_TRUE(sock_->WasEverUsed());
733 sock_->Disconnect();
734 EXPECT_TRUE(sock_->WasEverUsed());
735}
736
737TEST_P(QuicProxyClientSocketTest, GetPeerAddressReturnsCorrectValues) {
Zhongyi Shi32f2fd02018-04-16 18:23:43738 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
739 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
740 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55741 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
742 mock_quic_data_.AddRead(ASYNC, 0); // EOF
743
744 Initialize();
745
746 IPEndPoint addr;
747 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
748
749 AssertConnectSucceeds();
750 EXPECT_TRUE(sock_->IsConnected());
751 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsOk());
752
753 ResumeAndRun();
754
755 EXPECT_FALSE(sock_->IsConnected());
756 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
757
758 sock_->Disconnect();
759
760 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
761}
762
763TEST_P(QuicProxyClientSocketTest, IsConnectedAndIdle) {
Zhongyi Shi32f2fd02018-04-16 18:23:43764 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
765 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
766 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55767 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
768
Victor Vasiliev076657c2019-03-12 02:46:43769 std::string header = ConstructDataHeader(kLen1);
770 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
771 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43772 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55773 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52774 mock_quic_data_.AddWrite(
775 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55776
777 Initialize();
778
779 EXPECT_FALSE(sock_->IsConnectedAndIdle());
780
781 AssertConnectSucceeds();
782
783 EXPECT_TRUE(sock_->IsConnectedAndIdle());
784
785 // The next read is consumed and buffered.
786 ResumeAndRun();
787
788 EXPECT_FALSE(sock_->IsConnectedAndIdle());
789
790 AssertSyncReadEquals(kMsg1, kLen1);
791
792 EXPECT_TRUE(sock_->IsConnectedAndIdle());
793}
794
795TEST_P(QuicProxyClientSocketTest, GetTotalReceivedBytes) {
Zhongyi Shi32f2fd02018-04-16 18:23:43796 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
797 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
798 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55799 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
800
Victor Vasiliev076657c2019-03-12 02:46:43801 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:41802 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:43803 ASYNC,
804 ConstructServerDataPacket(2, 0, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:43805 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55806 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52807 mock_quic_data_.AddWrite(
808 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55809
810 Initialize();
811
812 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
813
814 AssertConnectSucceeds();
815
816 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
817
818 // The next read is consumed and buffered.
819 ResumeAndRun();
820
821 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
822
823 // The payload from the single large data frame will be read across
824 // two different reads.
825 AssertSyncReadEquals(kMsg33, kLen33);
826
Renjief49758b2019-01-11 23:32:41827 EXPECT_EQ((int64_t)(kLen33 + header.length()),
828 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55829
830 AssertSyncReadEquals(kMsg3, kLen3);
831
Renjief49758b2019-01-11 23:32:41832 EXPECT_EQ((int64_t)(kLen333 + header.length()),
833 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55834}
835
Lily Chenf11e1292018-11-29 16:42:09836TEST_P(QuicProxyClientSocketTest, SetStreamPriority) {
837 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
Matt Menkeedaf3b82019-03-14 21:39:44838 // Despite setting the priority to HIGHEST, the requets initial priority of
839 // LOWEST is used.
Lily Chenf11e1292018-11-29 16:42:09840 mock_quic_data_.AddWrite(SYNCHRONOUS,
Matt Menkeedaf3b82019-03-14 21:39:44841 ConstructConnectRequestPacket(2, LOWEST));
Lily Chenf11e1292018-11-29 16:42:09842 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
843 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
844 mock_quic_data_.AddWrite(
845 SYNCHRONOUS,
846 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
847
848 Initialize();
849
850 sock_->SetStreamPriority(HIGHEST);
851 AssertConnectSucceeds();
852}
Yixin Wang0d2c6b7e12017-08-16 21:12:55853
854TEST_P(QuicProxyClientSocketTest, WriteSendsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43855 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
856 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
857 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55858 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:56859 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:43860 std::string header = ConstructDataHeader(kLen1);
861 mock_quic_data_.AddWrite(
862 SYNCHRONOUS, ConstructAckAndMultipleDataFramesPacket(
863 3, 1, 1, 1, 0, {header, std::string(kMsg1, kLen1)}));
864 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:41865 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:35866 SYNCHRONOUS,
Victor Vasiliev076657c2019-03-12 02:46:43867 ConstructMultipleDataFramesPacket(
868 4, kLen1 + header.length(), {header2, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:41869 mock_quic_data_.AddWrite(
870 SYNCHRONOUS,
Renjied172e812019-01-16 05:12:35871 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED,
Renjief49758b2019-01-11 23:32:41872 kLen1 + kLen2 + header.length() + header2.length()));
873 } else {
874 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41875 SYNCHRONOUS,
Victor Vasiliev076657c2019-03-12 02:46:43876 ConstructAckAndDataPacket(3, 1, 1, 1, 0, std::string(kMsg1, kLen1)));
877 mock_quic_data_.AddWrite(
878 SYNCHRONOUS, ConstructDataPacket(4, kLen1, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:41879 mock_quic_data_.AddWrite(
880 SYNCHRONOUS,
881 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED, kLen1 + kLen2));
882 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55883
884 Initialize();
885
886 AssertConnectSucceeds();
887
888 AssertSyncWriteSucceeds(kMsg1, kLen1);
889 AssertSyncWriteSucceeds(kMsg2, kLen2);
890}
891
892TEST_P(QuicProxyClientSocketTest, WriteSplitsLargeDataIntoMultiplePackets) {
Renjief49758b2019-01-11 23:32:41893 int write_packet_index = 1;
894 mock_quic_data_.AddWrite(SYNCHRONOUS,
895 ConstructSettingsPacket(write_packet_index++));
896 mock_quic_data_.AddWrite(SYNCHRONOUS,
897 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:43898 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55899 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:43900 std::string header = ConstructDataHeader(kLen1);
Nick Harper23290b82019-05-02 00:02:56901 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41902 mock_quic_data_.AddWrite(
903 SYNCHRONOUS, ConstructAckAndDataPacket(write_packet_index++, 1, 1, 1, 0,
Victor Vasiliev076657c2019-03-12 02:46:43904 std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:41905 } else {
Renjied172e812019-01-16 05:12:35906 mock_quic_data_.AddWrite(SYNCHRONOUS,
907 ConstructAckAndMultipleDataFramesPacket(
908 write_packet_index++, 1, 1, 1, 0,
Victor Vasiliev076657c2019-03-12 02:46:43909 {header, std::string(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41910 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55911
912 // Expect |kNumDataPackets| data packets, each containing the max possible
913 // amount of data.
Renjied172e812019-01-16 05:12:35914 int numDataPackets = 3;
915 std::string data(numDataPackets * quic::kDefaultMaxPacketSize, 'x');
Renjief49758b2019-01-11 23:32:41916 quic::QuicStreamOffset offset = kLen1 + header.length();
Renjied172e812019-01-16 05:12:35917
Nick Harper23290b82019-05-02 00:02:56918 if (version_.transport_version == quic::QUIC_VERSION_99) {
Renjied172e812019-01-16 05:12:35919 numDataPackets++;
Renjief49758b2019-01-11 23:32:41920 }
Renjied172e812019-01-16 05:12:35921 size_t total_data_length = 0;
922 for (int i = 0; i < numDataPackets; ++i) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55923 size_t max_packet_data_length = GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52924 quic::kDefaultMaxPacketSize, version_, !kIncludeVersion,
925 !kIncludeDiversificationNonce, quic::PACKET_8BYTE_CONNECTION_ID,
926 quic::PACKET_1BYTE_PACKET_NUMBER, offset);
Nick Harper23290b82019-05-02 00:02:56927 if (version_.transport_version == quic::QUIC_VERSION_99 && i == 0) {
Renjied172e812019-01-16 05:12:35928 // 3973 is the data frame length from packet length.
Victor Vasiliev076657c2019-03-12 02:46:43929 std::string header2 = ConstructDataHeader(3973);
Renjied172e812019-01-16 05:12:35930 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43931 SYNCHRONOUS, ConstructMultipleDataFramesPacket(
932 write_packet_index++, offset,
933 {header2, std::string(data.c_str(),
934 max_packet_data_length - 7)}));
Renjied172e812019-01-16 05:12:35935 offset += max_packet_data_length - header2.length() - 1;
Nick Harper23290b82019-05-02 00:02:56936 } else if (version_.transport_version == quic::QUIC_VERSION_99 &&
937 i == numDataPackets - 1) {
Renjied172e812019-01-16 05:12:35938 mock_quic_data_.AddWrite(
939 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, offset,
Victor Vasiliev076657c2019-03-12 02:46:43940 std::string(data.c_str(), 7)));
Renjied172e812019-01-16 05:12:35941 offset += 7;
942 } else {
943 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43944 SYNCHRONOUS, ConstructDataPacket(
945 write_packet_index++, offset,
946 std::string(data.c_str(), max_packet_data_length)));
Renjied172e812019-01-16 05:12:35947 offset += max_packet_data_length;
948 }
949 if (i != 3) {
950 total_data_length += max_packet_data_length;
951 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55952 }
Renjied172e812019-01-16 05:12:35953
Yixin Wang0d2c6b7e12017-08-16 21:12:55954 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41955 SYNCHRONOUS, ConstructRstPacket(write_packet_index++,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52956 quic::QUIC_STREAM_CANCELLED, offset));
Yixin Wang0d2c6b7e12017-08-16 21:12:55957
958 Initialize();
959
960 AssertConnectSucceeds();
961
962 // Make a small write. An ACK and STOP_WAITING will be bundled. This prevents
963 // ACK and STOP_WAITING from being bundled with the subsequent large write.
964 // This allows the test code for computing the size of data sent in each
965 // packet to not become too complicated.
966 AssertSyncWriteSucceeds(kMsg1, kLen1);
967
968 // Make large write that should be split up
969 AssertSyncWriteSucceeds(data.c_str(), total_data_length);
970}
971
972// ----------- Read
973
974TEST_P(QuicProxyClientSocketTest, ReadReadsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43975 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
976 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
977 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55978 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
979
Victor Vasiliev076657c2019-03-12 02:46:43980 std::string header = ConstructDataHeader(kLen1);
981 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
982 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43983 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55984 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52985 mock_quic_data_.AddWrite(
986 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55987
988 Initialize();
989
990 AssertConnectSucceeds();
991
992 ResumeAndRun();
993 AssertSyncReadEquals(kMsg1, kLen1);
994}
995
996TEST_P(QuicProxyClientSocketTest, ReadDataFromBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:43997 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
998 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
999 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551000 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1001
Victor Vasiliev076657c2019-03-12 02:46:431002 std::string header = ConstructDataHeader(kLen1);
1003 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1004 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431005 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551006 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1007
Victor Vasiliev076657c2019-03-12 02:46:431008 std::string header2 = ConstructDataHeader(kLen2);
1009 mock_quic_data_.AddRead(
1010 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1011 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551012 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1013
1014 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521015 SYNCHRONOUS,
1016 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551017
1018 Initialize();
1019
1020 AssertConnectSucceeds();
1021
1022 ResumeAndRun();
1023 AssertSyncReadEquals(kMsg1, kLen1);
1024
1025 ResumeAndRun();
1026 AssertSyncReadEquals(kMsg2, kLen2);
1027}
1028
1029TEST_P(QuicProxyClientSocketTest, ReadDataMultipleBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431030 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1031 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1032 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551033 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1034
Victor Vasiliev076657c2019-03-12 02:46:431035 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411036 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
Victor Vasiliev076657c2019-03-12 02:46:431037 2, 0, header + std::string(kMsg1, kLen1)));
1038 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1039 std::string header2 = ConstructDataHeader(kLen2);
1040 mock_quic_data_.AddRead(
1041 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1042 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551043 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1044
1045 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521046 SYNCHRONOUS,
1047 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551048
1049 Initialize();
1050
1051 AssertConnectSucceeds();
1052
1053 // The next two reads are consumed and buffered.
1054 ResumeAndRun();
1055
1056 AssertSyncReadEquals(kMsg1, kLen1);
1057 AssertSyncReadEquals(kMsg2, kLen2);
1058}
1059
1060TEST_P(QuicProxyClientSocketTest, LargeReadWillMergeDataFromDifferentFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431061 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1062 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1063 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551064 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1065
Victor Vasiliev076657c2019-03-12 02:46:431066 std::string header = ConstructDataHeader(kLen3);
Renjief49758b2019-01-11 23:32:411067 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
Victor Vasiliev076657c2019-03-12 02:46:431068 2, 0, header + std::string(kMsg3, kLen3)));
1069 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1070 std::string header2 = ConstructDataHeader(kLen3);
1071 mock_quic_data_.AddRead(
1072 ASYNC, ConstructServerDataPacket(3, kLen3 + header.length(),
1073 header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551074 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1075
1076 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521077 SYNCHRONOUS,
1078 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551079
1080 Initialize();
1081
1082 AssertConnectSucceeds();
1083
1084 // The next two reads are consumed and buffered.
1085 ResumeAndRun();
1086 // The payload from two data frames, each with kMsg3 will be combined
1087 // together into a single read().
1088 AssertSyncReadEquals(kMsg33, kLen33);
1089}
1090
1091TEST_P(QuicProxyClientSocketTest, MultipleShortReadsThenMoreRead) {
Zhongyi Shi32f2fd02018-04-16 18:23:431092 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1093 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1094 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551095 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1096
1097 int offset = 0;
1098
Victor Vasiliev076657c2019-03-12 02:46:431099 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411100 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431101 ASYNC,
1102 ConstructServerDataPacket(2, offset, header + std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411103 offset += kLen1 + header.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431104 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551105
Victor Vasiliev076657c2019-03-12 02:46:431106 std::string header2 = ConstructDataHeader(kLen3);
Renjief49758b2019-01-11 23:32:411107 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431108 ASYNC, ConstructServerDataPacket(3, offset,
1109 header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411110 offset += kLen3 + header2.length();
1111 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431112 ASYNC, ConstructServerDataPacket(4, offset,
1113 header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411114 offset += kLen3 + header2.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431115 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(4, 4, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551116
Victor Vasiliev076657c2019-03-12 02:46:431117 std::string header3 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411118 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431119 ASYNC, ConstructServerDataPacket(5, offset,
1120 header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411121 offset += kLen2 + header3.length();
Yixin Wang0d2c6b7e12017-08-16 21:12:551122 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1123
1124 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521125 SYNCHRONOUS,
1126 ConstructAckAndRstPacket(5, quic::QUIC_STREAM_CANCELLED, 5, 5, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551127
1128 Initialize();
1129
1130 AssertConnectSucceeds();
1131
1132 // The next 4 reads are consumed and buffered.
1133 ResumeAndRun();
1134
1135 AssertSyncReadEquals(kMsg1, kLen1);
1136 // The payload from two data frames, each with kMsg3 will be combined
1137 // together into a single read().
1138 AssertSyncReadEquals(kMsg33, kLen33);
1139 AssertSyncReadEquals(kMsg2, kLen2);
1140}
1141
1142TEST_P(QuicProxyClientSocketTest, ReadWillSplitDataFromLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431143 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1144 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1145 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551146 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1147
Victor Vasiliev076657c2019-03-12 02:46:431148 std::string header = ConstructDataHeader(kLen1);
1149 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1150 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431151 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431152 std::string header2 = ConstructDataHeader(kLen33);
Renjief49758b2019-01-11 23:32:411153 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431154 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1155 header2 + std::string(kMsg33, kLen33)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551156 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1157
1158 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521159 SYNCHRONOUS,
1160 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551161
1162 Initialize();
1163
1164 AssertConnectSucceeds();
1165
1166 // The next 2 reads are consumed and buffered.
1167 ResumeAndRun();
1168
1169 AssertSyncReadEquals(kMsg1, kLen1);
1170 // The payload from the single large data frame will be read across
1171 // two different reads.
1172 AssertSyncReadEquals(kMsg3, kLen3);
1173 AssertSyncReadEquals(kMsg3, kLen3);
1174}
1175
1176TEST_P(QuicProxyClientSocketTest, MultipleReadsFromSameLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431177 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1178 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1179 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551180 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1181
Victor Vasiliev076657c2019-03-12 02:46:431182 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:411183 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431184 ASYNC,
1185 ConstructServerDataPacket(2, 0, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:431186 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551187 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1188
Ryan Hamilton8d9ee76e2018-05-29 23:52:521189 mock_quic_data_.AddWrite(
1190 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551191
1192 Initialize();
1193
1194 AssertConnectSucceeds();
1195
1196 // The next read is consumed and buffered.
1197 ResumeAndRun();
1198
1199 // The payload from the single large data frame will be read across
1200 // two different reads.
1201 AssertSyncReadEquals(kMsg33, kLen33);
1202
1203 // Now attempt to do a read of more data than remains buffered
Victor Costan9c7302b2018-08-27 16:39:441204 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kLen33);
Bence Békyd8a21fc32018-06-27 18:29:581205 ASSERT_EQ(kLen3, sock_->Read(buf.get(), kLen33, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:131206 ASSERT_EQ(spdy::SpdyString(kMsg3, kLen3),
1207 spdy::SpdyString(buf->data(), kLen3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551208 ASSERT_TRUE(sock_->IsConnected());
1209}
1210
1211TEST_P(QuicProxyClientSocketTest, ReadAuthResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431212 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1213 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1214 mock_quic_data_.AddRead(ASYNC,
1215 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551216 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1217
Victor Vasiliev076657c2019-03-12 02:46:431218 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411219 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
Victor Vasiliev076657c2019-03-12 02:46:431220 2, 0, header + std::string(kMsg1, kLen1)));
1221 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1222 std::string header2 = ConstructDataHeader(kLen2);
1223 mock_quic_data_.AddRead(
1224 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1225 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551226 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1227
1228 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521229 SYNCHRONOUS,
1230 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551231
1232 Initialize();
1233
1234 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
1235
1236 // The next two reads are consumed and buffered.
1237 ResumeAndRun();
1238
1239 AssertSyncReadEquals(kMsg1, kLen1);
1240 AssertSyncReadEquals(kMsg2, kLen2);
1241}
1242
1243TEST_P(QuicProxyClientSocketTest, ReadErrorResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431244 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1245 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1246 mock_quic_data_.AddRead(ASYNC,
1247 ConstructServerConnectErrorReplyPacket(1, !kFin));
Victor Vasiliev076657c2019-03-12 02:46:431248 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411249 mock_quic_data_.AddRead(
1250 SYNCHRONOUS,
Victor Vasiliev076657c2019-03-12 02:46:431251 ConstructServerDataPacket(2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431252 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431253 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411254 mock_quic_data_.AddRead(
1255 SYNCHRONOUS,
1256 ConstructServerDataPacket(3, kLen1 + header.length(),
Victor Vasiliev076657c2019-03-12 02:46:431257 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551258 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1259
1260 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521261 SYNCHRONOUS,
1262 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551263 Initialize();
1264
1265 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
1266}
1267
1268// ----------- Reads and Writes
1269
1270TEST_P(QuicProxyClientSocketTest, AsyncReadAroundWrite) {
Renjief49758b2019-01-11 23:32:411271 int write_packet_index = 1;
1272 mock_quic_data_.AddWrite(SYNCHRONOUS,
1273 ConstructSettingsPacket(write_packet_index++));
1274 mock_quic_data_.AddWrite(SYNCHRONOUS,
1275 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431276 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551277 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1278
Victor Vasiliev076657c2019-03-12 02:46:431279 std::string header = ConstructDataHeader(kLen1);
1280 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1281 2, 0, header + std::string(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
Victor Vasiliev076657c2019-03-12 02:46:431285 std::string header2 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561286 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:431287 mock_quic_data_.AddWrite(
1288 SYNCHRONOUS,
1289 ConstructMultipleDataFramesPacket(
1290 write_packet_index++, 0, {header2, std::string(kMsg2, kLen2)}));
Renjied172e812019-01-16 05:12:351291 } else {
Renjief49758b2019-01-11 23:32:411292 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431293 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, header2.length(),
1294 std::string(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
Victor Vasiliev076657c2019-03-12 02:46:431299 std::string header3 = ConstructDataHeader(kLen3);
1300 mock_quic_data_.AddRead(
1301 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1302 header3 + std::string(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
Victor Vasiliev076657c2019-03-12 02:46:431336 std::string header = ConstructDataHeader(kLen1);
1337 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1338 2, 0, header + std::string(kMsg1, kLen1)));
Renjied172e812019-01-16 05:12:351339 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551340 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1341
Victor Vasiliev076657c2019-03-12 02:46:431342 std::string header2 = ConstructDataHeader(kLen3);
1343 mock_quic_data_.AddRead(
1344 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1345 header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551346 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1347
1348 mock_quic_data_.AddWrite(ASYNC, ERR_IO_PENDING); // Pause
1349
Victor Vasiliev076657c2019-03-12 02:46:431350 std::string header3 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561351 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:411352 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431353 ASYNC, ConstructDataPacket(4, 0, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411354 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351355 SYNCHRONOUS, ConstructAckAndDataPacket(5, 3, 3, 1, kLen2,
Victor Vasiliev076657c2019-03-12 02:46:431356 std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411357 } else {
Victor Vasiliev076657c2019-03-12 02:46:431358 mock_quic_data_.AddWrite(ASYNC,
1359 ConstructMultipleDataFramesPacket(
1360 4, 0, {header3, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:411361 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431362 ASYNC, ConstructAckAndDataPacket(5, 3, 3, 1, header3.length() + kLen2,
1363 header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411364 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551365
Renjied172e812019-01-16 05:12:351366 mock_quic_data_.AddWrite(
1367 SYNCHRONOUS, ConstructRstPacket(6, quic::QUIC_STREAM_CANCELLED,
1368 kLen2 + kLen2 + 2 * header3.length()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551369
1370 Initialize();
1371
1372 AssertConnectSucceeds();
1373
1374 ResumeAndRun();
1375 AssertSyncReadEquals(kMsg1, kLen1);
1376
1377 // Write should block until the next read completes.
1378 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1379 // asynchronous starting with the second time it's called while the UDP socket
1380 // is write-blocked. Therefore, at least two writes need to be called on
1381 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351382 AssertWriteReturns(kMsg2, kLen2, kLen2);
Yixin Wang0d2c6b7e12017-08-16 21:12:551383 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1384
1385 AssertAsyncReadEquals(kMsg3, kLen3);
1386
1387 ASSERT_FALSE(write_callback_.have_result());
1388
1389 // Now the write will complete
1390 ResumeAndRun();
Yixin Wangdbbd8752018-01-17 21:50:021391 EXPECT_EQ(kLen2, write_callback_.WaitForResult());
Yixin Wang0d2c6b7e12017-08-16 21:12:551392}
1393
1394// ----------- Reading/Writing on Closed socket
1395
1396// Reading from an already closed socket should return 0
1397TEST_P(QuicProxyClientSocketTest, ReadOnClosedSocketReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431398 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1399 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1400 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551401 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1402
1403 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1404
1405 Initialize();
1406
1407 AssertConnectSucceeds();
1408
1409 ResumeAndRun();
1410
1411 ASSERT_FALSE(sock_->IsConnected());
Raul Tambre94493c652019-03-11 17:18:351412 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1413 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1414 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551415 ASSERT_FALSE(sock_->IsConnectedAndIdle());
1416}
1417
1418// Read pending when socket is closed should return 0
1419TEST_P(QuicProxyClientSocketTest, PendingReadOnCloseReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431420 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1421 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1422 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551423 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1424
1425 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1426
1427 Initialize();
1428
1429 AssertConnectSucceeds();
1430
1431 AssertReadStarts(kMsg1, kLen1);
1432
1433 ResumeAndRun();
1434
1435 ASSERT_EQ(0, read_callback_.WaitForResult());
1436}
1437
1438// Reading from a disconnected socket is an error
1439TEST_P(QuicProxyClientSocketTest, ReadOnDisconnectSocketReturnsNotConnected) {
Zhongyi Shi32f2fd02018-04-16 18:23:431440 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1441 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1442 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551443 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1444 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521445 SYNCHRONOUS,
1446 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551447
1448 Initialize();
1449
1450 AssertConnectSucceeds();
1451
1452 sock_->Disconnect();
1453
1454 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581455 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551456}
1457
1458// Reading data after receiving FIN should return buffered data received before
1459// FIN, then 0.
1460TEST_P(QuicProxyClientSocketTest, ReadAfterFinReceivedReturnsBufferedData) {
Zhongyi Shi32f2fd02018-04-16 18:23:431461 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1462 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1463 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551464 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1465
Victor Vasiliev076657c2019-03-12 02:46:431466 std::string header = ConstructDataHeader(kLen1);
1467 mock_quic_data_.AddRead(ASYNC, ConstructServerDataFinPacket(
1468 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431469 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551470 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521471 mock_quic_data_.AddWrite(
1472 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551473
1474 Initialize();
1475
1476 AssertConnectSucceeds();
1477
1478 ResumeAndRun();
1479
1480 AssertSyncReadEquals(kMsg1, kLen1);
Raul Tambre94493c652019-03-11 17:18:351481 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1482 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551483
1484 sock_->Disconnect();
1485 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581486 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551487}
1488
1489// Calling Write() on a closed socket is an error.
1490TEST_P(QuicProxyClientSocketTest, WriteOnClosedStream) {
Zhongyi Shi32f2fd02018-04-16 18:23:431491 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1492 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1493 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551494 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1495
1496 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1497
1498 Initialize();
1499
1500 AssertConnectSucceeds();
1501
1502 ResumeAndRun();
1503
1504 AssertWriteReturns(kMsg1, kLen1, ERR_QUIC_PROTOCOL_ERROR);
1505}
1506
1507// Calling Write() on a disconnected socket is an error.
1508TEST_P(QuicProxyClientSocketTest, WriteOnDisconnectedSocket) {
Zhongyi Shi32f2fd02018-04-16 18:23:431509 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1510 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1511 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551512 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1513 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521514 SYNCHRONOUS,
1515 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551516
1517 Initialize();
1518
1519 AssertConnectSucceeds();
1520
1521 sock_->Disconnect();
1522
1523 AssertWriteReturns(kMsg1, kLen1, ERR_SOCKET_NOT_CONNECTED);
1524}
1525
1526// If the socket is closed with a pending Write(), the callback should be called
1527// with the same error the session was closed with.
1528TEST_P(QuicProxyClientSocketTest, WritePendingOnClose) {
Zhongyi Shi32f2fd02018-04-16 18:23:431529 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1530 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1531 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551532 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1533 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1534
1535 Initialize();
1536
1537 AssertConnectSucceeds();
1538
1539 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1540 // asynchronous starting with the second time it's called while the UDP socket
1541 // is write-blocked. Therefore, at least two writes need to be called on
1542 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351543 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551544
1545 // This second write will be async. This is the pending write that's being
1546 // tested.
1547 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1548
1549 // Make sure the write actually starts.
1550 base::RunLoop().RunUntilIdle();
1551
Ryan Hamilton8d9ee76e2018-05-29 23:52:521552 session_->CloseSessionOnError(ERR_CONNECTION_CLOSED,
Renjieba55fae2018-09-20 03:05:161553 quic::QUIC_INTERNAL_ERROR,
1554 quic::ConnectionCloseBehavior::SILENT_CLOSE);
Yixin Wang0d2c6b7e12017-08-16 21:12:551555
1556 EXPECT_THAT(write_callback_.WaitForResult(), IsError(ERR_CONNECTION_CLOSED));
1557}
1558
1559TEST_P(QuicProxyClientSocketTest, DisconnectWithWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431560 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1561 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1562 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551563 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1564 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1565
1566 Initialize();
1567
1568 AssertConnectSucceeds();
1569
1570 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1571 // asynchronous starting with the second time it's called while the UDP socket
1572 // is write-blocked. Therefore, at least two writes need to be called on
1573 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351574 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551575
1576 // This second write will be async. This is the pending write that's being
1577 // tested.
1578 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1579
1580 // Make sure the write actually starts.
1581 base::RunLoop().RunUntilIdle();
1582
1583 sock_->Disconnect();
1584 EXPECT_FALSE(sock_->IsConnected());
1585
1586 base::RunLoop().RunUntilIdle();
1587
1588 EXPECT_FALSE(sock_->IsConnected());
1589 EXPECT_FALSE(write_callback_.have_result());
1590}
1591
1592// If the socket is Disconnected with a pending Read(), the callback
1593// should not be called.
1594TEST_P(QuicProxyClientSocketTest, DisconnectWithReadPending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431595 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1596 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1597 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551598 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1599 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521600 SYNCHRONOUS,
1601 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551602
1603 Initialize();
1604
1605 AssertConnectSucceeds();
1606
1607 EXPECT_TRUE(sock_->IsConnected());
1608
1609 AssertReadStarts(kMsg1, kLen1);
1610
1611 sock_->Disconnect();
1612 EXPECT_FALSE(sock_->IsConnected());
1613
1614 base::RunLoop().RunUntilIdle();
1615
1616 EXPECT_FALSE(sock_->IsConnected());
1617 EXPECT_FALSE(read_callback_.have_result());
1618}
1619
1620// If the socket is Reset when both a read and write are pending,
1621// both should be called back.
1622TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431623 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1624 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1625 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551626 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1627
1628 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521629 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang2bea3cf2017-11-09 18:11:031630 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:431631 std::string header = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561632 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:411633 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431634 ASYNC,
1635 ConstructAckAndDataPacket(3, 1, 1, 1, 0, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411636 mock_quic_data_.AddWrite(
1637 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1638 2, 2, 1, kLen2));
1639 } else {
Renjief49758b2019-01-11 23:32:411640 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351641 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Victor Vasiliev076657c2019-03-12 02:46:431642 3, 1, 1, 1, 0, {header, std::string(kMsg2, kLen2)}));
Frank Kastenholz684ea412019-02-13 18:48:181643 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
1644 4, quic::QUIC_STREAM_CANCELLED, 2,
1645 2, 1, header.length() + kLen2));
Renjief49758b2019-01-11 23:32:411646 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551647
1648 Initialize();
1649
1650 AssertConnectSucceeds();
1651
1652 EXPECT_TRUE(sock_->IsConnected());
1653
1654 AssertReadStarts(kMsg1, kLen1);
1655
1656 // Write should block until the next read completes.
1657 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1658 // asynchronous starting with the second time it's called while the UDP socket
1659 // is write-blocked. Therefore, at least two writes need to be called on
1660 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351661 AssertWriteReturns(kMsg2, kLen2, kLen2);
Renjief49758b2019-01-11 23:32:411662
Yixin Wang0d2c6b7e12017-08-16 21:12:551663 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1664
1665 ResumeAndRun();
1666
1667 EXPECT_TRUE(read_callback_.have_result());
1668 EXPECT_TRUE(write_callback_.have_result());
1669}
1670
1671// Makes sure the proxy client socket's source gets the expected NetLog events
1672// and only the expected NetLog events (No SpdySession events).
1673TEST_P(QuicProxyClientSocketTest, NetLog) {
Zhongyi Shi32f2fd02018-04-16 18:23:431674 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1675 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1676 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551677 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1678
Victor Vasiliev076657c2019-03-12 02:46:431679 std::string header = ConstructDataHeader(kLen1);
1680 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1681 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431682 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang2bea3cf2017-11-09 18:11:031683 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521684 mock_quic_data_.AddWrite(
1685 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551686
1687 Initialize();
1688
1689 AssertConnectSucceeds();
1690
1691 ResumeAndRun();
1692 AssertSyncReadEquals(kMsg1, kLen1);
1693
1694 NetLogSource sock_source = sock_->NetLog().source();
1695 sock_.reset();
1696
1697 TestNetLogEntry::List entry_list;
1698 net_log_.GetEntriesForSource(sock_source, &entry_list);
1699
1700 ASSERT_EQ(entry_list.size(), 10u);
1701 EXPECT_TRUE(
1702 LogContainsBeginEvent(entry_list, 0, NetLogEventType::SOCKET_ALIVE));
1703 EXPECT_TRUE(LogContainsEvent(entry_list, 1,
1704 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
1705 NetLogEventPhase::NONE));
1706 EXPECT_TRUE(LogContainsBeginEvent(
1707 entry_list, 2, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1708 EXPECT_TRUE(LogContainsEvent(
1709 entry_list, 3, NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
1710 NetLogEventPhase::NONE));
1711 EXPECT_TRUE(LogContainsEndEvent(
1712 entry_list, 4, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1713 EXPECT_TRUE(LogContainsBeginEvent(
1714 entry_list, 5, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1715 EXPECT_TRUE(LogContainsEvent(
1716 entry_list, 6,
1717 NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
1718 NetLogEventPhase::NONE));
1719 EXPECT_TRUE(LogContainsEndEvent(
1720 entry_list, 7, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1721 EXPECT_TRUE(LogContainsEvent(entry_list, 8,
1722 NetLogEventType::SOCKET_BYTES_RECEIVED,
1723 NetLogEventPhase::NONE));
1724 EXPECT_TRUE(
1725 LogContainsEndEvent(entry_list, 9, NetLogEventType::SOCKET_ALIVE));
1726}
1727
Bence Béky8ddc2492018-06-13 01:02:041728// A helper class that will delete |sock| when the callback is invoked.
Yixin Wang0d2c6b7e12017-08-16 21:12:551729class DeleteSockCallback : public TestCompletionCallbackBase {
1730 public:
1731 explicit DeleteSockCallback(std::unique_ptr<QuicProxyClientSocket>* sock)
Bence Béky8ddc2492018-06-13 01:02:041732 : sock_(sock) {}
Yixin Wang0d2c6b7e12017-08-16 21:12:551733
1734 ~DeleteSockCallback() override {}
1735
Bence Béky8ddc2492018-06-13 01:02:041736 CompletionOnceCallback callback() {
1737 return base::BindOnce(&DeleteSockCallback::OnComplete,
1738 base::Unretained(this));
1739 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551740
1741 private:
1742 void OnComplete(int result) {
Raul Tambre94493c652019-03-11 17:18:351743 sock_->reset(nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:551744 SetResult(result);
1745 }
1746
1747 std::unique_ptr<QuicProxyClientSocket>* sock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:551748
1749 DISALLOW_COPY_AND_ASSIGN(DeleteSockCallback);
1750};
1751
1752// If the socket is reset when both a read and write are pending, and the
1753// read callback causes the socket to be deleted, the write callback should
1754// not be called.
1755TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePendingDelete) {
Zhongyi Shi32f2fd02018-04-16 18:23:431756 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1757 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1758 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551759 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1760
1761 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521762 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551763 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:561764 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:411765 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431766 ASYNC,
1767 ConstructAckAndDataPacket(3, 1, 1, 1, 0, std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411768 mock_quic_data_.AddWrite(
1769 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1770 2, 2, 1, kLen1));
1771 } else {
Victor Vasiliev076657c2019-03-12 02:46:431772 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411773 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351774 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Victor Vasiliev076657c2019-03-12 02:46:431775 3, 1, 1, 1, 0, {header, std::string(kMsg1, kLen1)}));
Frank Kastenholz684ea412019-02-13 18:48:181776 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
1777 4, quic::QUIC_STREAM_CANCELLED, 2,
1778 2, 1, header.length() + kLen1));
Renjief49758b2019-01-11 23:32:411779 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551780
1781 Initialize();
1782
1783 AssertConnectSucceeds();
1784
1785 EXPECT_TRUE(sock_->IsConnected());
1786
1787 DeleteSockCallback read_callback(&sock_);
Victor Costan9c7302b2018-08-27 16:39:441788 scoped_refptr<IOBuffer> read_buf = base::MakeRefCounted<IOBuffer>(kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551789 ASSERT_EQ(ERR_IO_PENDING,
1790 sock_->Read(read_buf.get(), kLen1, read_callback.callback()));
1791
1792 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1793 // asynchronous starting with the second time it's called while the UDP socket
1794 // is write-blocked. Therefore, at least two writes need to be called on
1795 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351796 AssertWriteReturns(kMsg1, kLen1, kLen1);
Renjief49758b2019-01-11 23:32:411797
Yixin Wang0d2c6b7e12017-08-16 21:12:551798 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1799
1800 ResumeAndRun();
1801
1802 EXPECT_FALSE(sock_.get());
1803
1804 EXPECT_EQ(0, read_callback.WaitForResult());
1805 EXPECT_FALSE(write_callback_.have_result());
1806}
1807
Victor Costane635086f2019-01-27 05:20:301808INSTANTIATE_TEST_SUITE_P(
Bence Békyce380cb2018-04-26 23:39:551809 VersionIncludeStreamDependencySequence,
Yixin Wang079ad542018-01-11 04:06:051810 QuicProxyClientSocketTest,
Victor Vasiliev5d6cdc22019-05-28 20:37:431811 ::testing::Combine(::testing::ValuesIn(quic::AllVersionsExcept99()),
Nick Harper23290b82019-05-02 00:02:561812 ::testing::Bool()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551813
1814} // namespace test
Dan Zhangf11470172017-09-18 22:02:091815} // namespace net