blob: 75b361d2845b2b0727b60570650da38181a1426a [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"
Victor Vasiliev6bb59d22019-03-08 21:34:5145#include "net/third_party/quiche/src/quic/test_tools/crypto_test_utils.h"
46#include "net/third_party/quiche/src/quic/test_tools/mock_clock.h"
47#include "net/third_party/quiche/src/quic/test_tools/mock_random.h"
48#include "net/third_party/quiche/src/quic/test_tools/quic_connection_peer.h"
49#include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h"
[email protected]578968d42017-12-13 15:39:3250#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5551#include "testing/gmock/include/gmock/gmock.h"
52#include "testing/gtest/include/gtest/gtest.h"
53
54using testing::_;
55using testing::AnyNumber;
56using testing::Return;
57
58namespace {
59
60static const char kOriginHost[] = "www.google.com";
61static const int kOriginPort = 443;
62static const char kProxyUrl[] = "https://myproxy:6121/";
63static const char kProxyHost[] = "myproxy";
64static const int kProxyPort = 6121;
65static const char kUserAgent[] = "Mozilla/1.0";
66static const char kRedirectUrl[] = "https://example.com/";
67
68static const char kMsg1[] = "\0hello!\xff";
69static const int kLen1 = 8;
70static const char kMsg2[] = "\0a2345678\0";
71static const int kLen2 = 10;
72static const char kMsg3[] = "bye!";
73static const int kLen3 = 4;
74static const char kMsg33[] = "bye!bye!";
75static const int kLen33 = kLen3 + kLen3;
76static const char kMsg333[] = "bye!bye!bye!";
77static const int kLen333 = kLen3 + kLen3 + kLen3;
78
79} // anonymous namespace
80
81namespace net {
82namespace test {
83
Michael Warres74ee3ce2017-10-09 15:26:3784class QuicProxyClientSocketTest
Ryan Hamilton8d9ee76e2018-05-29 23:52:5285 : public ::testing::TestWithParam<
Nick Harper23290b82019-05-02 00:02:5686 std::tuple<quic::ParsedQuicVersion, bool>>,
Bence Béky98447b12018-05-08 03:14:0187 public WithScopedTaskEnvironment {
Yixin Wang0d2c6b7e12017-08-16 21:12:5588 protected:
89 static const bool kFin = true;
90 static const bool kIncludeVersion = true;
91 static const bool kIncludeDiversificationNonce = true;
92 static const bool kIncludeCongestionFeedback = true;
93 static const bool kSendFeedback = true;
94
95 static size_t GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:5296 quic::QuicByteCount packet_length,
Nick Harper23290b82019-05-02 00:02:5697 quic::ParsedQuicVersion version,
Yixin Wang0d2c6b7e12017-08-16 21:12:5598 bool include_version,
99 bool include_diversification_nonce,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52100 quic::QuicConnectionIdLength connection_id_length,
101 quic::QuicPacketNumberLength packet_number_length,
102 quic::QuicStreamOffset offset) {
Michael Warres167db3e2019-03-01 21:38:03103 quic::QuicVariableLengthIntegerLength retry_token_length_length =
104 quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
105 quic::QuicVariableLengthIntegerLength length_length =
Nick Harper23290b82019-05-02 00:02:56106 quic::QuicVersionHasLongHeaderLengths(version.transport_version) &&
107 include_version
Michael Warres167db3e2019-03-01 21:38:03108 ? quic::VARIABLE_LENGTH_INTEGER_LENGTH_2
109 : quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
Yixin Wang0d2c6b7e12017-08-16 21:12:55110 size_t min_data_length = 1;
111 size_t min_packet_length =
Ryan Hamilton8d9ee76e2018-05-29 23:52:52112 quic::NullEncrypter(quic::Perspective::IS_CLIENT)
Yixin Wang0d2c6b7e12017-08-16 21:12:55113 .GetCiphertextSize(min_data_length) +
Ryan Hamilton8d9ee76e2018-05-29 23:52:52114 quic::QuicPacketCreator::StreamFramePacketOverhead(
Nick Harper23290b82019-05-02 00:02:56115 version.transport_version, quic::PACKET_8BYTE_CONNECTION_ID,
Michael Warres60637ae2018-06-05 20:03:11116 quic::PACKET_0BYTE_CONNECTION_ID, include_version,
Michael Warres167db3e2019-03-01 21:38:03117 include_diversification_nonce, packet_number_length,
118 retry_token_length_length, length_length, offset);
Yixin Wang0d2c6b7e12017-08-16 21:12:55119
120 DCHECK(packet_length >= min_packet_length);
121 return min_data_length + packet_length - min_packet_length;
122 }
123
124 QuicProxyClientSocketTest()
Yixin Wang079ad542018-01-11 04:06:05125 : version_(std::get<0>(GetParam())),
Nick Harper23290b82019-05-02 00:02:56126 client_data_stream_id1_(
127 quic::QuicUtils::GetHeadersStreamId(version_.transport_version) +
128 quic::QuicUtils::StreamIdDelta(version_.transport_version)),
Yixin Wang079ad542018-01-11 04:06:05129 client_headers_include_h2_stream_dependency_(std::get<1>(GetParam())),
Ryan Hamiltonabad59e2019-06-06 04:02:59130 mock_quic_data_(version_),
Nick Harpera598fc5f2019-06-21 08:46:50131 crypto_config_(
132 quic::test::crypto_test_utils::ProofVerifierForTesting()),
David Schinazic8281052019-01-24 06:14:17133 connection_id_(quic::test::TestConnectionId(2)),
Yixin Wang0d2c6b7e12017-08-16 21:12:55134 client_maker_(version_,
135 connection_id_,
136 &clock_,
137 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52138 quic::Perspective::IS_CLIENT,
Yixin Wang079ad542018-01-11 04:06:05139 client_headers_include_h2_stream_dependency_),
Yixin Wang0d2c6b7e12017-08-16 21:12:55140 server_maker_(version_,
141 connection_id_,
142 &clock_,
143 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52144 quic::Perspective::IS_SERVER,
Yixin Wang079ad542018-01-11 04:06:05145 false),
Yixin Wang0d2c6b7e12017-08-16 21:12:55146 random_generator_(0),
Yixin Wang0d2c6b7e12017-08-16 21:12:55147 user_agent_(kUserAgent),
148 proxy_host_port_(kProxyHost, kProxyPort),
149 endpoint_host_port_(kOriginHost, kOriginPort),
150 host_resolver_(new MockCachingHostResolver()),
Eric Orthbe2efac2019-03-06 01:11:11151 http_auth_handler_factory_(HttpAuthHandlerFactory::CreateDefault()) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55152 IPAddress ip(192, 0, 2, 33);
153 peer_addr_ = IPEndPoint(ip, 443);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52154 clock_.AdvanceTime(quic::QuicTime::Delta::FromMilliseconds(20));
David Schinazi3f7465c2019-07-12 01:57:05155 if (version_.handshake_protocol == quic::PROTOCOL_TLS1_3) {
156 SetQuicFlag(FLAGS_quic_supports_tls_handshake, true);
157 }
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()),
Bence Béky1ceba552019-07-19 17:11:05223 /*require_confirmation=*/false,
224 /*max_allowed_push_id=*/0,
225 /*migrate_session_early_v2=*/false,
Zhongyi Shi757fcce2018-06-27 05:41:27226 /*migrate_session_on_network_change_v2=*/false,
227 /*default_network=*/NetworkChangeNotifier::kInvalidNetworkHandle,
Zhongyi Shie01f2db2019-02-22 19:53:23228 quic::QuicTime::Delta::FromMilliseconds(
Ryan Sleevi2e8255b2019-07-17 21:02:21229 kDefaultRetransmittableOnWireTimeout.InMilliseconds()),
230 /*migrate_idle_session=*/true, kDefaultIdleSessionMigrationPeriod,
231 kMaxTimeOnNonDefaultNetwork,
Zhongyi Shiee760762018-08-01 00:54:29232 kMaxMigrationsToNonDefaultNetworkOnWriteError,
Zhongyi Shi8b1e43f2017-12-13 20:46:30233 kMaxMigrationsToNonDefaultNetworkOnPathDegrading,
Zhongyi Shi5f587cc2017-11-21 23:24:17234 kQuicYieldAfterPacketsRead,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52235 quic::QuicTime::Delta::FromMilliseconds(
236 kQuicYieldAfterDurationMilliseconds),
Zhongyi Shidbce7f412019-02-01 23:16:29237 /*go_away_on_path_degrading*/ false,
Yixin Wang079ad542018-01-11 04:06:05238 client_headers_include_h2_stream_dependency_, /*cert_verify_flags=*/0,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52239 quic::test::DefaultQuicConfig(), &crypto_config_, "CONNECTION_UNKNOWN",
240 dns_start, dns_end, &push_promise_index_, nullptr,
Zhongyi Shic16b4102019-02-12 00:37:40241 base::DefaultTickClock::GetInstance(),
Yixin Wang0d2c6b7e12017-08-16 21:12:55242 base::ThreadTaskRunnerHandle::Get().get(),
243 /*socket_performance_watcher=*/nullptr, net_log_.bound().net_log()));
244
245 writer->set_delegate(session_.get());
246
Ryan Hamilton6c2a2a82017-12-15 02:06:28247 session_handle_ =
248 session_->CreateHandle(HostPortPair("mail.example.org", 80));
Yixin Wang0d2c6b7e12017-08-16 21:12:55249
250 session_->Initialize();
251 TestCompletionCallback callback;
252 EXPECT_THAT(session_->CryptoConnect(callback.callback()), IsOk());
253 EXPECT_TRUE(session_->IsCryptoHandshakeConfirmed());
254
Ramin Halavati683bcaa92018-02-14 08:42:39255 EXPECT_THAT(session_handle_->RequestStream(true, callback.callback(),
256 TRAFFIC_ANNOTATION_FOR_TESTS),
Yixin Wang0d2c6b7e12017-08-16 21:12:55257 IsOk());
258 std::unique_ptr<QuicChromiumClientStream::Handle> stream_handle =
259 session_handle_->ReleaseStream();
260 EXPECT_TRUE(stream_handle->IsOpen());
261
262 sock_.reset(new QuicProxyClientSocket(
263 std::move(stream_handle), std::move(session_handle_), user_agent_,
264 endpoint_host_port_, net_log_.bound(),
Eric Orthbe2efac2019-03-06 01:11:11265 new HttpAuthController(
266 HttpAuth::AUTH_PROXY,
267 GURL("https://" + proxy_host_port_.ToString()), &http_auth_cache_,
268 http_auth_handler_factory_.get(), host_resolver_.get())));
Yixin Wang0d2c6b7e12017-08-16 21:12:55269
270 session_->StartReading();
271 }
272
Ryan Hamilton0239aac2018-05-19 00:03:13273 void PopulateConnectRequestIR(spdy::SpdyHeaderBlock* block) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55274 (*block)[":method"] = "CONNECT";
275 (*block)[":authority"] = endpoint_host_port_.ToString();
276 (*block)["user-agent"] = kUserAgent;
277 }
278
279 // Helper functions for constructing packets sent by the client
280
Ryan Hamilton8d9ee76e2018-05-29 23:52:52281 std::unique_ptr<quic::QuicReceivedPacket> ConstructSettingsPacket(
Fan Yangac867502019-01-28 21:10:23282 uint64_t packet_number) {
Ryan Hamilton0d65a8c2019-06-07 00:46:02283 return client_maker_.MakeInitialSettingsPacket(packet_number);
Yixin Wang0d2c6b7e12017-08-16 21:12:55284 }
285
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41286 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstOnlyPacket(
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,
Frank Kastenholz684ea412019-02-13 18:48:18294 largest_received, smallest_received, least_unacked, kSendFeedback,
Frank Kastenholz684ea412019-02-13 18:48:18295 /*include_stop_sending=*/false);
296 }
297
Ryan Hamilton8d9ee76e2018-05-29 23:52:52298 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23299 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52300 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23301 uint64_t largest_received,
302 uint64_t smallest_received,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41303 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55304 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09305 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55306 largest_received, smallest_received, least_unacked, kSendFeedback,
Frank Kastenholz684ea412019-02-13 18:48:18307 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55308 }
309
Ryan Hamilton8d9ee76e2018-05-29 23:52:52310 std::unique_ptr<quic::QuicReceivedPacket> ConstructRstPacket(
Fan Yangac867502019-01-28 21:10:23311 uint64_t packet_number,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41312 quic::QuicRstStreamErrorCode error_code) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55313 return client_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09314 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18315 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55316 }
317
Ryan Hamilton8d9ee76e2018-05-29 23:52:52318 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectRequestPacket(
Fan Yangac867502019-01-28 21:10:23319 uint64_t packet_number,
Lily Chenf11e1292018-11-29 16:42:09320 RequestPriority request_priority = LOWEST) {
Ryan Hamilton0239aac2018-05-19 00:03:13321 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55322 PopulateConnectRequestIR(&block);
323 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09324 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Lily Chenf11e1292018-11-29 16:42:09325 ConvertRequestPriorityToQuicPriority(request_priority),
Ryan Hamilton0d65a8c2019-06-07 00:46:02326 std::move(block), 0, nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55327 }
328
Ryan Hamilton8d9ee76e2018-05-29 23:52:52329 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectAuthRequestPacket(
Fan Yangac867502019-01-28 21:10:23330 uint64_t packet_number) {
Ryan Hamilton0239aac2018-05-19 00:03:13331 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55332 PopulateConnectRequestIR(&block);
333 block["proxy-authorization"] = "Basic Zm9vOmJhcg==";
334 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09335 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Yixin Wang7a3f1b8d2018-01-17 21:40:48336 ConvertRequestPriorityToQuicPriority(LOWEST), std::move(block), 0,
Ryan Hamilton0d65a8c2019-06-07 00:46:02337 nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55338 }
339
Ryan Hamilton8d9ee76e2018-05-29 23:52:52340 std::unique_ptr<quic::QuicReceivedPacket> ConstructDataPacket(
Fan Yangac867502019-01-28 21:10:23341 uint64_t packet_number,
Renjief49758b2019-01-11 23:32:41342 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09343 return client_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17344 !kIncludeVersion, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55345 }
346
Renjied172e812019-01-16 05:12:35347 std::unique_ptr<quic::QuicReceivedPacket> ConstructMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23348 uint64_t packet_number,
Renjied172e812019-01-16 05:12:35349 const std::vector<std::string> data_writes) {
350 return client_maker_.MakeMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17351 packet_number, client_data_stream_id1_, !kIncludeVersion, !kFin,
Renjied172e812019-01-16 05:12:35352 data_writes);
353 }
354
Ryan Hamilton8d9ee76e2018-05-29 23:52:52355 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndDataPacket(
Fan Yangac867502019-01-28 21:10:23356 uint64_t packet_number,
357 uint64_t largest_received,
358 uint64_t smallest_received,
359 uint64_t least_unacked,
Renjief49758b2019-01-11 23:32:41360 quic::QuicStringPiece data) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55361 return client_maker_.MakeAckAndDataPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09362 packet_number, !kIncludeVersion, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17363 largest_received, smallest_received, least_unacked, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55364 }
365
Renjied172e812019-01-16 05:12:35366 std::unique_ptr<quic::QuicReceivedPacket>
367 ConstructAckAndMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23368 uint64_t packet_number,
369 uint64_t largest_received,
370 uint64_t smallest_received,
371 uint64_t least_unacked,
Renjied172e812019-01-16 05:12:35372 const std::vector<std::string> data_writes) {
373 return client_maker_.MakeAckAndMultipleDataFramesPacket(
374 packet_number, !kIncludeVersion, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17375 largest_received, smallest_received, least_unacked, !kFin, data_writes);
Renjied172e812019-01-16 05:12:35376 }
377
Ryan Hamilton8d9ee76e2018-05-29 23:52:52378 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckPacket(
Fan Yangac867502019-01-28 21:10:23379 uint64_t packet_number,
380 uint64_t largest_received,
381 uint64_t smallest_received,
382 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55383 return client_maker_.MakeAckPacket(packet_number, largest_received,
384 smallest_received, least_unacked,
385 kSendFeedback);
386 }
387
388 // Helper functions for constructing packets sent by the server
389
Ryan Hamilton8d9ee76e2018-05-29 23:52:52390 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerRstPacket(
Fan Yangac867502019-01-28 21:10:23391 uint64_t packet_number,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41392 quic::QuicRstStreamErrorCode error_code) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55393 return server_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09394 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18395 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55396 }
397
Ryan Hamilton8d9ee76e2018-05-29 23:52:52398 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataPacket(
Fan Yangac867502019-01-28 21:10:23399 uint64_t packet_number,
Renjief49758b2019-01-11 23:32:41400 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09401 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17402 !kIncludeVersion, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55403 }
404
Ryan Hamilton8d9ee76e2018-05-29 23:52:52405 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataFinPacket(
Fan Yangac867502019-01-28 21:10:23406 uint64_t packet_number,
Renjief49758b2019-01-11 23:32:41407 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09408 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17409 !kIncludeVersion, kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55410 }
411
Ryan Hamilton8d9ee76e2018-05-29 23:52:52412 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerConnectReplyPacket(
Fan Yangac867502019-01-28 21:10:23413 uint64_t packet_number,
Yixin Wang0d2c6b7e12017-08-16 21:12:55414 bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13415 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55416 block[":status"] = "200";
417
418 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09419 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02420 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55421 }
422
Ryan Hamilton8d9ee76e2018-05-29 23:52:52423 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23424 ConstructServerConnectAuthReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13425 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55426 block[":status"] = "407";
427 block["proxy-authenticate"] = "Basic realm=\"MyRealm1\"";
428 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09429 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02430 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55431 }
432
Ryan Hamilton8d9ee76e2018-05-29 23:52:52433 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23434 ConstructServerConnectRedirectReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13435 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55436 block[":status"] = "302";
437 block["location"] = kRedirectUrl;
438 block["set-cookie"] = "foo=bar";
439 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09440 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02441 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55442 }
443
Ryan Hamilton8d9ee76e2018-05-29 23:52:52444 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23445 ConstructServerConnectErrorReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13446 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55447 block[":status"] = "500";
448
449 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09450 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02451 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55452 }
453
454 void AssertConnectSucceeds() {
455 TestCompletionCallback callback;
456 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
457 ASSERT_THAT(callback.WaitForResult(), IsOk());
458 }
459
460 void AssertConnectFails(int result) {
461 TestCompletionCallback callback;
462 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
463 ASSERT_EQ(result, callback.WaitForResult());
464 }
465
466 void ResumeAndRun() {
467 // Run until the pause, if the provider isn't paused yet.
468 SequencedSocketData* data = mock_quic_data_.GetSequencedSocketData();
469 data->RunUntilPaused();
470 data->Resume();
471 base::RunLoop().RunUntilIdle();
472 }
473
474 void AssertWriteReturns(const char* data, int len, int rv) {
Victor Costan9c7302b2018-08-27 16:39:44475 scoped_refptr<IOBufferWithSize> buf =
476 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55477 memcpy(buf->data(), data, len);
478 EXPECT_EQ(rv,
[email protected]578968d42017-12-13 15:39:32479 sock_->Write(buf.get(), buf->size(), write_callback_.callback(),
480 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55481 }
482
483 void AssertSyncWriteSucceeds(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44484 scoped_refptr<IOBufferWithSize> buf =
485 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55486 memcpy(buf->data(), data, len);
Bence Békyd8a21fc32018-06-27 18:29:58487 EXPECT_EQ(len,
488 sock_->Write(buf.get(), buf->size(), CompletionOnceCallback(),
489 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55490 }
491
492 void AssertSyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44493 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Bence Békyd8a21fc32018-06-27 18:29:58494 ASSERT_EQ(len, sock_->Read(buf.get(), len, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:13495 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55496 ASSERT_TRUE(sock_->IsConnected());
497 }
498
499 void AssertAsyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44500 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55501 ASSERT_EQ(ERR_IO_PENDING,
502 sock_->Read(buf.get(), len, read_callback_.callback()));
503 EXPECT_TRUE(sock_->IsConnected());
504
505 ResumeAndRun();
506
507 EXPECT_EQ(len, read_callback_.WaitForResult());
508 EXPECT_TRUE(sock_->IsConnected());
Ryan Hamilton0239aac2018-05-19 00:03:13509 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55510 }
511
512 void AssertReadStarts(const char* data, int len) {
513 // Issue the read, which will be completed asynchronously.
Victor Costan9c7302b2018-08-27 16:39:44514 read_buf_ = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55515 ASSERT_EQ(ERR_IO_PENDING,
516 sock_->Read(read_buf_.get(), len, read_callback_.callback()));
517 EXPECT_TRUE(sock_->IsConnected());
518 }
519
520 void AssertReadReturns(const char* data, int len) {
521 EXPECT_TRUE(sock_->IsConnected());
522
523 // Now the read will return.
524 EXPECT_EQ(len, read_callback_.WaitForResult());
Ryan Hamilton0239aac2018-05-19 00:03:13525 ASSERT_EQ(spdy::SpdyString(data, len),
526 spdy::SpdyString(read_buf_->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55527 }
528
Victor Vasiliev076657c2019-03-12 02:46:43529 std::string ConstructDataHeader(size_t body_len) {
Nick Harper23290b82019-05-02 00:02:56530 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41531 return "";
532 }
533 quic::HttpEncoder encoder;
534 std::unique_ptr<char[]> buffer;
535 auto header_length = encoder.SerializeDataFrameHeader(body_len, &buffer);
Victor Vasiliev076657c2019-03-12 02:46:43536 return std::string(buffer.get(), header_length);
Renjief49758b2019-01-11 23:32:41537 }
538
Eric Roman2346c1c2019-07-15 22:22:00539 BoundTestNetLog net_log_;
David Schinazi3f7465c2019-07-12 01:57:05540 QuicFlagSaver saver_;
Nick Harper23290b82019-05-02 00:02:56541 const quic::ParsedQuicVersion version_;
Ryan Hamilton47cf9d12018-10-17 04:33:09542 const quic::QuicStreamId client_data_stream_id1_;
Yixin Wang079ad542018-01-11 04:06:05543 const bool client_headers_include_h2_stream_dependency_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55544
545 // order of destruction of these members matter
Ryan Hamilton8d9ee76e2018-05-29 23:52:52546 quic::MockClock clock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55547 MockQuicData mock_quic_data_;
548 std::unique_ptr<QuicChromiumConnectionHelper> helper_;
549 std::unique_ptr<QuicChromiumClientSession> session_;
550 std::unique_ptr<QuicChromiumClientSession::Handle> session_handle_;
551 std::unique_ptr<QuicProxyClientSocket> sock_;
552
Ryan Hamilton8d9ee76e2018-05-29 23:52:52553 quic::test::MockSendAlgorithm* send_algorithm_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55554 scoped_refptr<TestTaskRunner> runner_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55555
556 std::unique_ptr<QuicChromiumAlarmFactory> alarm_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52557 testing::StrictMock<quic::test::MockQuicConnectionVisitor> visitor_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55558 TransportSecurityState transport_security_state_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52559 quic::QuicCryptoClientConfig crypto_config_;
560 quic::QuicClientPushPromiseIndex push_promise_index_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55561
Ryan Hamilton8d9ee76e2018-05-29 23:52:52562 const quic::QuicConnectionId connection_id_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55563 QuicTestPacketMaker client_maker_;
564 QuicTestPacketMaker server_maker_;
565 IPEndPoint peer_addr_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52566 quic::test::MockRandom random_generator_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55567 ProofVerifyDetailsChromium verify_details_;
568 MockCryptoClientStreamFactory crypto_client_stream_factory_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55569
570 std::string user_agent_;
571 HostPortPair proxy_host_port_;
572 HostPortPair endpoint_host_port_;
573 HttpAuthCache http_auth_cache_;
574 std::unique_ptr<MockHostResolverBase> host_resolver_;
575 std::unique_ptr<HttpAuthHandlerRegistryFactory> http_auth_handler_factory_;
576
577 TestCompletionCallback read_callback_;
578 scoped_refptr<IOBuffer> read_buf_;
579
580 TestCompletionCallback write_callback_;
581
582 DISALLOW_COPY_AND_ASSIGN(QuicProxyClientSocketTest);
583};
584
585TEST_P(QuicProxyClientSocketTest, ConnectSendsCorrectRequest) {
Zhongyi Shi32f2fd02018-04-16 18:23:43586 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
587 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
588 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55589 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
590 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52591 SYNCHRONOUS,
592 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55593
594 Initialize();
595
596 ASSERT_FALSE(sock_->IsConnected());
597
598 AssertConnectSucceeds();
599
600 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
601 ASSERT_TRUE(response != nullptr);
602 ASSERT_EQ(200, response->headers->response_code());
603}
604
605TEST_P(QuicProxyClientSocketTest, ConnectWithAuthRequested) {
Zhongyi Shi32f2fd02018-04-16 18:23:43606 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
607 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
608 mock_quic_data_.AddRead(ASYNC,
609 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55610 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
611 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52612 SYNCHRONOUS,
613 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55614
615 Initialize();
616
617 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
618
619 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
620 ASSERT_TRUE(response != nullptr);
621 ASSERT_EQ(407, response->headers->response_code());
622}
623
624TEST_P(QuicProxyClientSocketTest, ConnectWithAuthCredentials) {
Zhongyi Shi32f2fd02018-04-16 18:23:43625 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
626 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectAuthRequestPacket(2));
627 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55628 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
629 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52630 SYNCHRONOUS,
631 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55632
633 Initialize();
634
635 // Add auth to cache
636 const base::string16 kFoo(base::ASCIIToUTF16("foo"));
637 const base::string16 kBar(base::ASCIIToUTF16("bar"));
638 http_auth_cache_.Add(GURL(kProxyUrl), "MyRealm1", HttpAuth::AUTH_SCHEME_BASIC,
639 "Basic realm=MyRealm1", AuthCredentials(kFoo, kBar),
640 "/");
641
642 AssertConnectSucceeds();
643
644 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
645 ASSERT_TRUE(response != nullptr);
646 ASSERT_EQ(200, response->headers->response_code());
647}
648
Eric Roman96c5b292019-04-23 18:04:59649// Tests that a redirect response from a CONNECT fails.
Yixin Wang0d2c6b7e12017-08-16 21:12:55650TEST_P(QuicProxyClientSocketTest, ConnectRedirects) {
Zhongyi Shi32f2fd02018-04-16 18:23:43651 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
652 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
653 mock_quic_data_.AddRead(ASYNC,
654 ConstructServerConnectRedirectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55655 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
656 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52657 SYNCHRONOUS,
658 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55659
660 Initialize();
661
Eric Roman96c5b292019-04-23 18:04:59662 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
Yixin Wang0d2c6b7e12017-08-16 21:12:55663
664 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
665 ASSERT_TRUE(response != nullptr);
666
667 const HttpResponseHeaders* headers = response->headers.get();
668 ASSERT_EQ(302, headers->response_code());
Eric Roman96c5b292019-04-23 18:04:59669 ASSERT_TRUE(headers->HasHeader("set-cookie"));
Yixin Wang0d2c6b7e12017-08-16 21:12:55670
671 std::string location;
672 ASSERT_TRUE(headers->IsRedirect(&location));
673 ASSERT_EQ(location, kRedirectUrl);
674}
675
676TEST_P(QuicProxyClientSocketTest, ConnectFails) {
Zhongyi Shi32f2fd02018-04-16 18:23:43677 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
678 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Yixin Wang0d2c6b7e12017-08-16 21:12:55679 mock_quic_data_.AddRead(ASYNC, 0); // EOF
680
681 Initialize();
682
683 ASSERT_FALSE(sock_->IsConnected());
684
685 AssertConnectFails(ERR_QUIC_PROTOCOL_ERROR);
686
687 ASSERT_FALSE(sock_->IsConnected());
688}
689
690TEST_P(QuicProxyClientSocketTest, WasEverUsedReturnsCorrectValue) {
Zhongyi Shi32f2fd02018-04-16 18:23:43691 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
692 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
693 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55694 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
695 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52696 SYNCHRONOUS,
697 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55698
699 Initialize();
700
701 EXPECT_TRUE(sock_->WasEverUsed()); // Used due to crypto handshake
702 AssertConnectSucceeds();
703 EXPECT_TRUE(sock_->WasEverUsed());
704 sock_->Disconnect();
705 EXPECT_TRUE(sock_->WasEverUsed());
706}
707
708TEST_P(QuicProxyClientSocketTest, GetPeerAddressReturnsCorrectValues) {
Zhongyi Shi32f2fd02018-04-16 18:23:43709 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
710 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
711 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55712 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
713 mock_quic_data_.AddRead(ASYNC, 0); // EOF
714
715 Initialize();
716
717 IPEndPoint addr;
718 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
719
720 AssertConnectSucceeds();
721 EXPECT_TRUE(sock_->IsConnected());
722 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsOk());
723
724 ResumeAndRun();
725
726 EXPECT_FALSE(sock_->IsConnected());
727 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
728
729 sock_->Disconnect();
730
731 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
732}
733
734TEST_P(QuicProxyClientSocketTest, IsConnectedAndIdle) {
Zhongyi Shi32f2fd02018-04-16 18:23:43735 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
736 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
737 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55738 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
739
Victor Vasiliev076657c2019-03-12 02:46:43740 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:17741 mock_quic_data_.AddRead(
742 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43743 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55744 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41745 mock_quic_data_.AddWrite(SYNCHRONOUS,
746 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55747
748 Initialize();
749
750 EXPECT_FALSE(sock_->IsConnectedAndIdle());
751
752 AssertConnectSucceeds();
753
754 EXPECT_TRUE(sock_->IsConnectedAndIdle());
755
756 // The next read is consumed and buffered.
757 ResumeAndRun();
758
759 EXPECT_FALSE(sock_->IsConnectedAndIdle());
760
761 AssertSyncReadEquals(kMsg1, kLen1);
762
763 EXPECT_TRUE(sock_->IsConnectedAndIdle());
764}
765
766TEST_P(QuicProxyClientSocketTest, GetTotalReceivedBytes) {
Zhongyi Shi32f2fd02018-04-16 18:23:43767 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
768 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
769 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55770 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
771
Victor Vasiliev076657c2019-03-12 02:46:43772 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:41773 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:43774 ASYNC,
Ryan Hamilton7505eb92019-06-08 00:22:17775 ConstructServerDataPacket(2, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:43776 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55777 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41778 mock_quic_data_.AddWrite(SYNCHRONOUS,
779 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55780
781 Initialize();
782
783 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
784
785 AssertConnectSucceeds();
786
787 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
788
789 // The next read is consumed and buffered.
790 ResumeAndRun();
791
792 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
793
794 // The payload from the single large data frame will be read across
795 // two different reads.
796 AssertSyncReadEquals(kMsg33, kLen33);
797
Renjief49758b2019-01-11 23:32:41798 EXPECT_EQ((int64_t)(kLen33 + header.length()),
799 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55800
801 AssertSyncReadEquals(kMsg3, kLen3);
802
Renjief49758b2019-01-11 23:32:41803 EXPECT_EQ((int64_t)(kLen333 + header.length()),
804 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55805}
806
Lily Chenf11e1292018-11-29 16:42:09807TEST_P(QuicProxyClientSocketTest, SetStreamPriority) {
808 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
Matt Menkeedaf3b82019-03-14 21:39:44809 // Despite setting the priority to HIGHEST, the requets initial priority of
810 // LOWEST is used.
Lily Chenf11e1292018-11-29 16:42:09811 mock_quic_data_.AddWrite(SYNCHRONOUS,
Matt Menkeedaf3b82019-03-14 21:39:44812 ConstructConnectRequestPacket(2, LOWEST));
Lily Chenf11e1292018-11-29 16:42:09813 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
814 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
815 mock_quic_data_.AddWrite(
816 SYNCHRONOUS,
817 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
818
819 Initialize();
820
821 sock_->SetStreamPriority(HIGHEST);
822 AssertConnectSucceeds();
823}
Yixin Wang0d2c6b7e12017-08-16 21:12:55824
825TEST_P(QuicProxyClientSocketTest, WriteSendsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43826 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
827 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
828 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55829 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:56830 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:43831 std::string header = ConstructDataHeader(kLen1);
832 mock_quic_data_.AddWrite(
833 SYNCHRONOUS, ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17834 3, 1, 1, 1, {header, std::string(kMsg1, kLen1)}));
Victor Vasiliev076657c2019-03-12 02:46:43835 std::string header2 = ConstructDataHeader(kLen2);
Ryan Hamilton7505eb92019-06-08 00:22:17836 mock_quic_data_.AddWrite(SYNCHRONOUS,
837 ConstructMultipleDataFramesPacket(
838 4, {header2, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:41839 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41840 SYNCHRONOUS, ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED));
Renjief49758b2019-01-11 23:32:41841 } else {
842 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41843 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:17844 ConstructAckAndDataPacket(3, 1, 1, 1, std::string(kMsg1, kLen1)));
845 mock_quic_data_.AddWrite(SYNCHRONOUS,
846 ConstructDataPacket(4, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:41847 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41848 SYNCHRONOUS, ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED));
Renjief49758b2019-01-11 23:32:41849 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55850
851 Initialize();
852
853 AssertConnectSucceeds();
854
855 AssertSyncWriteSucceeds(kMsg1, kLen1);
856 AssertSyncWriteSucceeds(kMsg2, kLen2);
857}
858
859TEST_P(QuicProxyClientSocketTest, WriteSplitsLargeDataIntoMultiplePackets) {
Renjief49758b2019-01-11 23:32:41860 int write_packet_index = 1;
861 mock_quic_data_.AddWrite(SYNCHRONOUS,
862 ConstructSettingsPacket(write_packet_index++));
863 mock_quic_data_.AddWrite(SYNCHRONOUS,
864 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:43865 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55866 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:43867 std::string header = ConstructDataHeader(kLen1);
Nick Harper23290b82019-05-02 00:02:56868 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41869 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:17870 SYNCHRONOUS, ConstructAckAndDataPacket(write_packet_index++, 1, 1, 1,
Victor Vasiliev076657c2019-03-12 02:46:43871 std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:41872 } else {
Renjied172e812019-01-16 05:12:35873 mock_quic_data_.AddWrite(SYNCHRONOUS,
874 ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17875 write_packet_index++, 1, 1, 1,
Victor Vasiliev076657c2019-03-12 02:46:43876 {header, std::string(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41877 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55878
879 // Expect |kNumDataPackets| data packets, each containing the max possible
880 // amount of data.
Renjied172e812019-01-16 05:12:35881 int numDataPackets = 3;
882 std::string data(numDataPackets * quic::kDefaultMaxPacketSize, 'x');
Renjief49758b2019-01-11 23:32:41883 quic::QuicStreamOffset offset = kLen1 + header.length();
Renjied172e812019-01-16 05:12:35884
David Schinazi3f7465c2019-07-12 01:57:05885 if (version_.transport_version == quic::QUIC_VERSION_99 ||
886 version_.handshake_protocol == quic::PROTOCOL_TLS1_3) {
Renjied172e812019-01-16 05:12:35887 numDataPackets++;
Renjief49758b2019-01-11 23:32:41888 }
Renjied172e812019-01-16 05:12:35889 size_t total_data_length = 0;
890 for (int i = 0; i < numDataPackets; ++i) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55891 size_t max_packet_data_length = GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52892 quic::kDefaultMaxPacketSize, version_, !kIncludeVersion,
893 !kIncludeDiversificationNonce, quic::PACKET_8BYTE_CONNECTION_ID,
894 quic::PACKET_1BYTE_PACKET_NUMBER, offset);
Nick Harper23290b82019-05-02 00:02:56895 if (version_.transport_version == quic::QUIC_VERSION_99 && i == 0) {
Renjied172e812019-01-16 05:12:35896 // 3973 is the data frame length from packet length.
Victor Vasiliev076657c2019-03-12 02:46:43897 std::string header2 = ConstructDataHeader(3973);
Renjied172e812019-01-16 05:12:35898 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43899 SYNCHRONOUS, ConstructMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17900 write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:43901 {header2, std::string(data.c_str(),
902 max_packet_data_length - 7)}));
Renjied172e812019-01-16 05:12:35903 offset += max_packet_data_length - header2.length() - 1;
David Schinazi3f7465c2019-07-12 01:57:05904 } else if (version_.handshake_protocol == quic::PROTOCOL_TLS1_3 && i == 0) {
905 mock_quic_data_.AddWrite(
906 SYNCHRONOUS,
907 ConstructDataPacket(
908 write_packet_index++,
909 std::string(data.c_str(), max_packet_data_length - 4)));
910 offset += max_packet_data_length - 4;
Nick Harper23290b82019-05-02 00:02:56911 } else if (version_.transport_version == quic::QUIC_VERSION_99 &&
912 i == numDataPackets - 1) {
Renjied172e812019-01-16 05:12:35913 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:17914 SYNCHRONOUS, ConstructDataPacket(write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:43915 std::string(data.c_str(), 7)));
Renjied172e812019-01-16 05:12:35916 offset += 7;
David Schinazi3f7465c2019-07-12 01:57:05917 } else if (version_.handshake_protocol == quic::PROTOCOL_TLS1_3 &&
918 i == numDataPackets - 1) {
919 mock_quic_data_.AddWrite(
920 SYNCHRONOUS, ConstructDataPacket(write_packet_index++,
921 std::string(data.c_str(), 12)));
922 offset += 12;
923 } else if (version_.handshake_protocol == quic::PROTOCOL_TLS1_3) {
924 mock_quic_data_.AddWrite(
925 SYNCHRONOUS,
926 ConstructDataPacket(
927 write_packet_index++,
928 std::string(data.c_str(), max_packet_data_length - 4)));
929 offset += max_packet_data_length - 4;
Renjied172e812019-01-16 05:12:35930 } else {
931 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43932 SYNCHRONOUS, ConstructDataPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17933 write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:43934 std::string(data.c_str(), max_packet_data_length)));
Renjied172e812019-01-16 05:12:35935 offset += max_packet_data_length;
936 }
937 if (i != 3) {
938 total_data_length += max_packet_data_length;
939 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55940 }
Renjied172e812019-01-16 05:12:35941
Yixin Wang0d2c6b7e12017-08-16 21:12:55942 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41943 SYNCHRONOUS,
944 ConstructRstPacket(write_packet_index++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55945
946 Initialize();
947
948 AssertConnectSucceeds();
949
950 // Make a small write. An ACK and STOP_WAITING will be bundled. This prevents
951 // ACK and STOP_WAITING from being bundled with the subsequent large write.
952 // This allows the test code for computing the size of data sent in each
953 // packet to not become too complicated.
954 AssertSyncWriteSucceeds(kMsg1, kLen1);
955
956 // Make large write that should be split up
957 AssertSyncWriteSucceeds(data.c_str(), total_data_length);
958}
959
960// ----------- Read
961
962TEST_P(QuicProxyClientSocketTest, ReadReadsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43963 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
964 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
965 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55966 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
967
Victor Vasiliev076657c2019-03-12 02:46:43968 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:17969 mock_quic_data_.AddRead(
970 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43971 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55972 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41973 mock_quic_data_.AddWrite(SYNCHRONOUS,
974 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55975
976 Initialize();
977
978 AssertConnectSucceeds();
979
980 ResumeAndRun();
981 AssertSyncReadEquals(kMsg1, kLen1);
982}
983
984TEST_P(QuicProxyClientSocketTest, ReadDataFromBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:43985 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
986 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
987 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55988 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
989
Victor Vasiliev076657c2019-03-12 02:46:43990 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:17991 mock_quic_data_.AddRead(
992 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43993 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55994 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
995
Victor Vasiliev076657c2019-03-12 02:46:43996 std::string header2 = ConstructDataHeader(kLen2);
997 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:17998 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:55999 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1000
1001 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521002 SYNCHRONOUS,
1003 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551004
1005 Initialize();
1006
1007 AssertConnectSucceeds();
1008
1009 ResumeAndRun();
1010 AssertSyncReadEquals(kMsg1, kLen1);
1011
1012 ResumeAndRun();
1013 AssertSyncReadEquals(kMsg2, kLen2);
1014}
1015
1016TEST_P(QuicProxyClientSocketTest, ReadDataMultipleBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431017 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1018 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1019 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551020 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1021
Victor Vasiliev076657c2019-03-12 02:46:431022 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171023 mock_quic_data_.AddRead(
1024 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Victor Vasiliev076657c2019-03-12 02:46:431025 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1026 std::string header2 = ConstructDataHeader(kLen2);
1027 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171028 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551029 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1030
1031 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521032 SYNCHRONOUS,
1033 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551034
1035 Initialize();
1036
1037 AssertConnectSucceeds();
1038
1039 // The next two reads are consumed and buffered.
1040 ResumeAndRun();
1041
1042 AssertSyncReadEquals(kMsg1, kLen1);
1043 AssertSyncReadEquals(kMsg2, kLen2);
1044}
1045
1046TEST_P(QuicProxyClientSocketTest, LargeReadWillMergeDataFromDifferentFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431047 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1048 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1049 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551050 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1051
Victor Vasiliev076657c2019-03-12 02:46:431052 std::string header = ConstructDataHeader(kLen3);
Ryan Hamilton7505eb92019-06-08 00:22:171053 mock_quic_data_.AddRead(
1054 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg3, kLen3)));
Victor Vasiliev076657c2019-03-12 02:46:431055 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1056 std::string header2 = ConstructDataHeader(kLen3);
1057 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171058 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551059 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1060
1061 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521062 SYNCHRONOUS,
1063 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551064
1065 Initialize();
1066
1067 AssertConnectSucceeds();
1068
1069 // The next two reads are consumed and buffered.
1070 ResumeAndRun();
1071 // The payload from two data frames, each with kMsg3 will be combined
1072 // together into a single read().
1073 AssertSyncReadEquals(kMsg33, kLen33);
1074}
1075
1076TEST_P(QuicProxyClientSocketTest, MultipleShortReadsThenMoreRead) {
Zhongyi Shi32f2fd02018-04-16 18:23:431077 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1078 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1079 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551080 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1081
1082 int offset = 0;
1083
Victor Vasiliev076657c2019-03-12 02:46:431084 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411085 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171086 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411087 offset += kLen1 + header.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431088 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551089
Victor Vasiliev076657c2019-03-12 02:46:431090 std::string header2 = ConstructDataHeader(kLen3);
Renjief49758b2019-01-11 23:32:411091 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171092 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411093 offset += kLen3 + header2.length();
1094 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171095 ASYNC, ConstructServerDataPacket(4, header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411096 offset += kLen3 + header2.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431097 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(4, 4, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551098
Victor Vasiliev076657c2019-03-12 02:46:431099 std::string header3 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411100 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171101 ASYNC, ConstructServerDataPacket(5, header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411102 offset += kLen2 + header3.length();
Yixin Wang0d2c6b7e12017-08-16 21:12:551103 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1104
1105 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521106 SYNCHRONOUS,
1107 ConstructAckAndRstPacket(5, quic::QUIC_STREAM_CANCELLED, 5, 5, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551108
1109 Initialize();
1110
1111 AssertConnectSucceeds();
1112
1113 // The next 4 reads are consumed and buffered.
1114 ResumeAndRun();
1115
1116 AssertSyncReadEquals(kMsg1, kLen1);
1117 // The payload from two data frames, each with kMsg3 will be combined
1118 // together into a single read().
1119 AssertSyncReadEquals(kMsg33, kLen33);
1120 AssertSyncReadEquals(kMsg2, kLen2);
1121}
1122
1123TEST_P(QuicProxyClientSocketTest, ReadWillSplitDataFromLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431124 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1125 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1126 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551127 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1128
Victor Vasiliev076657c2019-03-12 02:46:431129 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171130 mock_quic_data_.AddRead(
1131 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431132 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431133 std::string header2 = ConstructDataHeader(kLen33);
Ryan Hamilton7505eb92019-06-08 00:22:171134 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1135 3, header2 + std::string(kMsg33, kLen33)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551136 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1137
1138 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521139 SYNCHRONOUS,
1140 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551141
1142 Initialize();
1143
1144 AssertConnectSucceeds();
1145
1146 // The next 2 reads are consumed and buffered.
1147 ResumeAndRun();
1148
1149 AssertSyncReadEquals(kMsg1, kLen1);
1150 // The payload from the single large data frame will be read across
1151 // two different reads.
1152 AssertSyncReadEquals(kMsg3, kLen3);
1153 AssertSyncReadEquals(kMsg3, kLen3);
1154}
1155
1156TEST_P(QuicProxyClientSocketTest, MultipleReadsFromSameLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431157 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1158 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1159 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551160 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1161
Victor Vasiliev076657c2019-03-12 02:46:431162 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:411163 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431164 ASYNC,
Ryan Hamilton7505eb92019-06-08 00:22:171165 ConstructServerDataPacket(2, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:431166 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551167 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1168
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411169 mock_quic_data_.AddWrite(SYNCHRONOUS,
1170 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551171
1172 Initialize();
1173
1174 AssertConnectSucceeds();
1175
1176 // The next read is consumed and buffered.
1177 ResumeAndRun();
1178
1179 // The payload from the single large data frame will be read across
1180 // two different reads.
1181 AssertSyncReadEquals(kMsg33, kLen33);
1182
1183 // Now attempt to do a read of more data than remains buffered
Victor Costan9c7302b2018-08-27 16:39:441184 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kLen33);
Bence Békyd8a21fc32018-06-27 18:29:581185 ASSERT_EQ(kLen3, sock_->Read(buf.get(), kLen33, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:131186 ASSERT_EQ(spdy::SpdyString(kMsg3, kLen3),
1187 spdy::SpdyString(buf->data(), kLen3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551188 ASSERT_TRUE(sock_->IsConnected());
1189}
1190
1191TEST_P(QuicProxyClientSocketTest, ReadAuthResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431192 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1193 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1194 mock_quic_data_.AddRead(ASYNC,
1195 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551196 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1197
Victor Vasiliev076657c2019-03-12 02:46:431198 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171199 mock_quic_data_.AddRead(
1200 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Victor Vasiliev076657c2019-03-12 02:46:431201 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1202 std::string header2 = ConstructDataHeader(kLen2);
1203 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171204 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551205 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1206
1207 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521208 SYNCHRONOUS,
1209 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551210
1211 Initialize();
1212
1213 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
1214
1215 // The next two reads are consumed and buffered.
1216 ResumeAndRun();
1217
1218 AssertSyncReadEquals(kMsg1, kLen1);
1219 AssertSyncReadEquals(kMsg2, kLen2);
1220}
1221
1222TEST_P(QuicProxyClientSocketTest, ReadErrorResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431223 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1224 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1225 mock_quic_data_.AddRead(ASYNC,
1226 ConstructServerConnectErrorReplyPacket(1, !kFin));
Victor Vasiliev076657c2019-03-12 02:46:431227 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411228 mock_quic_data_.AddRead(
1229 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:171230 ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431231 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431232 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411233 mock_quic_data_.AddRead(
1234 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:171235 ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551236 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1237
1238 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521239 SYNCHRONOUS,
1240 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551241 Initialize();
1242
1243 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
1244}
1245
1246// ----------- Reads and Writes
1247
1248TEST_P(QuicProxyClientSocketTest, AsyncReadAroundWrite) {
Renjief49758b2019-01-11 23:32:411249 int write_packet_index = 1;
1250 mock_quic_data_.AddWrite(SYNCHRONOUS,
1251 ConstructSettingsPacket(write_packet_index++));
1252 mock_quic_data_.AddWrite(SYNCHRONOUS,
1253 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431254 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551255 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1256
Victor Vasiliev076657c2019-03-12 02:46:431257 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171258 mock_quic_data_.AddRead(
1259 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431260 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjief49758b2019-01-11 23:32:411261 ConstructAckPacket(write_packet_index++, 2, 1, 1));
1262
Victor Vasiliev076657c2019-03-12 02:46:431263 std::string header2 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561264 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:431265 mock_quic_data_.AddWrite(
1266 SYNCHRONOUS,
1267 ConstructMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171268 write_packet_index++, {header2, std::string(kMsg2, kLen2)}));
Renjied172e812019-01-16 05:12:351269 } else {
Renjief49758b2019-01-11 23:32:411270 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171271 SYNCHRONOUS,
1272 ConstructDataPacket(write_packet_index++, std::string(kMsg2, kLen2)));
Renjied172e812019-01-16 05:12:351273 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551274
1275 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1276
Victor Vasiliev076657c2019-03-12 02:46:431277 std::string header3 = ConstructDataHeader(kLen3);
1278 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171279 ASYNC, ConstructServerDataPacket(3, header3 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551280 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1281
1282 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411283 SYNCHRONOUS,
1284 ConstructAckAndRstPacket(write_packet_index++,
1285 quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551286
1287 Initialize();
1288
1289 AssertConnectSucceeds();
1290
1291 ResumeAndRun();
1292
1293 AssertSyncReadEquals(kMsg1, kLen1);
1294
1295 AssertReadStarts(kMsg3, kLen3);
1296 // Read should block until after the write succeeds.
1297
1298 AssertSyncWriteSucceeds(kMsg2, kLen2);
1299
1300 ASSERT_FALSE(read_callback_.have_result());
1301 ResumeAndRun();
1302
1303 // Now the read will return.
1304 AssertReadReturns(kMsg3, kLen3);
1305}
1306
1307TEST_P(QuicProxyClientSocketTest, AsyncWriteAroundReads) {
Renjied172e812019-01-16 05:12:351308 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1309 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Zhongyi Shi32f2fd02018-04-16 18:23:431310 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551311 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1312
Victor Vasiliev076657c2019-03-12 02:46:431313 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171314 mock_quic_data_.AddRead(
1315 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjied172e812019-01-16 05:12:351316 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551317 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1318
Victor Vasiliev076657c2019-03-12 02:46:431319 std::string header2 = ConstructDataHeader(kLen3);
1320 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171321 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551322 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1323
1324 mock_quic_data_.AddWrite(ASYNC, ERR_IO_PENDING); // Pause
1325
Victor Vasiliev076657c2019-03-12 02:46:431326 std::string header3 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561327 if (version_.transport_version != quic::QUIC_VERSION_99) {
Ryan Hamilton7505eb92019-06-08 00:22:171328 mock_quic_data_.AddWrite(ASYNC,
1329 ConstructDataPacket(4, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411330 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171331 SYNCHRONOUS,
1332 ConstructAckAndDataPacket(5, 3, 3, 1, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411333 } else {
Victor Vasiliev076657c2019-03-12 02:46:431334 mock_quic_data_.AddWrite(ASYNC,
1335 ConstructMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171336 4, {header3, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:411337 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171338 ASYNC, ConstructAckAndDataPacket(5, 3, 3, 1,
Victor Vasiliev076657c2019-03-12 02:46:431339 header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411340 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551341
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411342 mock_quic_data_.AddWrite(SYNCHRONOUS,
1343 ConstructRstPacket(6, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551344
1345 Initialize();
1346
1347 AssertConnectSucceeds();
1348
1349 ResumeAndRun();
1350 AssertSyncReadEquals(kMsg1, kLen1);
1351
1352 // Write should block until the next read completes.
1353 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1354 // asynchronous starting with the second time it's called while the UDP socket
1355 // is write-blocked. Therefore, at least two writes need to be called on
1356 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351357 AssertWriteReturns(kMsg2, kLen2, kLen2);
Yixin Wang0d2c6b7e12017-08-16 21:12:551358 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1359
1360 AssertAsyncReadEquals(kMsg3, kLen3);
1361
1362 ASSERT_FALSE(write_callback_.have_result());
1363
1364 // Now the write will complete
1365 ResumeAndRun();
Yixin Wangdbbd8752018-01-17 21:50:021366 EXPECT_EQ(kLen2, write_callback_.WaitForResult());
Yixin Wang0d2c6b7e12017-08-16 21:12:551367}
1368
1369// ----------- Reading/Writing on Closed socket
1370
1371// Reading from an already closed socket should return 0
1372TEST_P(QuicProxyClientSocketTest, ReadOnClosedSocketReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431373 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1374 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1375 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551376 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1377
1378 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1379
1380 Initialize();
1381
1382 AssertConnectSucceeds();
1383
1384 ResumeAndRun();
1385
1386 ASSERT_FALSE(sock_->IsConnected());
Raul Tambre94493c652019-03-11 17:18:351387 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1388 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1389 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551390 ASSERT_FALSE(sock_->IsConnectedAndIdle());
1391}
1392
1393// Read pending when socket is closed should return 0
1394TEST_P(QuicProxyClientSocketTest, PendingReadOnCloseReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431395 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1396 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1397 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551398 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1399
1400 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1401
1402 Initialize();
1403
1404 AssertConnectSucceeds();
1405
1406 AssertReadStarts(kMsg1, kLen1);
1407
1408 ResumeAndRun();
1409
1410 ASSERT_EQ(0, read_callback_.WaitForResult());
1411}
1412
1413// Reading from a disconnected socket is an error
1414TEST_P(QuicProxyClientSocketTest, ReadOnDisconnectSocketReturnsNotConnected) {
Zhongyi Shi32f2fd02018-04-16 18:23:431415 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1416 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1417 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551418 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1419 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521420 SYNCHRONOUS,
1421 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551422
1423 Initialize();
1424
1425 AssertConnectSucceeds();
1426
1427 sock_->Disconnect();
1428
1429 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581430 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551431}
1432
1433// Reading data after receiving FIN should return buffered data received before
1434// FIN, then 0.
1435TEST_P(QuicProxyClientSocketTest, ReadAfterFinReceivedReturnsBufferedData) {
Zhongyi Shi32f2fd02018-04-16 18:23:431436 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1437 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1438 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551439 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1440
Victor Vasiliev076657c2019-03-12 02:46:431441 std::string header = ConstructDataHeader(kLen1);
1442 mock_quic_data_.AddRead(ASYNC, ConstructServerDataFinPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171443 2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431444 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551445 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411446 mock_quic_data_.AddWrite(SYNCHRONOUS,
1447 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551448
1449 Initialize();
1450
1451 AssertConnectSucceeds();
1452
1453 ResumeAndRun();
1454
1455 AssertSyncReadEquals(kMsg1, kLen1);
Raul Tambre94493c652019-03-11 17:18:351456 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1457 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551458
1459 sock_->Disconnect();
1460 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581461 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551462}
1463
1464// Calling Write() on a closed socket is an error.
1465TEST_P(QuicProxyClientSocketTest, WriteOnClosedStream) {
Zhongyi Shi32f2fd02018-04-16 18:23:431466 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1467 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1468 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551469 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1470
1471 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1472
1473 Initialize();
1474
1475 AssertConnectSucceeds();
1476
1477 ResumeAndRun();
1478
1479 AssertWriteReturns(kMsg1, kLen1, ERR_QUIC_PROTOCOL_ERROR);
1480}
1481
1482// Calling Write() on a disconnected socket is an error.
1483TEST_P(QuicProxyClientSocketTest, WriteOnDisconnectedSocket) {
Zhongyi Shi32f2fd02018-04-16 18:23:431484 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1485 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1486 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551487 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1488 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521489 SYNCHRONOUS,
1490 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551491
1492 Initialize();
1493
1494 AssertConnectSucceeds();
1495
1496 sock_->Disconnect();
1497
1498 AssertWriteReturns(kMsg1, kLen1, ERR_SOCKET_NOT_CONNECTED);
1499}
1500
1501// If the socket is closed with a pending Write(), the callback should be called
1502// with the same error the session was closed with.
1503TEST_P(QuicProxyClientSocketTest, WritePendingOnClose) {
Zhongyi Shi32f2fd02018-04-16 18:23:431504 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1505 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1506 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551507 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1508 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1509
1510 Initialize();
1511
1512 AssertConnectSucceeds();
1513
1514 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1515 // asynchronous starting with the second time it's called while the UDP socket
1516 // is write-blocked. Therefore, at least two writes need to be called on
1517 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351518 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551519
1520 // This second write will be async. This is the pending write that's being
1521 // tested.
1522 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1523
1524 // Make sure the write actually starts.
1525 base::RunLoop().RunUntilIdle();
1526
Ryan Hamilton8d9ee76e2018-05-29 23:52:521527 session_->CloseSessionOnError(ERR_CONNECTION_CLOSED,
Renjieba55fae2018-09-20 03:05:161528 quic::QUIC_INTERNAL_ERROR,
1529 quic::ConnectionCloseBehavior::SILENT_CLOSE);
Yixin Wang0d2c6b7e12017-08-16 21:12:551530
1531 EXPECT_THAT(write_callback_.WaitForResult(), IsError(ERR_CONNECTION_CLOSED));
1532}
1533
1534TEST_P(QuicProxyClientSocketTest, DisconnectWithWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431535 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1536 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1537 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551538 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1539 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1540
1541 Initialize();
1542
1543 AssertConnectSucceeds();
1544
1545 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1546 // asynchronous starting with the second time it's called while the UDP socket
1547 // is write-blocked. Therefore, at least two writes need to be called on
1548 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351549 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551550
1551 // This second write will be async. This is the pending write that's being
1552 // tested.
1553 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1554
1555 // Make sure the write actually starts.
1556 base::RunLoop().RunUntilIdle();
1557
1558 sock_->Disconnect();
1559 EXPECT_FALSE(sock_->IsConnected());
1560
1561 base::RunLoop().RunUntilIdle();
1562
1563 EXPECT_FALSE(sock_->IsConnected());
1564 EXPECT_FALSE(write_callback_.have_result());
1565}
1566
1567// If the socket is Disconnected with a pending Read(), the callback
1568// should not be called.
1569TEST_P(QuicProxyClientSocketTest, DisconnectWithReadPending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431570 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1571 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1572 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551573 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1574 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521575 SYNCHRONOUS,
1576 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551577
1578 Initialize();
1579
1580 AssertConnectSucceeds();
1581
1582 EXPECT_TRUE(sock_->IsConnected());
1583
1584 AssertReadStarts(kMsg1, kLen1);
1585
1586 sock_->Disconnect();
1587 EXPECT_FALSE(sock_->IsConnected());
1588
1589 base::RunLoop().RunUntilIdle();
1590
1591 EXPECT_FALSE(sock_->IsConnected());
1592 EXPECT_FALSE(read_callback_.have_result());
1593}
1594
1595// If the socket is Reset when both a read and write are pending,
1596// both should be called back.
1597TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431598 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1599 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1600 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551601 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1602
1603 mock_quic_data_.AddRead(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411604 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED));
Yixin Wang2bea3cf2017-11-09 18:11:031605 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:431606 std::string header = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561607 if (version_.transport_version != quic::QUIC_VERSION_99) {
Ryan Hamilton7505eb92019-06-08 00:22:171608 mock_quic_data_.AddWrite(ASYNC, ConstructAckAndDataPacket(
1609 3, 1, 1, 1, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411610 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411611 SYNCHRONOUS,
1612 ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411613 } else {
Renjief49758b2019-01-11 23:32:411614 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351615 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171616 3, 1, 1, 1, {header, std::string(kMsg2, kLen2)}));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411617 mock_quic_data_.AddWrite(
1618 SYNCHRONOUS,
1619 ConstructAckAndRstOnlyPacket(4, quic::QUIC_STREAM_CANCELLED, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411620 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551621
1622 Initialize();
1623
1624 AssertConnectSucceeds();
1625
1626 EXPECT_TRUE(sock_->IsConnected());
1627
1628 AssertReadStarts(kMsg1, kLen1);
1629
1630 // Write should block until the next read completes.
1631 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1632 // asynchronous starting with the second time it's called while the UDP socket
1633 // is write-blocked. Therefore, at least two writes need to be called on
1634 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351635 AssertWriteReturns(kMsg2, kLen2, kLen2);
Renjief49758b2019-01-11 23:32:411636
Yixin Wang0d2c6b7e12017-08-16 21:12:551637 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1638
1639 ResumeAndRun();
1640
1641 EXPECT_TRUE(read_callback_.have_result());
1642 EXPECT_TRUE(write_callback_.have_result());
1643}
1644
1645// Makes sure the proxy client socket's source gets the expected NetLog events
1646// and only the expected NetLog events (No SpdySession events).
1647TEST_P(QuicProxyClientSocketTest, NetLog) {
Zhongyi Shi32f2fd02018-04-16 18:23:431648 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1649 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1650 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551651 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1652
Victor Vasiliev076657c2019-03-12 02:46:431653 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171654 mock_quic_data_.AddRead(
1655 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431656 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang2bea3cf2017-11-09 18:11:031657 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411658 mock_quic_data_.AddWrite(SYNCHRONOUS,
1659 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551660
1661 Initialize();
1662
1663 AssertConnectSucceeds();
1664
1665 ResumeAndRun();
1666 AssertSyncReadEquals(kMsg1, kLen1);
1667
1668 NetLogSource sock_source = sock_->NetLog().source();
1669 sock_.reset();
1670
Eric Roman79cc7552019-07-19 02:17:541671 auto entry_list = net_log_.GetEntriesForSource(sock_source);
Yixin Wang0d2c6b7e12017-08-16 21:12:551672
1673 ASSERT_EQ(entry_list.size(), 10u);
1674 EXPECT_TRUE(
1675 LogContainsBeginEvent(entry_list, 0, NetLogEventType::SOCKET_ALIVE));
1676 EXPECT_TRUE(LogContainsEvent(entry_list, 1,
1677 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
1678 NetLogEventPhase::NONE));
1679 EXPECT_TRUE(LogContainsBeginEvent(
1680 entry_list, 2, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1681 EXPECT_TRUE(LogContainsEvent(
1682 entry_list, 3, NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
1683 NetLogEventPhase::NONE));
1684 EXPECT_TRUE(LogContainsEndEvent(
1685 entry_list, 4, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1686 EXPECT_TRUE(LogContainsBeginEvent(
1687 entry_list, 5, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1688 EXPECT_TRUE(LogContainsEvent(
1689 entry_list, 6,
1690 NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
1691 NetLogEventPhase::NONE));
1692 EXPECT_TRUE(LogContainsEndEvent(
1693 entry_list, 7, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1694 EXPECT_TRUE(LogContainsEvent(entry_list, 8,
1695 NetLogEventType::SOCKET_BYTES_RECEIVED,
1696 NetLogEventPhase::NONE));
1697 EXPECT_TRUE(
1698 LogContainsEndEvent(entry_list, 9, NetLogEventType::SOCKET_ALIVE));
1699}
1700
Bence Béky8ddc2492018-06-13 01:02:041701// A helper class that will delete |sock| when the callback is invoked.
Yixin Wang0d2c6b7e12017-08-16 21:12:551702class DeleteSockCallback : public TestCompletionCallbackBase {
1703 public:
1704 explicit DeleteSockCallback(std::unique_ptr<QuicProxyClientSocket>* sock)
Bence Béky8ddc2492018-06-13 01:02:041705 : sock_(sock) {}
Yixin Wang0d2c6b7e12017-08-16 21:12:551706
1707 ~DeleteSockCallback() override {}
1708
Bence Béky8ddc2492018-06-13 01:02:041709 CompletionOnceCallback callback() {
1710 return base::BindOnce(&DeleteSockCallback::OnComplete,
1711 base::Unretained(this));
1712 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551713
1714 private:
1715 void OnComplete(int result) {
Raul Tambre94493c652019-03-11 17:18:351716 sock_->reset(nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:551717 SetResult(result);
1718 }
1719
1720 std::unique_ptr<QuicProxyClientSocket>* sock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:551721
1722 DISALLOW_COPY_AND_ASSIGN(DeleteSockCallback);
1723};
1724
1725// If the socket is reset when both a read and write are pending, and the
1726// read callback causes the socket to be deleted, the write callback should
1727// not be called.
1728TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePendingDelete) {
Zhongyi Shi32f2fd02018-04-16 18:23:431729 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1730 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1731 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551732 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1733
1734 mock_quic_data_.AddRead(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411735 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551736 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:561737 if (version_.transport_version != quic::QUIC_VERSION_99) {
Ryan Hamilton7505eb92019-06-08 00:22:171738 mock_quic_data_.AddWrite(ASYNC, ConstructAckAndDataPacket(
1739 3, 1, 1, 1, std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411740 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411741 SYNCHRONOUS,
1742 ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411743 } else {
Victor Vasiliev076657c2019-03-12 02:46:431744 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411745 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351746 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171747 3, 1, 1, 1, {header, std::string(kMsg1, kLen1)}));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411748 mock_quic_data_.AddWrite(
1749 SYNCHRONOUS,
1750 ConstructAckAndRstOnlyPacket(4, quic::QUIC_STREAM_CANCELLED, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411751 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551752
1753 Initialize();
1754
1755 AssertConnectSucceeds();
1756
1757 EXPECT_TRUE(sock_->IsConnected());
1758
1759 DeleteSockCallback read_callback(&sock_);
Victor Costan9c7302b2018-08-27 16:39:441760 scoped_refptr<IOBuffer> read_buf = base::MakeRefCounted<IOBuffer>(kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551761 ASSERT_EQ(ERR_IO_PENDING,
1762 sock_->Read(read_buf.get(), kLen1, read_callback.callback()));
1763
1764 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1765 // asynchronous starting with the second time it's called while the UDP socket
1766 // is write-blocked. Therefore, at least two writes need to be called on
1767 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351768 AssertWriteReturns(kMsg1, kLen1, kLen1);
Renjief49758b2019-01-11 23:32:411769
Yixin Wang0d2c6b7e12017-08-16 21:12:551770 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1771
1772 ResumeAndRun();
1773
1774 EXPECT_FALSE(sock_.get());
1775
1776 EXPECT_EQ(0, read_callback.WaitForResult());
1777 EXPECT_FALSE(write_callback_.have_result());
1778}
1779
Victor Costane635086f2019-01-27 05:20:301780INSTANTIATE_TEST_SUITE_P(
Bence Békyce380cb2018-04-26 23:39:551781 VersionIncludeStreamDependencySequence,
Yixin Wang079ad542018-01-11 04:06:051782 QuicProxyClientSocketTest,
Victor Vasiliev5d6cdc22019-05-28 20:37:431783 ::testing::Combine(::testing::ValuesIn(quic::AllVersionsExcept99()),
Nick Harper23290b82019-05-02 00:02:561784 ::testing::Bool()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551785
1786} // namespace test
Dan Zhangf11470172017-09-18 22:02:091787} // namespace net