blob: f41a99a2e61740b9bfb9f1b271dca34dffa131da [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>>,
Gabriel Charette694c3c332019-08-19 14:53:0587 public WithTaskEnvironment {
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_,
Kevin DiClementef07119082019-08-12 13:31:34268 http_auth_handler_factory_.get(), host_resolver_.get(),
269 HttpAuthPreferences::ALLOW_DEFAULT_CREDENTIALS)));
Yixin Wang0d2c6b7e12017-08-16 21:12:55270
271 session_->StartReading();
272 }
273
Ryan Hamilton0239aac2018-05-19 00:03:13274 void PopulateConnectRequestIR(spdy::SpdyHeaderBlock* block) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55275 (*block)[":method"] = "CONNECT";
276 (*block)[":authority"] = endpoint_host_port_.ToString();
277 (*block)["user-agent"] = kUserAgent;
278 }
279
280 // Helper functions for constructing packets sent by the client
281
Ryan Hamilton8d9ee76e2018-05-29 23:52:52282 std::unique_ptr<quic::QuicReceivedPacket> ConstructSettingsPacket(
Fan Yangac867502019-01-28 21:10:23283 uint64_t packet_number) {
Ryan Hamilton0d65a8c2019-06-07 00:46:02284 return client_maker_.MakeInitialSettingsPacket(packet_number);
Yixin Wang0d2c6b7e12017-08-16 21:12:55285 }
286
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41287 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstOnlyPacket(
Fan Yangac867502019-01-28 21:10:23288 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52289 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23290 uint64_t largest_received,
291 uint64_t smallest_received,
292 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55293 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09294 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18295 largest_received, smallest_received, least_unacked, kSendFeedback,
Frank Kastenholz684ea412019-02-13 18:48:18296 /*include_stop_sending=*/false);
297 }
298
Ryan Hamilton8d9ee76e2018-05-29 23:52:52299 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23300 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52301 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23302 uint64_t largest_received,
303 uint64_t smallest_received,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41304 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55305 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09306 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55307 largest_received, smallest_received, least_unacked, kSendFeedback,
Frank Kastenholz684ea412019-02-13 18:48:18308 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55309 }
310
Ryan Hamilton8d9ee76e2018-05-29 23:52:52311 std::unique_ptr<quic::QuicReceivedPacket> ConstructRstPacket(
Fan Yangac867502019-01-28 21:10:23312 uint64_t packet_number,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41313 quic::QuicRstStreamErrorCode error_code) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55314 return client_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09315 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18316 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55317 }
318
Ryan Hamilton8d9ee76e2018-05-29 23:52:52319 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectRequestPacket(
Fan Yangac867502019-01-28 21:10:23320 uint64_t packet_number,
Lily Chenf11e1292018-11-29 16:42:09321 RequestPriority request_priority = LOWEST) {
Ryan Hamilton0239aac2018-05-19 00:03:13322 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55323 PopulateConnectRequestIR(&block);
324 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09325 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Lily Chenf11e1292018-11-29 16:42:09326 ConvertRequestPriorityToQuicPriority(request_priority),
Ryan Hamilton0d65a8c2019-06-07 00:46:02327 std::move(block), 0, nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55328 }
329
Ryan Hamilton8d9ee76e2018-05-29 23:52:52330 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectAuthRequestPacket(
Fan Yangac867502019-01-28 21:10:23331 uint64_t packet_number) {
Ryan Hamilton0239aac2018-05-19 00:03:13332 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55333 PopulateConnectRequestIR(&block);
334 block["proxy-authorization"] = "Basic Zm9vOmJhcg==";
335 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09336 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Yixin Wang7a3f1b8d2018-01-17 21:40:48337 ConvertRequestPriorityToQuicPriority(LOWEST), std::move(block), 0,
Ryan Hamilton0d65a8c2019-06-07 00:46:02338 nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55339 }
340
Ryan Hamilton8d9ee76e2018-05-29 23:52:52341 std::unique_ptr<quic::QuicReceivedPacket> ConstructDataPacket(
Fan Yangac867502019-01-28 21:10:23342 uint64_t packet_number,
Renjief49758b2019-01-11 23:32:41343 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09344 return client_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17345 !kIncludeVersion, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55346 }
347
Renjied172e812019-01-16 05:12:35348 std::unique_ptr<quic::QuicReceivedPacket> ConstructMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23349 uint64_t packet_number,
Renjied172e812019-01-16 05:12:35350 const std::vector<std::string> data_writes) {
351 return client_maker_.MakeMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17352 packet_number, client_data_stream_id1_, !kIncludeVersion, !kFin,
Renjied172e812019-01-16 05:12:35353 data_writes);
354 }
355
Ryan Hamilton8d9ee76e2018-05-29 23:52:52356 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndDataPacket(
Fan Yangac867502019-01-28 21:10:23357 uint64_t packet_number,
358 uint64_t largest_received,
359 uint64_t smallest_received,
360 uint64_t least_unacked,
Renjief49758b2019-01-11 23:32:41361 quic::QuicStringPiece data) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55362 return client_maker_.MakeAckAndDataPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09363 packet_number, !kIncludeVersion, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17364 largest_received, smallest_received, least_unacked, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55365 }
366
Renjied172e812019-01-16 05:12:35367 std::unique_ptr<quic::QuicReceivedPacket>
368 ConstructAckAndMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23369 uint64_t packet_number,
370 uint64_t largest_received,
371 uint64_t smallest_received,
372 uint64_t least_unacked,
Renjied172e812019-01-16 05:12:35373 const std::vector<std::string> data_writes) {
374 return client_maker_.MakeAckAndMultipleDataFramesPacket(
375 packet_number, !kIncludeVersion, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17376 largest_received, smallest_received, least_unacked, !kFin, data_writes);
Renjied172e812019-01-16 05:12:35377 }
378
Ryan Hamilton8d9ee76e2018-05-29 23:52:52379 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckPacket(
Fan Yangac867502019-01-28 21:10:23380 uint64_t packet_number,
381 uint64_t largest_received,
382 uint64_t smallest_received,
383 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55384 return client_maker_.MakeAckPacket(packet_number, largest_received,
385 smallest_received, least_unacked,
386 kSendFeedback);
387 }
388
389 // Helper functions for constructing packets sent by the server
390
Ryan Hamilton8d9ee76e2018-05-29 23:52:52391 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerRstPacket(
Fan Yangac867502019-01-28 21:10:23392 uint64_t packet_number,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41393 quic::QuicRstStreamErrorCode error_code) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55394 return server_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09395 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18396 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55397 }
398
Ryan Hamilton8d9ee76e2018-05-29 23:52:52399 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataPacket(
Fan Yangac867502019-01-28 21:10:23400 uint64_t packet_number,
Renjief49758b2019-01-11 23:32:41401 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09402 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17403 !kIncludeVersion, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55404 }
405
Ryan Hamilton8d9ee76e2018-05-29 23:52:52406 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataFinPacket(
Fan Yangac867502019-01-28 21:10:23407 uint64_t packet_number,
Renjief49758b2019-01-11 23:32:41408 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09409 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17410 !kIncludeVersion, kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55411 }
412
Ryan Hamilton8d9ee76e2018-05-29 23:52:52413 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerConnectReplyPacket(
Fan Yangac867502019-01-28 21:10:23414 uint64_t packet_number,
Yixin Wang0d2c6b7e12017-08-16 21:12:55415 bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13416 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55417 block[":status"] = "200";
418
419 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09420 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02421 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55422 }
423
Ryan Hamilton8d9ee76e2018-05-29 23:52:52424 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23425 ConstructServerConnectAuthReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13426 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55427 block[":status"] = "407";
428 block["proxy-authenticate"] = "Basic realm=\"MyRealm1\"";
429 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09430 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02431 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55432 }
433
Ryan Hamilton8d9ee76e2018-05-29 23:52:52434 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23435 ConstructServerConnectRedirectReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13436 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55437 block[":status"] = "302";
438 block["location"] = kRedirectUrl;
439 block["set-cookie"] = "foo=bar";
440 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09441 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02442 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55443 }
444
Ryan Hamilton8d9ee76e2018-05-29 23:52:52445 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23446 ConstructServerConnectErrorReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13447 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55448 block[":status"] = "500";
449
450 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09451 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02452 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55453 }
454
455 void AssertConnectSucceeds() {
456 TestCompletionCallback callback;
457 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
458 ASSERT_THAT(callback.WaitForResult(), IsOk());
459 }
460
461 void AssertConnectFails(int result) {
462 TestCompletionCallback callback;
463 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
464 ASSERT_EQ(result, callback.WaitForResult());
465 }
466
467 void ResumeAndRun() {
468 // Run until the pause, if the provider isn't paused yet.
469 SequencedSocketData* data = mock_quic_data_.GetSequencedSocketData();
470 data->RunUntilPaused();
471 data->Resume();
472 base::RunLoop().RunUntilIdle();
473 }
474
475 void AssertWriteReturns(const char* data, int len, int rv) {
Victor Costan9c7302b2018-08-27 16:39:44476 scoped_refptr<IOBufferWithSize> buf =
477 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55478 memcpy(buf->data(), data, len);
479 EXPECT_EQ(rv,
[email protected]578968d42017-12-13 15:39:32480 sock_->Write(buf.get(), buf->size(), write_callback_.callback(),
481 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55482 }
483
484 void AssertSyncWriteSucceeds(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44485 scoped_refptr<IOBufferWithSize> buf =
486 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55487 memcpy(buf->data(), data, len);
Bence Békyd8a21fc32018-06-27 18:29:58488 EXPECT_EQ(len,
489 sock_->Write(buf.get(), buf->size(), CompletionOnceCallback(),
490 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55491 }
492
493 void AssertSyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44494 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Bence Békyd8a21fc32018-06-27 18:29:58495 ASSERT_EQ(len, sock_->Read(buf.get(), len, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:13496 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55497 ASSERT_TRUE(sock_->IsConnected());
498 }
499
500 void AssertAsyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44501 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55502 ASSERT_EQ(ERR_IO_PENDING,
503 sock_->Read(buf.get(), len, read_callback_.callback()));
504 EXPECT_TRUE(sock_->IsConnected());
505
506 ResumeAndRun();
507
508 EXPECT_EQ(len, read_callback_.WaitForResult());
509 EXPECT_TRUE(sock_->IsConnected());
Ryan Hamilton0239aac2018-05-19 00:03:13510 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55511 }
512
513 void AssertReadStarts(const char* data, int len) {
514 // Issue the read, which will be completed asynchronously.
Victor Costan9c7302b2018-08-27 16:39:44515 read_buf_ = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55516 ASSERT_EQ(ERR_IO_PENDING,
517 sock_->Read(read_buf_.get(), len, read_callback_.callback()));
518 EXPECT_TRUE(sock_->IsConnected());
519 }
520
521 void AssertReadReturns(const char* data, int len) {
522 EXPECT_TRUE(sock_->IsConnected());
523
524 // Now the read will return.
525 EXPECT_EQ(len, read_callback_.WaitForResult());
Ryan Hamilton0239aac2018-05-19 00:03:13526 ASSERT_EQ(spdy::SpdyString(data, len),
527 spdy::SpdyString(read_buf_->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55528 }
529
Victor Vasiliev076657c2019-03-12 02:46:43530 std::string ConstructDataHeader(size_t body_len) {
Nick Harper23290b82019-05-02 00:02:56531 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41532 return "";
533 }
534 quic::HttpEncoder encoder;
535 std::unique_ptr<char[]> buffer;
536 auto header_length = encoder.SerializeDataFrameHeader(body_len, &buffer);
Victor Vasiliev076657c2019-03-12 02:46:43537 return std::string(buffer.get(), header_length);
Renjief49758b2019-01-11 23:32:41538 }
539
Eric Roman2346c1c2019-07-15 22:22:00540 BoundTestNetLog net_log_;
David Schinazi3f7465c2019-07-12 01:57:05541 QuicFlagSaver saver_;
Nick Harper23290b82019-05-02 00:02:56542 const quic::ParsedQuicVersion version_;
Ryan Hamilton47cf9d12018-10-17 04:33:09543 const quic::QuicStreamId client_data_stream_id1_;
Yixin Wang079ad542018-01-11 04:06:05544 const bool client_headers_include_h2_stream_dependency_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55545
546 // order of destruction of these members matter
Ryan Hamilton8d9ee76e2018-05-29 23:52:52547 quic::MockClock clock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55548 MockQuicData mock_quic_data_;
549 std::unique_ptr<QuicChromiumConnectionHelper> helper_;
550 std::unique_ptr<QuicChromiumClientSession> session_;
551 std::unique_ptr<QuicChromiumClientSession::Handle> session_handle_;
552 std::unique_ptr<QuicProxyClientSocket> sock_;
553
Ryan Hamilton8d9ee76e2018-05-29 23:52:52554 quic::test::MockSendAlgorithm* send_algorithm_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55555 scoped_refptr<TestTaskRunner> runner_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55556
557 std::unique_ptr<QuicChromiumAlarmFactory> alarm_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52558 testing::StrictMock<quic::test::MockQuicConnectionVisitor> visitor_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55559 TransportSecurityState transport_security_state_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52560 quic::QuicCryptoClientConfig crypto_config_;
561 quic::QuicClientPushPromiseIndex push_promise_index_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55562
Ryan Hamilton8d9ee76e2018-05-29 23:52:52563 const quic::QuicConnectionId connection_id_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55564 QuicTestPacketMaker client_maker_;
565 QuicTestPacketMaker server_maker_;
566 IPEndPoint peer_addr_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52567 quic::test::MockRandom random_generator_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55568 ProofVerifyDetailsChromium verify_details_;
569 MockCryptoClientStreamFactory crypto_client_stream_factory_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55570
571 std::string user_agent_;
572 HostPortPair proxy_host_port_;
573 HostPortPair endpoint_host_port_;
574 HttpAuthCache http_auth_cache_;
575 std::unique_ptr<MockHostResolverBase> host_resolver_;
576 std::unique_ptr<HttpAuthHandlerRegistryFactory> http_auth_handler_factory_;
577
578 TestCompletionCallback read_callback_;
579 scoped_refptr<IOBuffer> read_buf_;
580
581 TestCompletionCallback write_callback_;
582
583 DISALLOW_COPY_AND_ASSIGN(QuicProxyClientSocketTest);
584};
585
586TEST_P(QuicProxyClientSocketTest, ConnectSendsCorrectRequest) {
Zhongyi Shi32f2fd02018-04-16 18:23:43587 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
588 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
589 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55590 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
591 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52592 SYNCHRONOUS,
593 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55594
595 Initialize();
596
597 ASSERT_FALSE(sock_->IsConnected());
598
599 AssertConnectSucceeds();
600
601 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
602 ASSERT_TRUE(response != nullptr);
603 ASSERT_EQ(200, response->headers->response_code());
604}
605
606TEST_P(QuicProxyClientSocketTest, ConnectWithAuthRequested) {
Zhongyi Shi32f2fd02018-04-16 18:23:43607 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
608 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
609 mock_quic_data_.AddRead(ASYNC,
610 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55611 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
612 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52613 SYNCHRONOUS,
614 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55615
616 Initialize();
617
618 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
619
620 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
621 ASSERT_TRUE(response != nullptr);
622 ASSERT_EQ(407, response->headers->response_code());
623}
624
625TEST_P(QuicProxyClientSocketTest, ConnectWithAuthCredentials) {
Zhongyi Shi32f2fd02018-04-16 18:23:43626 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
627 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectAuthRequestPacket(2));
628 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55629 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
630 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52631 SYNCHRONOUS,
632 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55633
634 Initialize();
635
636 // Add auth to cache
637 const base::string16 kFoo(base::ASCIIToUTF16("foo"));
638 const base::string16 kBar(base::ASCIIToUTF16("bar"));
639 http_auth_cache_.Add(GURL(kProxyUrl), "MyRealm1", HttpAuth::AUTH_SCHEME_BASIC,
640 "Basic realm=MyRealm1", AuthCredentials(kFoo, kBar),
641 "/");
642
643 AssertConnectSucceeds();
644
645 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
646 ASSERT_TRUE(response != nullptr);
647 ASSERT_EQ(200, response->headers->response_code());
648}
649
Eric Roman96c5b292019-04-23 18:04:59650// Tests that a redirect response from a CONNECT fails.
Yixin Wang0d2c6b7e12017-08-16 21:12:55651TEST_P(QuicProxyClientSocketTest, ConnectRedirects) {
Zhongyi Shi32f2fd02018-04-16 18:23:43652 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
653 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
654 mock_quic_data_.AddRead(ASYNC,
655 ConstructServerConnectRedirectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55656 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
657 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52658 SYNCHRONOUS,
659 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55660
661 Initialize();
662
Eric Roman96c5b292019-04-23 18:04:59663 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
Yixin Wang0d2c6b7e12017-08-16 21:12:55664
665 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
666 ASSERT_TRUE(response != nullptr);
667
668 const HttpResponseHeaders* headers = response->headers.get();
669 ASSERT_EQ(302, headers->response_code());
Eric Roman96c5b292019-04-23 18:04:59670 ASSERT_TRUE(headers->HasHeader("set-cookie"));
Yixin Wang0d2c6b7e12017-08-16 21:12:55671
672 std::string location;
673 ASSERT_TRUE(headers->IsRedirect(&location));
674 ASSERT_EQ(location, kRedirectUrl);
675}
676
677TEST_P(QuicProxyClientSocketTest, ConnectFails) {
Zhongyi Shi32f2fd02018-04-16 18:23:43678 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
679 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Yixin Wang0d2c6b7e12017-08-16 21:12:55680 mock_quic_data_.AddRead(ASYNC, 0); // EOF
681
682 Initialize();
683
684 ASSERT_FALSE(sock_->IsConnected());
685
686 AssertConnectFails(ERR_QUIC_PROTOCOL_ERROR);
687
688 ASSERT_FALSE(sock_->IsConnected());
689}
690
691TEST_P(QuicProxyClientSocketTest, WasEverUsedReturnsCorrectValue) {
Zhongyi Shi32f2fd02018-04-16 18:23:43692 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
693 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
694 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55695 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
696 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52697 SYNCHRONOUS,
698 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55699
700 Initialize();
701
702 EXPECT_TRUE(sock_->WasEverUsed()); // Used due to crypto handshake
703 AssertConnectSucceeds();
704 EXPECT_TRUE(sock_->WasEverUsed());
705 sock_->Disconnect();
706 EXPECT_TRUE(sock_->WasEverUsed());
707}
708
709TEST_P(QuicProxyClientSocketTest, GetPeerAddressReturnsCorrectValues) {
Zhongyi Shi32f2fd02018-04-16 18:23:43710 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
711 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
712 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55713 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
714 mock_quic_data_.AddRead(ASYNC, 0); // EOF
715
716 Initialize();
717
718 IPEndPoint addr;
719 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
720
721 AssertConnectSucceeds();
722 EXPECT_TRUE(sock_->IsConnected());
723 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsOk());
724
725 ResumeAndRun();
726
727 EXPECT_FALSE(sock_->IsConnected());
728 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
729
730 sock_->Disconnect();
731
732 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
733}
734
735TEST_P(QuicProxyClientSocketTest, IsConnectedAndIdle) {
Zhongyi Shi32f2fd02018-04-16 18:23:43736 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
737 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
738 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55739 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
740
Victor Vasiliev076657c2019-03-12 02:46:43741 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:17742 mock_quic_data_.AddRead(
743 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43744 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55745 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41746 mock_quic_data_.AddWrite(SYNCHRONOUS,
747 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55748
749 Initialize();
750
751 EXPECT_FALSE(sock_->IsConnectedAndIdle());
752
753 AssertConnectSucceeds();
754
755 EXPECT_TRUE(sock_->IsConnectedAndIdle());
756
757 // The next read is consumed and buffered.
758 ResumeAndRun();
759
760 EXPECT_FALSE(sock_->IsConnectedAndIdle());
761
762 AssertSyncReadEquals(kMsg1, kLen1);
763
764 EXPECT_TRUE(sock_->IsConnectedAndIdle());
765}
766
767TEST_P(QuicProxyClientSocketTest, GetTotalReceivedBytes) {
Zhongyi Shi32f2fd02018-04-16 18:23:43768 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
769 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
770 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55771 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
772
Victor Vasiliev076657c2019-03-12 02:46:43773 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:41774 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:43775 ASYNC,
Ryan Hamilton7505eb92019-06-08 00:22:17776 ConstructServerDataPacket(2, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:43777 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55778 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41779 mock_quic_data_.AddWrite(SYNCHRONOUS,
780 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55781
782 Initialize();
783
784 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
785
786 AssertConnectSucceeds();
787
788 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
789
790 // The next read is consumed and buffered.
791 ResumeAndRun();
792
793 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
794
795 // The payload from the single large data frame will be read across
796 // two different reads.
797 AssertSyncReadEquals(kMsg33, kLen33);
798
Renjief49758b2019-01-11 23:32:41799 EXPECT_EQ((int64_t)(kLen33 + header.length()),
800 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55801
802 AssertSyncReadEquals(kMsg3, kLen3);
803
Renjief49758b2019-01-11 23:32:41804 EXPECT_EQ((int64_t)(kLen333 + header.length()),
805 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55806}
807
Lily Chenf11e1292018-11-29 16:42:09808TEST_P(QuicProxyClientSocketTest, SetStreamPriority) {
809 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
Matt Menkeedaf3b82019-03-14 21:39:44810 // Despite setting the priority to HIGHEST, the requets initial priority of
811 // LOWEST is used.
Lily Chenf11e1292018-11-29 16:42:09812 mock_quic_data_.AddWrite(SYNCHRONOUS,
Matt Menkeedaf3b82019-03-14 21:39:44813 ConstructConnectRequestPacket(2, LOWEST));
Lily Chenf11e1292018-11-29 16:42:09814 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
815 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
816 mock_quic_data_.AddWrite(
817 SYNCHRONOUS,
818 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
819
820 Initialize();
821
822 sock_->SetStreamPriority(HIGHEST);
823 AssertConnectSucceeds();
824}
Yixin Wang0d2c6b7e12017-08-16 21:12:55825
826TEST_P(QuicProxyClientSocketTest, WriteSendsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43827 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
828 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
829 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55830 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:56831 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:43832 std::string header = ConstructDataHeader(kLen1);
833 mock_quic_data_.AddWrite(
834 SYNCHRONOUS, ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17835 3, 1, 1, 1, {header, std::string(kMsg1, kLen1)}));
Victor Vasiliev076657c2019-03-12 02:46:43836 std::string header2 = ConstructDataHeader(kLen2);
Ryan Hamilton7505eb92019-06-08 00:22:17837 mock_quic_data_.AddWrite(SYNCHRONOUS,
838 ConstructMultipleDataFramesPacket(
839 4, {header2, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:41840 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41841 SYNCHRONOUS, ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED));
Renjief49758b2019-01-11 23:32:41842 } else {
843 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41844 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:17845 ConstructAckAndDataPacket(3, 1, 1, 1, std::string(kMsg1, kLen1)));
846 mock_quic_data_.AddWrite(SYNCHRONOUS,
847 ConstructDataPacket(4, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:41848 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41849 SYNCHRONOUS, ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED));
Renjief49758b2019-01-11 23:32:41850 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55851
852 Initialize();
853
854 AssertConnectSucceeds();
855
856 AssertSyncWriteSucceeds(kMsg1, kLen1);
857 AssertSyncWriteSucceeds(kMsg2, kLen2);
858}
859
860TEST_P(QuicProxyClientSocketTest, WriteSplitsLargeDataIntoMultiplePackets) {
Renjief49758b2019-01-11 23:32:41861 int write_packet_index = 1;
862 mock_quic_data_.AddWrite(SYNCHRONOUS,
863 ConstructSettingsPacket(write_packet_index++));
864 mock_quic_data_.AddWrite(SYNCHRONOUS,
865 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:43866 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55867 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:43868 std::string header = ConstructDataHeader(kLen1);
Nick Harper23290b82019-05-02 00:02:56869 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41870 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:17871 SYNCHRONOUS, ConstructAckAndDataPacket(write_packet_index++, 1, 1, 1,
Victor Vasiliev076657c2019-03-12 02:46:43872 std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:41873 } else {
Renjied172e812019-01-16 05:12:35874 mock_quic_data_.AddWrite(SYNCHRONOUS,
875 ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17876 write_packet_index++, 1, 1, 1,
Victor Vasiliev076657c2019-03-12 02:46:43877 {header, std::string(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41878 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55879
880 // Expect |kNumDataPackets| data packets, each containing the max possible
881 // amount of data.
Renjied172e812019-01-16 05:12:35882 int numDataPackets = 3;
883 std::string data(numDataPackets * quic::kDefaultMaxPacketSize, 'x');
Renjief49758b2019-01-11 23:32:41884 quic::QuicStreamOffset offset = kLen1 + header.length();
Renjied172e812019-01-16 05:12:35885
David Schinazi3f7465c2019-07-12 01:57:05886 if (version_.transport_version == quic::QUIC_VERSION_99 ||
887 version_.handshake_protocol == quic::PROTOCOL_TLS1_3) {
Renjied172e812019-01-16 05:12:35888 numDataPackets++;
Renjief49758b2019-01-11 23:32:41889 }
Renjied172e812019-01-16 05:12:35890 size_t total_data_length = 0;
891 for (int i = 0; i < numDataPackets; ++i) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55892 size_t max_packet_data_length = GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52893 quic::kDefaultMaxPacketSize, version_, !kIncludeVersion,
894 !kIncludeDiversificationNonce, quic::PACKET_8BYTE_CONNECTION_ID,
895 quic::PACKET_1BYTE_PACKET_NUMBER, offset);
Nick Harper23290b82019-05-02 00:02:56896 if (version_.transport_version == quic::QUIC_VERSION_99 && i == 0) {
Renjied172e812019-01-16 05:12:35897 // 3973 is the data frame length from packet length.
Victor Vasiliev076657c2019-03-12 02:46:43898 std::string header2 = ConstructDataHeader(3973);
Renjied172e812019-01-16 05:12:35899 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43900 SYNCHRONOUS, ConstructMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17901 write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:43902 {header2, std::string(data.c_str(),
903 max_packet_data_length - 7)}));
Renjied172e812019-01-16 05:12:35904 offset += max_packet_data_length - header2.length() - 1;
David Schinazi3f7465c2019-07-12 01:57:05905 } else if (version_.handshake_protocol == quic::PROTOCOL_TLS1_3 && i == 0) {
906 mock_quic_data_.AddWrite(
907 SYNCHRONOUS,
908 ConstructDataPacket(
909 write_packet_index++,
910 std::string(data.c_str(), max_packet_data_length - 4)));
911 offset += max_packet_data_length - 4;
Nick Harper23290b82019-05-02 00:02:56912 } else if (version_.transport_version == quic::QUIC_VERSION_99 &&
913 i == numDataPackets - 1) {
Renjied172e812019-01-16 05:12:35914 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:17915 SYNCHRONOUS, ConstructDataPacket(write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:43916 std::string(data.c_str(), 7)));
Renjied172e812019-01-16 05:12:35917 offset += 7;
David Schinazi3f7465c2019-07-12 01:57:05918 } else if (version_.handshake_protocol == quic::PROTOCOL_TLS1_3 &&
919 i == numDataPackets - 1) {
920 mock_quic_data_.AddWrite(
921 SYNCHRONOUS, ConstructDataPacket(write_packet_index++,
922 std::string(data.c_str(), 12)));
923 offset += 12;
924 } else if (version_.handshake_protocol == quic::PROTOCOL_TLS1_3) {
925 mock_quic_data_.AddWrite(
926 SYNCHRONOUS,
927 ConstructDataPacket(
928 write_packet_index++,
929 std::string(data.c_str(), max_packet_data_length - 4)));
930 offset += max_packet_data_length - 4;
Renjied172e812019-01-16 05:12:35931 } else {
932 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43933 SYNCHRONOUS, ConstructDataPacket(
Ryan Hamilton7505eb92019-06-08 00:22:17934 write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:43935 std::string(data.c_str(), max_packet_data_length)));
Renjied172e812019-01-16 05:12:35936 offset += max_packet_data_length;
937 }
938 if (i != 3) {
939 total_data_length += max_packet_data_length;
940 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55941 }
Renjied172e812019-01-16 05:12:35942
Yixin Wang0d2c6b7e12017-08-16 21:12:55943 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41944 SYNCHRONOUS,
945 ConstructRstPacket(write_packet_index++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55946
947 Initialize();
948
949 AssertConnectSucceeds();
950
951 // Make a small write. An ACK and STOP_WAITING will be bundled. This prevents
952 // ACK and STOP_WAITING from being bundled with the subsequent large write.
953 // This allows the test code for computing the size of data sent in each
954 // packet to not become too complicated.
955 AssertSyncWriteSucceeds(kMsg1, kLen1);
956
957 // Make large write that should be split up
958 AssertSyncWriteSucceeds(data.c_str(), total_data_length);
959}
960
961// ----------- Read
962
963TEST_P(QuicProxyClientSocketTest, ReadReadsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43964 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
965 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
966 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55967 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
968
Victor Vasiliev076657c2019-03-12 02:46:43969 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:17970 mock_quic_data_.AddRead(
971 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43972 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55973 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41974 mock_quic_data_.AddWrite(SYNCHRONOUS,
975 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55976
977 Initialize();
978
979 AssertConnectSucceeds();
980
981 ResumeAndRun();
982 AssertSyncReadEquals(kMsg1, kLen1);
983}
984
985TEST_P(QuicProxyClientSocketTest, ReadDataFromBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:43986 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
987 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
988 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55989 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
990
Victor Vasiliev076657c2019-03-12 02:46:43991 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:17992 mock_quic_data_.AddRead(
993 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43994 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55995 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
996
Victor Vasiliev076657c2019-03-12 02:46:43997 std::string header2 = ConstructDataHeader(kLen2);
998 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:17999 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551000 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1001
1002 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521003 SYNCHRONOUS,
1004 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551005
1006 Initialize();
1007
1008 AssertConnectSucceeds();
1009
1010 ResumeAndRun();
1011 AssertSyncReadEquals(kMsg1, kLen1);
1012
1013 ResumeAndRun();
1014 AssertSyncReadEquals(kMsg2, kLen2);
1015}
1016
1017TEST_P(QuicProxyClientSocketTest, ReadDataMultipleBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431018 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1019 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1020 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551021 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1022
Victor Vasiliev076657c2019-03-12 02:46:431023 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171024 mock_quic_data_.AddRead(
1025 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Victor Vasiliev076657c2019-03-12 02:46:431026 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1027 std::string header2 = ConstructDataHeader(kLen2);
1028 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171029 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551030 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1031
1032 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521033 SYNCHRONOUS,
1034 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551035
1036 Initialize();
1037
1038 AssertConnectSucceeds();
1039
1040 // The next two reads are consumed and buffered.
1041 ResumeAndRun();
1042
1043 AssertSyncReadEquals(kMsg1, kLen1);
1044 AssertSyncReadEquals(kMsg2, kLen2);
1045}
1046
1047TEST_P(QuicProxyClientSocketTest, LargeReadWillMergeDataFromDifferentFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431048 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1049 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1050 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551051 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1052
Victor Vasiliev076657c2019-03-12 02:46:431053 std::string header = ConstructDataHeader(kLen3);
Ryan Hamilton7505eb92019-06-08 00:22:171054 mock_quic_data_.AddRead(
1055 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg3, kLen3)));
Victor Vasiliev076657c2019-03-12 02:46:431056 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1057 std::string header2 = ConstructDataHeader(kLen3);
1058 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171059 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551060 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1061
1062 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521063 SYNCHRONOUS,
1064 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551065
1066 Initialize();
1067
1068 AssertConnectSucceeds();
1069
1070 // The next two reads are consumed and buffered.
1071 ResumeAndRun();
1072 // The payload from two data frames, each with kMsg3 will be combined
1073 // together into a single read().
1074 AssertSyncReadEquals(kMsg33, kLen33);
1075}
1076
1077TEST_P(QuicProxyClientSocketTest, MultipleShortReadsThenMoreRead) {
Zhongyi Shi32f2fd02018-04-16 18:23:431078 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1079 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1080 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551081 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1082
1083 int offset = 0;
1084
Victor Vasiliev076657c2019-03-12 02:46:431085 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411086 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171087 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411088 offset += kLen1 + header.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431089 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551090
Victor Vasiliev076657c2019-03-12 02:46:431091 std::string header2 = ConstructDataHeader(kLen3);
Renjief49758b2019-01-11 23:32:411092 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171093 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411094 offset += kLen3 + header2.length();
1095 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171096 ASYNC, ConstructServerDataPacket(4, header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411097 offset += kLen3 + header2.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431098 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(4, 4, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551099
Victor Vasiliev076657c2019-03-12 02:46:431100 std::string header3 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411101 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171102 ASYNC, ConstructServerDataPacket(5, header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411103 offset += kLen2 + header3.length();
Yixin Wang0d2c6b7e12017-08-16 21:12:551104 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1105
1106 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521107 SYNCHRONOUS,
1108 ConstructAckAndRstPacket(5, quic::QUIC_STREAM_CANCELLED, 5, 5, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551109
1110 Initialize();
1111
1112 AssertConnectSucceeds();
1113
1114 // The next 4 reads are consumed and buffered.
1115 ResumeAndRun();
1116
1117 AssertSyncReadEquals(kMsg1, kLen1);
1118 // The payload from two data frames, each with kMsg3 will be combined
1119 // together into a single read().
1120 AssertSyncReadEquals(kMsg33, kLen33);
1121 AssertSyncReadEquals(kMsg2, kLen2);
1122}
1123
1124TEST_P(QuicProxyClientSocketTest, ReadWillSplitDataFromLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431125 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1126 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1127 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551128 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1129
Victor Vasiliev076657c2019-03-12 02:46:431130 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171131 mock_quic_data_.AddRead(
1132 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431133 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431134 std::string header2 = ConstructDataHeader(kLen33);
Ryan Hamilton7505eb92019-06-08 00:22:171135 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1136 3, header2 + std::string(kMsg33, kLen33)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551137 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1138
1139 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521140 SYNCHRONOUS,
1141 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551142
1143 Initialize();
1144
1145 AssertConnectSucceeds();
1146
1147 // The next 2 reads are consumed and buffered.
1148 ResumeAndRun();
1149
1150 AssertSyncReadEquals(kMsg1, kLen1);
1151 // The payload from the single large data frame will be read across
1152 // two different reads.
1153 AssertSyncReadEquals(kMsg3, kLen3);
1154 AssertSyncReadEquals(kMsg3, kLen3);
1155}
1156
1157TEST_P(QuicProxyClientSocketTest, MultipleReadsFromSameLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431158 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1159 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1160 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551161 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1162
Victor Vasiliev076657c2019-03-12 02:46:431163 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:411164 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431165 ASYNC,
Ryan Hamilton7505eb92019-06-08 00:22:171166 ConstructServerDataPacket(2, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:431167 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551168 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1169
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411170 mock_quic_data_.AddWrite(SYNCHRONOUS,
1171 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551172
1173 Initialize();
1174
1175 AssertConnectSucceeds();
1176
1177 // The next read is consumed and buffered.
1178 ResumeAndRun();
1179
1180 // The payload from the single large data frame will be read across
1181 // two different reads.
1182 AssertSyncReadEquals(kMsg33, kLen33);
1183
1184 // Now attempt to do a read of more data than remains buffered
Victor Costan9c7302b2018-08-27 16:39:441185 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kLen33);
Bence Békyd8a21fc32018-06-27 18:29:581186 ASSERT_EQ(kLen3, sock_->Read(buf.get(), kLen33, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:131187 ASSERT_EQ(spdy::SpdyString(kMsg3, kLen3),
1188 spdy::SpdyString(buf->data(), kLen3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551189 ASSERT_TRUE(sock_->IsConnected());
1190}
1191
1192TEST_P(QuicProxyClientSocketTest, ReadAuthResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431193 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1194 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1195 mock_quic_data_.AddRead(ASYNC,
1196 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551197 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1198
Victor Vasiliev076657c2019-03-12 02:46:431199 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171200 mock_quic_data_.AddRead(
1201 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Victor Vasiliev076657c2019-03-12 02:46:431202 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1203 std::string header2 = ConstructDataHeader(kLen2);
1204 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171205 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551206 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1207
1208 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521209 SYNCHRONOUS,
1210 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551211
1212 Initialize();
1213
1214 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
1215
1216 // The next two reads are consumed and buffered.
1217 ResumeAndRun();
1218
1219 AssertSyncReadEquals(kMsg1, kLen1);
1220 AssertSyncReadEquals(kMsg2, kLen2);
1221}
1222
1223TEST_P(QuicProxyClientSocketTest, ReadErrorResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431224 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1225 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1226 mock_quic_data_.AddRead(ASYNC,
1227 ConstructServerConnectErrorReplyPacket(1, !kFin));
Victor Vasiliev076657c2019-03-12 02:46:431228 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411229 mock_quic_data_.AddRead(
1230 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:171231 ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431232 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431233 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411234 mock_quic_data_.AddRead(
1235 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:171236 ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551237 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1238
1239 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521240 SYNCHRONOUS,
1241 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551242 Initialize();
1243
1244 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
1245}
1246
1247// ----------- Reads and Writes
1248
1249TEST_P(QuicProxyClientSocketTest, AsyncReadAroundWrite) {
Renjief49758b2019-01-11 23:32:411250 int write_packet_index = 1;
1251 mock_quic_data_.AddWrite(SYNCHRONOUS,
1252 ConstructSettingsPacket(write_packet_index++));
1253 mock_quic_data_.AddWrite(SYNCHRONOUS,
1254 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431255 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551256 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1257
Victor Vasiliev076657c2019-03-12 02:46:431258 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171259 mock_quic_data_.AddRead(
1260 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431261 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjief49758b2019-01-11 23:32:411262 ConstructAckPacket(write_packet_index++, 2, 1, 1));
1263
Victor Vasiliev076657c2019-03-12 02:46:431264 std::string header2 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561265 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:431266 mock_quic_data_.AddWrite(
1267 SYNCHRONOUS,
1268 ConstructMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171269 write_packet_index++, {header2, std::string(kMsg2, kLen2)}));
Renjied172e812019-01-16 05:12:351270 } else {
Renjief49758b2019-01-11 23:32:411271 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171272 SYNCHRONOUS,
1273 ConstructDataPacket(write_packet_index++, std::string(kMsg2, kLen2)));
Renjied172e812019-01-16 05:12:351274 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551275
1276 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1277
Victor Vasiliev076657c2019-03-12 02:46:431278 std::string header3 = ConstructDataHeader(kLen3);
1279 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171280 ASYNC, ConstructServerDataPacket(3, header3 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551281 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1282
1283 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411284 SYNCHRONOUS,
1285 ConstructAckAndRstPacket(write_packet_index++,
1286 quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551287
1288 Initialize();
1289
1290 AssertConnectSucceeds();
1291
1292 ResumeAndRun();
1293
1294 AssertSyncReadEquals(kMsg1, kLen1);
1295
1296 AssertReadStarts(kMsg3, kLen3);
1297 // Read should block until after the write succeeds.
1298
1299 AssertSyncWriteSucceeds(kMsg2, kLen2);
1300
1301 ASSERT_FALSE(read_callback_.have_result());
1302 ResumeAndRun();
1303
1304 // Now the read will return.
1305 AssertReadReturns(kMsg3, kLen3);
1306}
1307
1308TEST_P(QuicProxyClientSocketTest, AsyncWriteAroundReads) {
Renjied172e812019-01-16 05:12:351309 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1310 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Zhongyi Shi32f2fd02018-04-16 18:23:431311 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551312 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1313
Victor Vasiliev076657c2019-03-12 02:46:431314 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171315 mock_quic_data_.AddRead(
1316 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjied172e812019-01-16 05:12:351317 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551318 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1319
Victor Vasiliev076657c2019-03-12 02:46:431320 std::string header2 = ConstructDataHeader(kLen3);
1321 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171322 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551323 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1324
1325 mock_quic_data_.AddWrite(ASYNC, ERR_IO_PENDING); // Pause
1326
Victor Vasiliev076657c2019-03-12 02:46:431327 std::string header3 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561328 if (version_.transport_version != quic::QUIC_VERSION_99) {
Ryan Hamilton7505eb92019-06-08 00:22:171329 mock_quic_data_.AddWrite(ASYNC,
1330 ConstructDataPacket(4, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411331 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171332 SYNCHRONOUS,
1333 ConstructAckAndDataPacket(5, 3, 3, 1, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411334 } else {
Victor Vasiliev076657c2019-03-12 02:46:431335 mock_quic_data_.AddWrite(ASYNC,
1336 ConstructMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171337 4, {header3, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:411338 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171339 ASYNC, ConstructAckAndDataPacket(5, 3, 3, 1,
Victor Vasiliev076657c2019-03-12 02:46:431340 header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411341 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551342
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411343 mock_quic_data_.AddWrite(SYNCHRONOUS,
1344 ConstructRstPacket(6, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551345
1346 Initialize();
1347
1348 AssertConnectSucceeds();
1349
1350 ResumeAndRun();
1351 AssertSyncReadEquals(kMsg1, kLen1);
1352
1353 // Write should block until the next read completes.
1354 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1355 // asynchronous starting with the second time it's called while the UDP socket
1356 // is write-blocked. Therefore, at least two writes need to be called on
1357 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351358 AssertWriteReturns(kMsg2, kLen2, kLen2);
Yixin Wang0d2c6b7e12017-08-16 21:12:551359 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1360
1361 AssertAsyncReadEquals(kMsg3, kLen3);
1362
1363 ASSERT_FALSE(write_callback_.have_result());
1364
1365 // Now the write will complete
1366 ResumeAndRun();
Yixin Wangdbbd8752018-01-17 21:50:021367 EXPECT_EQ(kLen2, write_callback_.WaitForResult());
Yixin Wang0d2c6b7e12017-08-16 21:12:551368}
1369
1370// ----------- Reading/Writing on Closed socket
1371
1372// Reading from an already closed socket should return 0
1373TEST_P(QuicProxyClientSocketTest, ReadOnClosedSocketReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431374 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1375 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1376 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551377 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1378
1379 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1380
1381 Initialize();
1382
1383 AssertConnectSucceeds();
1384
1385 ResumeAndRun();
1386
1387 ASSERT_FALSE(sock_->IsConnected());
Raul Tambre94493c652019-03-11 17:18:351388 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1389 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1390 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551391 ASSERT_FALSE(sock_->IsConnectedAndIdle());
1392}
1393
1394// Read pending when socket is closed should return 0
1395TEST_P(QuicProxyClientSocketTest, PendingReadOnCloseReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431396 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1397 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1398 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551399 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1400
1401 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1402
1403 Initialize();
1404
1405 AssertConnectSucceeds();
1406
1407 AssertReadStarts(kMsg1, kLen1);
1408
1409 ResumeAndRun();
1410
1411 ASSERT_EQ(0, read_callback_.WaitForResult());
1412}
1413
1414// Reading from a disconnected socket is an error
1415TEST_P(QuicProxyClientSocketTest, ReadOnDisconnectSocketReturnsNotConnected) {
Zhongyi Shi32f2fd02018-04-16 18:23:431416 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1417 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1418 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551419 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1420 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521421 SYNCHRONOUS,
1422 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551423
1424 Initialize();
1425
1426 AssertConnectSucceeds();
1427
1428 sock_->Disconnect();
1429
1430 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581431 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551432}
1433
1434// Reading data after receiving FIN should return buffered data received before
1435// FIN, then 0.
1436TEST_P(QuicProxyClientSocketTest, ReadAfterFinReceivedReturnsBufferedData) {
Zhongyi Shi32f2fd02018-04-16 18:23:431437 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1438 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1439 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551440 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1441
Victor Vasiliev076657c2019-03-12 02:46:431442 std::string header = ConstructDataHeader(kLen1);
1443 mock_quic_data_.AddRead(ASYNC, ConstructServerDataFinPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171444 2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431445 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551446 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411447 mock_quic_data_.AddWrite(SYNCHRONOUS,
1448 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551449
1450 Initialize();
1451
1452 AssertConnectSucceeds();
1453
1454 ResumeAndRun();
1455
1456 AssertSyncReadEquals(kMsg1, kLen1);
Raul Tambre94493c652019-03-11 17:18:351457 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1458 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551459
1460 sock_->Disconnect();
1461 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581462 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551463}
1464
1465// Calling Write() on a closed socket is an error.
1466TEST_P(QuicProxyClientSocketTest, WriteOnClosedStream) {
Zhongyi Shi32f2fd02018-04-16 18:23:431467 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1468 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1469 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551470 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1471
1472 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1473
1474 Initialize();
1475
1476 AssertConnectSucceeds();
1477
1478 ResumeAndRun();
1479
1480 AssertWriteReturns(kMsg1, kLen1, ERR_QUIC_PROTOCOL_ERROR);
1481}
1482
1483// Calling Write() on a disconnected socket is an error.
1484TEST_P(QuicProxyClientSocketTest, WriteOnDisconnectedSocket) {
Zhongyi Shi32f2fd02018-04-16 18:23:431485 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1486 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1487 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551488 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1489 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521490 SYNCHRONOUS,
1491 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551492
1493 Initialize();
1494
1495 AssertConnectSucceeds();
1496
1497 sock_->Disconnect();
1498
1499 AssertWriteReturns(kMsg1, kLen1, ERR_SOCKET_NOT_CONNECTED);
1500}
1501
1502// If the socket is closed with a pending Write(), the callback should be called
1503// with the same error the session was closed with.
1504TEST_P(QuicProxyClientSocketTest, WritePendingOnClose) {
Zhongyi Shi32f2fd02018-04-16 18:23:431505 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1506 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1507 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551508 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1509 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1510
1511 Initialize();
1512
1513 AssertConnectSucceeds();
1514
1515 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1516 // asynchronous starting with the second time it's called while the UDP socket
1517 // is write-blocked. Therefore, at least two writes need to be called on
1518 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351519 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551520
1521 // This second write will be async. This is the pending write that's being
1522 // tested.
1523 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1524
1525 // Make sure the write actually starts.
1526 base::RunLoop().RunUntilIdle();
1527
Ryan Hamilton8d9ee76e2018-05-29 23:52:521528 session_->CloseSessionOnError(ERR_CONNECTION_CLOSED,
Renjieba55fae2018-09-20 03:05:161529 quic::QUIC_INTERNAL_ERROR,
1530 quic::ConnectionCloseBehavior::SILENT_CLOSE);
Yixin Wang0d2c6b7e12017-08-16 21:12:551531
1532 EXPECT_THAT(write_callback_.WaitForResult(), IsError(ERR_CONNECTION_CLOSED));
1533}
1534
1535TEST_P(QuicProxyClientSocketTest, DisconnectWithWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431536 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1537 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1538 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551539 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1540 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1541
1542 Initialize();
1543
1544 AssertConnectSucceeds();
1545
1546 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1547 // asynchronous starting with the second time it's called while the UDP socket
1548 // is write-blocked. Therefore, at least two writes need to be called on
1549 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351550 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551551
1552 // This second write will be async. This is the pending write that's being
1553 // tested.
1554 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1555
1556 // Make sure the write actually starts.
1557 base::RunLoop().RunUntilIdle();
1558
1559 sock_->Disconnect();
1560 EXPECT_FALSE(sock_->IsConnected());
1561
1562 base::RunLoop().RunUntilIdle();
1563
1564 EXPECT_FALSE(sock_->IsConnected());
1565 EXPECT_FALSE(write_callback_.have_result());
1566}
1567
1568// If the socket is Disconnected with a pending Read(), the callback
1569// should not be called.
1570TEST_P(QuicProxyClientSocketTest, DisconnectWithReadPending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431571 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1572 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1573 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551574 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1575 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521576 SYNCHRONOUS,
1577 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551578
1579 Initialize();
1580
1581 AssertConnectSucceeds();
1582
1583 EXPECT_TRUE(sock_->IsConnected());
1584
1585 AssertReadStarts(kMsg1, kLen1);
1586
1587 sock_->Disconnect();
1588 EXPECT_FALSE(sock_->IsConnected());
1589
1590 base::RunLoop().RunUntilIdle();
1591
1592 EXPECT_FALSE(sock_->IsConnected());
1593 EXPECT_FALSE(read_callback_.have_result());
1594}
1595
1596// If the socket is Reset when both a read and write are pending,
1597// both should be called back.
1598TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431599 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1600 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1601 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551602 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1603
1604 mock_quic_data_.AddRead(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411605 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED));
Yixin Wang2bea3cf2017-11-09 18:11:031606 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:431607 std::string header = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561608 if (version_.transport_version != quic::QUIC_VERSION_99) {
Ryan Hamilton7505eb92019-06-08 00:22:171609 mock_quic_data_.AddWrite(ASYNC, ConstructAckAndDataPacket(
1610 3, 1, 1, 1, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411611 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411612 SYNCHRONOUS,
1613 ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411614 } else {
Renjief49758b2019-01-11 23:32:411615 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351616 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171617 3, 1, 1, 1, {header, std::string(kMsg2, kLen2)}));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411618 mock_quic_data_.AddWrite(
1619 SYNCHRONOUS,
1620 ConstructAckAndRstOnlyPacket(4, quic::QUIC_STREAM_CANCELLED, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411621 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551622
1623 Initialize();
1624
1625 AssertConnectSucceeds();
1626
1627 EXPECT_TRUE(sock_->IsConnected());
1628
1629 AssertReadStarts(kMsg1, kLen1);
1630
1631 // Write should block until the next read completes.
1632 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1633 // asynchronous starting with the second time it's called while the UDP socket
1634 // is write-blocked. Therefore, at least two writes need to be called on
1635 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351636 AssertWriteReturns(kMsg2, kLen2, kLen2);
Renjief49758b2019-01-11 23:32:411637
Yixin Wang0d2c6b7e12017-08-16 21:12:551638 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1639
1640 ResumeAndRun();
1641
1642 EXPECT_TRUE(read_callback_.have_result());
1643 EXPECT_TRUE(write_callback_.have_result());
1644}
1645
1646// Makes sure the proxy client socket's source gets the expected NetLog events
1647// and only the expected NetLog events (No SpdySession events).
1648TEST_P(QuicProxyClientSocketTest, NetLog) {
Zhongyi Shi32f2fd02018-04-16 18:23:431649 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1650 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1651 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551652 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1653
Victor Vasiliev076657c2019-03-12 02:46:431654 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171655 mock_quic_data_.AddRead(
1656 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431657 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang2bea3cf2017-11-09 18:11:031658 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411659 mock_quic_data_.AddWrite(SYNCHRONOUS,
1660 ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551661
1662 Initialize();
1663
1664 AssertConnectSucceeds();
1665
1666 ResumeAndRun();
1667 AssertSyncReadEquals(kMsg1, kLen1);
1668
1669 NetLogSource sock_source = sock_->NetLog().source();
1670 sock_.reset();
1671
Eric Roman79cc7552019-07-19 02:17:541672 auto entry_list = net_log_.GetEntriesForSource(sock_source);
Yixin Wang0d2c6b7e12017-08-16 21:12:551673
1674 ASSERT_EQ(entry_list.size(), 10u);
1675 EXPECT_TRUE(
1676 LogContainsBeginEvent(entry_list, 0, NetLogEventType::SOCKET_ALIVE));
1677 EXPECT_TRUE(LogContainsEvent(entry_list, 1,
1678 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
1679 NetLogEventPhase::NONE));
1680 EXPECT_TRUE(LogContainsBeginEvent(
1681 entry_list, 2, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1682 EXPECT_TRUE(LogContainsEvent(
1683 entry_list, 3, NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
1684 NetLogEventPhase::NONE));
1685 EXPECT_TRUE(LogContainsEndEvent(
1686 entry_list, 4, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1687 EXPECT_TRUE(LogContainsBeginEvent(
1688 entry_list, 5, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1689 EXPECT_TRUE(LogContainsEvent(
1690 entry_list, 6,
1691 NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
1692 NetLogEventPhase::NONE));
1693 EXPECT_TRUE(LogContainsEndEvent(
1694 entry_list, 7, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1695 EXPECT_TRUE(LogContainsEvent(entry_list, 8,
1696 NetLogEventType::SOCKET_BYTES_RECEIVED,
1697 NetLogEventPhase::NONE));
1698 EXPECT_TRUE(
1699 LogContainsEndEvent(entry_list, 9, NetLogEventType::SOCKET_ALIVE));
1700}
1701
Bence Béky8ddc2492018-06-13 01:02:041702// A helper class that will delete |sock| when the callback is invoked.
Yixin Wang0d2c6b7e12017-08-16 21:12:551703class DeleteSockCallback : public TestCompletionCallbackBase {
1704 public:
1705 explicit DeleteSockCallback(std::unique_ptr<QuicProxyClientSocket>* sock)
Bence Béky8ddc2492018-06-13 01:02:041706 : sock_(sock) {}
Yixin Wang0d2c6b7e12017-08-16 21:12:551707
1708 ~DeleteSockCallback() override {}
1709
Bence Béky8ddc2492018-06-13 01:02:041710 CompletionOnceCallback callback() {
1711 return base::BindOnce(&DeleteSockCallback::OnComplete,
1712 base::Unretained(this));
1713 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551714
1715 private:
1716 void OnComplete(int result) {
Raul Tambre94493c652019-03-11 17:18:351717 sock_->reset(nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:551718 SetResult(result);
1719 }
1720
1721 std::unique_ptr<QuicProxyClientSocket>* sock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:551722
1723 DISALLOW_COPY_AND_ASSIGN(DeleteSockCallback);
1724};
1725
1726// If the socket is reset when both a read and write are pending, and the
1727// read callback causes the socket to be deleted, the write callback should
1728// not be called.
1729TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePendingDelete) {
Zhongyi Shi32f2fd02018-04-16 18:23:431730 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1731 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1732 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551733 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1734
1735 mock_quic_data_.AddRead(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411736 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551737 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:561738 if (version_.transport_version != quic::QUIC_VERSION_99) {
Ryan Hamilton7505eb92019-06-08 00:22:171739 mock_quic_data_.AddWrite(ASYNC, ConstructAckAndDataPacket(
1740 3, 1, 1, 1, std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411741 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411742 SYNCHRONOUS,
1743 ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411744 } else {
Victor Vasiliev076657c2019-03-12 02:46:431745 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411746 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351747 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171748 3, 1, 1, 1, {header, std::string(kMsg1, kLen1)}));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411749 mock_quic_data_.AddWrite(
1750 SYNCHRONOUS,
1751 ConstructAckAndRstOnlyPacket(4, quic::QUIC_STREAM_CANCELLED, 2, 2, 1));
Renjief49758b2019-01-11 23:32:411752 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551753
1754 Initialize();
1755
1756 AssertConnectSucceeds();
1757
1758 EXPECT_TRUE(sock_->IsConnected());
1759
1760 DeleteSockCallback read_callback(&sock_);
Victor Costan9c7302b2018-08-27 16:39:441761 scoped_refptr<IOBuffer> read_buf = base::MakeRefCounted<IOBuffer>(kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551762 ASSERT_EQ(ERR_IO_PENDING,
1763 sock_->Read(read_buf.get(), kLen1, read_callback.callback()));
1764
1765 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1766 // asynchronous starting with the second time it's called while the UDP socket
1767 // is write-blocked. Therefore, at least two writes need to be called on
1768 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351769 AssertWriteReturns(kMsg1, kLen1, kLen1);
Renjief49758b2019-01-11 23:32:411770
Yixin Wang0d2c6b7e12017-08-16 21:12:551771 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1772
1773 ResumeAndRun();
1774
1775 EXPECT_FALSE(sock_.get());
1776
1777 EXPECT_EQ(0, read_callback.WaitForResult());
1778 EXPECT_FALSE(write_callback_.have_result());
1779}
1780
Victor Costane635086f2019-01-27 05:20:301781INSTANTIATE_TEST_SUITE_P(
Bence Békyce380cb2018-04-26 23:39:551782 VersionIncludeStreamDependencySequence,
Yixin Wang079ad542018-01-11 04:06:051783 QuicProxyClientSocketTest,
Victor Vasiliev5d6cdc22019-05-28 20:37:431784 ::testing::Combine(::testing::ValuesIn(quic::AllVersionsExcept99()),
Nick Harper23290b82019-05-02 00:02:561785 ::testing::Bool()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551786
1787} // namespace test
Dan Zhangf11470172017-09-18 22:02:091788} // namespace net