blob: d0d1ffb81ee04889eca3f831c574f77a0e5642f0 [file] [log] [blame]
Yixin Wang0d2c6b7e12017-08-16 21:12:551// Copyright (c) 2017 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Ryan Hamiltona3ee93a72018-08-01 22:03:085#include "net/quic/quic_proxy_client_socket.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:556
Bence Béky8f9d7d3952017-10-09 19:58:047#include <memory>
Eric Orthbe2efac2019-03-06 01:11:118#include <tuple>
Bence Békyd8a21fc32018-06-27 18:29:589#include <utility>
Eric Orthbe2efac2019-03-06 01:11:1110#include <vector>
Bence Béky8f9d7d3952017-10-09 19:58:0411
Sebastien Marchand6d0558fd2019-01-25 16:49:3712#include "base/bind.h"
Bence Béky8f9d7d3952017-10-09 19:58:0413#include "base/memory/ptr_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5514#include "base/run_loop.h"
15#include "base/strings/utf_string_conversions.h"
Lei Zhang52637ed2019-02-20 01:38:3716#include "base/threading/thread_task_runner_handle.h"
Zhongyi Shic16b4102019-02-12 00:37:4017#include "base/time/default_tick_clock.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5518#include "net/dns/mock_host_resolver.h"
19#include "net/http/http_auth_cache.h"
20#include "net/http/http_auth_handler_factory.h"
21#include "net/http/http_response_headers.h"
22#include "net/http/transport_security_state.h"
23#include "net/log/test_net_log.h"
24#include "net/log/test_net_log_util.h"
Victor Vasiliev4f6fb892019-05-31 16:58:3125#include "net/quic/address_utils.h"
Ryan Hamiltona3ee93a72018-08-01 22:03:0826#include "net/quic/crypto/proof_verifier_chromium.h"
27#include "net/quic/mock_crypto_client_stream_factory.h"
28#include "net/quic/mock_quic_data.h"
29#include "net/quic/quic_chromium_alarm_factory.h"
30#include "net/quic/quic_chromium_client_session.h"
31#include "net/quic/quic_chromium_connection_helper.h"
32#include "net/quic/quic_chromium_packet_writer.h"
33#include "net/quic/quic_http_utils.h"
34#include "net/quic/quic_server_info.h"
35#include "net/quic/quic_stream_factory.h"
36#include "net/quic/quic_test_packet_maker.h"
37#include "net/quic/test_task_runner.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5538#include "net/socket/socket_test_util.h"
39#include "net/test/cert_test_util.h"
Victor Vasilievdcdb6192019-02-25 19:49:5640#include "net/test/gtest_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5541#include "net/test/test_data_directory.h"
Bence Béky98447b12018-05-08 03:14:0142#include "net/test/test_with_scoped_task_environment.h"
Victor Vasiliev6bb59d22019-03-08 21:34:5143#include "net/third_party/quiche/src/quic/core/crypto/null_encrypter.h"
44#include "net/third_party/quiche/src/quic/core/quic_utils.h"
45#include "net/third_party/quiche/src/quic/core/tls_client_handshaker.h"
46#include "net/third_party/quiche/src/quic/test_tools/crypto_test_utils.h"
47#include "net/third_party/quiche/src/quic/test_tools/mock_clock.h"
48#include "net/third_party/quiche/src/quic/test_tools/mock_random.h"
49#include "net/third_party/quiche/src/quic/test_tools/quic_connection_peer.h"
50#include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h"
[email protected]578968d42017-12-13 15:39:3251#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5552#include "testing/gmock/include/gmock/gmock.h"
53#include "testing/gtest/include/gtest/gtest.h"
54
55using testing::_;
56using testing::AnyNumber;
57using testing::Return;
58
59namespace {
60
61static const char kOriginHost[] = "www.google.com";
62static const int kOriginPort = 443;
63static const char kProxyUrl[] = "https://myproxy:6121/";
64static const char kProxyHost[] = "myproxy";
65static const int kProxyPort = 6121;
66static const char kUserAgent[] = "Mozilla/1.0";
67static const char kRedirectUrl[] = "https://example.com/";
68
69static const char kMsg1[] = "\0hello!\xff";
70static const int kLen1 = 8;
71static const char kMsg2[] = "\0a2345678\0";
72static const int kLen2 = 10;
73static const char kMsg3[] = "bye!";
74static const int kLen3 = 4;
75static const char kMsg33[] = "bye!bye!";
76static const int kLen33 = kLen3 + kLen3;
77static const char kMsg333[] = "bye!bye!bye!";
78static const int kLen333 = kLen3 + kLen3 + kLen3;
79
80} // anonymous namespace
81
82namespace net {
83namespace test {
84
Michael Warres74ee3ce2017-10-09 15:26:3785class QuicProxyClientSocketTest
Ryan Hamilton8d9ee76e2018-05-29 23:52:5286 : public ::testing::TestWithParam<
Nick Harper23290b82019-05-02 00:02:5687 std::tuple<quic::ParsedQuicVersion, bool>>,
Bence Béky98447b12018-05-08 03:14:0188 public WithScopedTaskEnvironment {
Yixin Wang0d2c6b7e12017-08-16 21:12:5589 protected:
90 static const bool kFin = true;
91 static const bool kIncludeVersion = true;
92 static const bool kIncludeDiversificationNonce = true;
93 static const bool kIncludeCongestionFeedback = true;
94 static const bool kSendFeedback = true;
95
96 static size_t GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:5297 quic::QuicByteCount packet_length,
Nick Harper23290b82019-05-02 00:02:5698 quic::ParsedQuicVersion version,
Yixin Wang0d2c6b7e12017-08-16 21:12:5599 bool include_version,
100 bool include_diversification_nonce,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52101 quic::QuicConnectionIdLength connection_id_length,
102 quic::QuicPacketNumberLength packet_number_length,
103 quic::QuicStreamOffset offset) {
Michael Warres167db3e2019-03-01 21:38:03104 quic::QuicVariableLengthIntegerLength retry_token_length_length =
105 quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
106 quic::QuicVariableLengthIntegerLength length_length =
Nick Harper23290b82019-05-02 00:02:56107 quic::QuicVersionHasLongHeaderLengths(version.transport_version) &&
108 include_version
Michael Warres167db3e2019-03-01 21:38:03109 ? quic::VARIABLE_LENGTH_INTEGER_LENGTH_2
110 : quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
Yixin Wang0d2c6b7e12017-08-16 21:12:55111 size_t min_data_length = 1;
112 size_t min_packet_length =
Ryan Hamilton8d9ee76e2018-05-29 23:52:52113 quic::NullEncrypter(quic::Perspective::IS_CLIENT)
Yixin Wang0d2c6b7e12017-08-16 21:12:55114 .GetCiphertextSize(min_data_length) +
Ryan Hamilton8d9ee76e2018-05-29 23:52:52115 quic::QuicPacketCreator::StreamFramePacketOverhead(
Nick Harper23290b82019-05-02 00:02:56116 version.transport_version, quic::PACKET_8BYTE_CONNECTION_ID,
Michael Warres60637ae2018-06-05 20:03:11117 quic::PACKET_0BYTE_CONNECTION_ID, include_version,
Michael Warres167db3e2019-03-01 21:38:03118 include_diversification_nonce, packet_number_length,
119 retry_token_length_length, length_length, offset);
Yixin Wang0d2c6b7e12017-08-16 21:12:55120
121 DCHECK(packet_length >= min_packet_length);
122 return min_data_length + packet_length - min_packet_length;
123 }
124
125 QuicProxyClientSocketTest()
Yixin Wang079ad542018-01-11 04:06:05126 : version_(std::get<0>(GetParam())),
Nick Harper23290b82019-05-02 00:02:56127 client_data_stream_id1_(
128 quic::QuicUtils::GetHeadersStreamId(version_.transport_version) +
129 quic::QuicUtils::StreamIdDelta(version_.transport_version)),
Yixin Wang079ad542018-01-11 04:06:05130 client_headers_include_h2_stream_dependency_(std::get<1>(GetParam())),
Ryan Hamilton8d9ee76e2018-05-29 23:52:52131 crypto_config_(quic::test::crypto_test_utils::ProofVerifierForTesting(),
132 quic::TlsClientHandshaker::CreateSslCtx()),
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),
147 header_stream_offset_(0),
148 response_offset_(0),
149 user_agent_(kUserAgent),
150 proxy_host_port_(kProxyHost, kProxyPort),
151 endpoint_host_port_(kOriginHost, kOriginPort),
152 host_resolver_(new MockCachingHostResolver()),
Eric Orthbe2efac2019-03-06 01:11:11153 http_auth_handler_factory_(HttpAuthHandlerFactory::CreateDefault()) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55154 IPAddress ip(192, 0, 2, 33);
155 peer_addr_ = IPEndPoint(ip, 443);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52156 clock_.AdvanceTime(quic::QuicTime::Delta::FromMilliseconds(20));
Yixin Wang0d2c6b7e12017-08-16 21:12:55157 }
158
159 void SetUp() override {}
160
Yixin Wang2bea3cf2017-11-09 18:11:03161 void TearDown() override {
162 sock_.reset();
163 EXPECT_TRUE(mock_quic_data_.AllReadDataConsumed());
164 EXPECT_TRUE(mock_quic_data_.AllWriteDataConsumed());
165 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55166
167 void Initialize() {
168 std::unique_ptr<MockUDPClientSocket> socket(new MockUDPClientSocket(
169 mock_quic_data_.InitializeAndGetSequencedSocketData(),
170 net_log_.bound().net_log()));
171 socket->Connect(peer_addr_);
172 runner_ = new TestTaskRunner(&clock_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52173 send_algorithm_ = new quic::test::MockSendAlgorithm();
Yixin Wang0d2c6b7e12017-08-16 21:12:55174 EXPECT_CALL(*send_algorithm_, InRecovery()).WillRepeatedly(Return(false));
175 EXPECT_CALL(*send_algorithm_, InSlowStart()).WillRepeatedly(Return(false));
176 EXPECT_CALL(*send_algorithm_, OnPacketSent(_, _, _, _, _))
Dan Zhangf11470172017-09-18 22:02:09177 .Times(testing::AtLeast(1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55178 EXPECT_CALL(*send_algorithm_, GetCongestionWindow())
Zhongyi Shica576df2019-04-12 17:43:40179 .WillRepeatedly(Return(quic::kMaxOutgoingPacketSize));
Yixin Wang0d2c6b7e12017-08-16 21:12:55180 EXPECT_CALL(*send_algorithm_, PacingRate(_))
Ryan Hamilton8d9ee76e2018-05-29 23:52:52181 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Michael Warres74ee3ce2017-10-09 15:26:37182 EXPECT_CALL(*send_algorithm_, CanSend(_)).WillRepeatedly(Return(true));
Yixin Wang0d2c6b7e12017-08-16 21:12:55183 EXPECT_CALL(*send_algorithm_, BandwidthEstimate())
Ryan Hamilton8d9ee76e2018-05-29 23:52:52184 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Yixin Wang0d2c6b7e12017-08-16 21:12:55185 EXPECT_CALL(*send_algorithm_, SetFromConfig(_, _)).Times(AnyNumber());
186 EXPECT_CALL(*send_algorithm_, OnApplicationLimited(_)).Times(AnyNumber());
187 EXPECT_CALL(*send_algorithm_, GetCongestionControlType())
188 .Times(AnyNumber());
189 helper_.reset(
190 new QuicChromiumConnectionHelper(&clock_, &random_generator_));
191 alarm_factory_.reset(new QuicChromiumAlarmFactory(runner_.get(), &clock_));
192
Ryan Hamilton9edcf1a2017-11-22 05:55:17193 QuicChromiumPacketWriter* writer = new QuicChromiumPacketWriter(
194 socket.get(), base::ThreadTaskRunnerHandle::Get().get());
Ryan Hamilton8d9ee76e2018-05-29 23:52:52195 quic::QuicConnection* connection = new quic::QuicConnection(
Victor Vasiliev4f6fb892019-05-31 16:58:31196 connection_id_, net::ToQuicSocketAddress(peer_addr_), helper_.get(),
197 alarm_factory_.get(), writer, true /* owns_writer */,
Nick Harper23290b82019-05-02 00:02:56198 quic::Perspective::IS_CLIENT, quic::test::SupportedVersions(version_));
Yixin Wang0d2c6b7e12017-08-16 21:12:55199 connection->set_visitor(&visitor_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52200 quic::test::QuicConnectionPeer::SetSendAlgorithm(connection,
201 send_algorithm_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55202
203 // Load a certificate that is valid for *.example.org
204 scoped_refptr<X509Certificate> test_cert(
205 ImportCertFromFile(GetTestCertsDirectory(), "wildcard.pem"));
206 EXPECT_TRUE(test_cert.get());
207
208 verify_details_.cert_verify_result.verified_cert = test_cert;
209 verify_details_.cert_verify_result.is_issued_by_known_root = true;
210 crypto_client_stream_factory_.AddProofVerifyDetails(&verify_details_);
211
212 base::TimeTicks dns_end = base::TimeTicks::Now();
213 base::TimeTicks dns_start = dns_end - base::TimeDelta::FromMilliseconds(1);
214
215 session_.reset(new QuicChromiumClientSession(
216 connection, std::move(socket),
217 /*stream_factory=*/nullptr, &crypto_client_stream_factory_, &clock_,
Nick Harper89bc7212018-07-31 19:07:57218 &transport_security_state_, /*ssl_config_service=*/nullptr,
Yixin Wang0d2c6b7e12017-08-16 21:12:55219 base::WrapUnique(static_cast<QuicServerInfo*>(nullptr)),
Paul Jensen8e3c5d32018-02-19 17:06:33220 QuicSessionKey("mail.example.org", 80, PRIVACY_MODE_DISABLED,
221 SocketTag()),
Zhongyi Shi757fcce2018-06-27 05:41:27222 /*require_confirmation=*/false, /*migrate_session_early_v2=*/false,
223 /*migrate_session_on_network_change_v2=*/false,
224 /*default_network=*/NetworkChangeNotifier::kInvalidNetworkHandle,
Zhongyi Shie01f2db2019-02-22 19:53:23225 quic::QuicTime::Delta::FromMilliseconds(
226 kDefaultRetransmittableOnWireTimeoutMillisecs),
Zhongyi Shi32fe14d42019-02-28 00:25:36227 /*migrate_idle_session=*/true,
Zhongyi Shic16b4102019-02-12 00:37:40228 base::TimeDelta::FromSeconds(kDefaultIdleSessionMigrationPeriodSeconds),
Zhongyi Shi73f23ca872017-12-13 18:37:13229 base::TimeDelta::FromSeconds(kMaxTimeOnNonDefaultNetworkSecs),
Zhongyi Shiee760762018-08-01 00:54:29230 kMaxMigrationsToNonDefaultNetworkOnWriteError,
Zhongyi Shi8b1e43f2017-12-13 20:46:30231 kMaxMigrationsToNonDefaultNetworkOnPathDegrading,
Zhongyi Shi5f587cc2017-11-21 23:24:17232 kQuicYieldAfterPacketsRead,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52233 quic::QuicTime::Delta::FromMilliseconds(
234 kQuicYieldAfterDurationMilliseconds),
Zhongyi Shidbce7f412019-02-01 23:16:29235 /*go_away_on_path_degrading*/ false,
Yixin Wang079ad542018-01-11 04:06:05236 client_headers_include_h2_stream_dependency_, /*cert_verify_flags=*/0,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52237 quic::test::DefaultQuicConfig(), &crypto_config_, "CONNECTION_UNKNOWN",
238 dns_start, dns_end, &push_promise_index_, nullptr,
Zhongyi Shic16b4102019-02-12 00:37:40239 base::DefaultTickClock::GetInstance(),
Yixin Wang0d2c6b7e12017-08-16 21:12:55240 base::ThreadTaskRunnerHandle::Get().get(),
241 /*socket_performance_watcher=*/nullptr, net_log_.bound().net_log()));
242
243 writer->set_delegate(session_.get());
244
Ryan Hamilton6c2a2a82017-12-15 02:06:28245 session_handle_ =
246 session_->CreateHandle(HostPortPair("mail.example.org", 80));
Yixin Wang0d2c6b7e12017-08-16 21:12:55247
248 session_->Initialize();
249 TestCompletionCallback callback;
250 EXPECT_THAT(session_->CryptoConnect(callback.callback()), IsOk());
251 EXPECT_TRUE(session_->IsCryptoHandshakeConfirmed());
252
Ramin Halavati683bcaa92018-02-14 08:42:39253 EXPECT_THAT(session_handle_->RequestStream(true, callback.callback(),
254 TRAFFIC_ANNOTATION_FOR_TESTS),
Yixin Wang0d2c6b7e12017-08-16 21:12:55255 IsOk());
256 std::unique_ptr<QuicChromiumClientStream::Handle> stream_handle =
257 session_handle_->ReleaseStream();
258 EXPECT_TRUE(stream_handle->IsOpen());
259
260 sock_.reset(new QuicProxyClientSocket(
261 std::move(stream_handle), std::move(session_handle_), user_agent_,
262 endpoint_host_port_, net_log_.bound(),
Eric Orthbe2efac2019-03-06 01:11:11263 new HttpAuthController(
264 HttpAuth::AUTH_PROXY,
265 GURL("https://" + proxy_host_port_.ToString()), &http_auth_cache_,
266 http_auth_handler_factory_.get(), host_resolver_.get())));
Yixin Wang0d2c6b7e12017-08-16 21:12:55267
268 session_->StartReading();
269 }
270
Ryan Hamilton0239aac2018-05-19 00:03:13271 void PopulateConnectRequestIR(spdy::SpdyHeaderBlock* block) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55272 (*block)[":method"] = "CONNECT";
273 (*block)[":authority"] = endpoint_host_port_.ToString();
274 (*block)["user-agent"] = kUserAgent;
275 }
276
277 // Helper functions for constructing packets sent by the client
278
Ryan Hamilton8d9ee76e2018-05-29 23:52:52279 std::unique_ptr<quic::QuicReceivedPacket> ConstructSettingsPacket(
Fan Yangac867502019-01-28 21:10:23280 uint64_t packet_number) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55281 return client_maker_.MakeInitialSettingsPacket(packet_number,
282 &header_stream_offset_);
283 }
284
Ryan Hamilton8d9ee76e2018-05-29 23:52:52285 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23286 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52287 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23288 uint64_t largest_received,
289 uint64_t smallest_received,
290 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55291 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09292 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55293 largest_received, smallest_received, least_unacked, kSendFeedback);
294 }
295
Frank Kastenholz684ea412019-02-13 18:48:18296 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstOnlyPacket(
297 uint64_t packet_number,
298 quic::QuicRstStreamErrorCode error_code,
299 uint64_t largest_received,
300 uint64_t smallest_received,
301 uint64_t least_unacked,
302 size_t bytes_written) {
303 return client_maker_.MakeAckAndRstPacket(
304 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
305 largest_received, smallest_received, least_unacked, kSendFeedback,
306 bytes_written,
307 /*include_stop_sending=*/false);
308 }
309
Ryan Hamilton8d9ee76e2018-05-29 23:52:52310 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23311 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52312 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23313 uint64_t largest_received,
314 uint64_t smallest_received,
315 uint64_t least_unacked,
Yixin Wang0d2c6b7e12017-08-16 21:12:55316 size_t bytes_written) {
317 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09318 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55319 largest_received, smallest_received, least_unacked, kSendFeedback,
Frank Kastenholz684ea412019-02-13 18:48:18320 bytes_written,
321 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55322 }
323
Ryan Hamilton8d9ee76e2018-05-29 23:52:52324 std::unique_ptr<quic::QuicReceivedPacket> ConstructRstPacket(
Fan Yangac867502019-01-28 21:10:23325 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52326 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55327 size_t bytes_written) {
328 return client_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09329 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18330 bytes_written,
331 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55332 }
333
Ryan Hamilton8d9ee76e2018-05-29 23:52:52334 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectRequestPacket(
Fan Yangac867502019-01-28 21:10:23335 uint64_t packet_number,
Lily Chenf11e1292018-11-29 16:42:09336 RequestPriority request_priority = LOWEST) {
Ryan Hamilton0239aac2018-05-19 00:03:13337 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55338 PopulateConnectRequestIR(&block);
339 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09340 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Lily Chenf11e1292018-11-29 16:42:09341 ConvertRequestPriorityToQuicPriority(request_priority),
342 std::move(block), 0, nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55343 }
344
Ryan Hamilton8d9ee76e2018-05-29 23:52:52345 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectAuthRequestPacket(
Fan Yangac867502019-01-28 21:10:23346 uint64_t packet_number) {
Ryan Hamilton0239aac2018-05-19 00:03:13347 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55348 PopulateConnectRequestIR(&block);
349 block["proxy-authorization"] = "Basic Zm9vOmJhcg==";
350 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09351 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Yixin Wang7a3f1b8d2018-01-17 21:40:48352 ConvertRequestPriorityToQuicPriority(LOWEST), std::move(block), 0,
353 nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55354 }
355
Ryan Hamilton8d9ee76e2018-05-29 23:52:52356 std::unique_ptr<quic::QuicReceivedPacket> ConstructDataPacket(
Fan Yangac867502019-01-28 21:10:23357 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52358 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41359 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09360 return client_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41361 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55362 }
363
Renjied172e812019-01-16 05:12:35364 std::unique_ptr<quic::QuicReceivedPacket> ConstructMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23365 uint64_t packet_number,
Renjied172e812019-01-16 05:12:35366 quic::QuicStreamOffset offset,
367 const std::vector<std::string> data_writes) {
368 return client_maker_.MakeMultipleDataFramesPacket(
369 packet_number, client_data_stream_id1_, !kIncludeVersion, !kFin, offset,
370 data_writes);
371 }
372
Ryan Hamilton8d9ee76e2018-05-29 23:52:52373 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndDataPacket(
Fan Yangac867502019-01-28 21:10:23374 uint64_t packet_number,
375 uint64_t largest_received,
376 uint64_t smallest_received,
377 uint64_t least_unacked,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52378 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41379 quic::QuicStringPiece data) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55380 return client_maker_.MakeAckAndDataPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09381 packet_number, !kIncludeVersion, client_data_stream_id1_,
382 largest_received, smallest_received, least_unacked, !kFin, offset,
Renjief49758b2019-01-11 23:32:41383 data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55384 }
385
Renjied172e812019-01-16 05:12:35386 std::unique_ptr<quic::QuicReceivedPacket>
387 ConstructAckAndMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23388 uint64_t packet_number,
389 uint64_t largest_received,
390 uint64_t smallest_received,
391 uint64_t least_unacked,
Renjied172e812019-01-16 05:12:35392 quic::QuicStreamOffset offset,
393 const std::vector<std::string> data_writes) {
394 return client_maker_.MakeAckAndMultipleDataFramesPacket(
395 packet_number, !kIncludeVersion, client_data_stream_id1_,
396 largest_received, smallest_received, least_unacked, !kFin, offset,
397 data_writes);
398 }
399
Ryan Hamilton8d9ee76e2018-05-29 23:52:52400 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckPacket(
Fan Yangac867502019-01-28 21:10:23401 uint64_t packet_number,
402 uint64_t largest_received,
403 uint64_t smallest_received,
404 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55405 return client_maker_.MakeAckPacket(packet_number, largest_received,
406 smallest_received, least_unacked,
407 kSendFeedback);
408 }
409
410 // Helper functions for constructing packets sent by the server
411
Ryan Hamilton8d9ee76e2018-05-29 23:52:52412 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerRstPacket(
Fan Yangac867502019-01-28 21:10:23413 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52414 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55415 size_t bytes_written) {
416 return server_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09417 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18418 bytes_written,
419 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55420 }
421
Ryan Hamilton8d9ee76e2018-05-29 23:52:52422 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataPacket(
Fan Yangac867502019-01-28 21:10:23423 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52424 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41425 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09426 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41427 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55428 }
429
Ryan Hamilton8d9ee76e2018-05-29 23:52:52430 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataFinPacket(
Fan Yangac867502019-01-28 21:10:23431 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52432 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41433 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09434 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41435 !kIncludeVersion, kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55436 }
437
Ryan Hamilton8d9ee76e2018-05-29 23:52:52438 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerConnectReplyPacket(
Fan Yangac867502019-01-28 21:10:23439 uint64_t packet_number,
Yixin Wang0d2c6b7e12017-08-16 21:12:55440 bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13441 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55442 block[":status"] = "200";
443
444 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09445 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55446 std::move(block), nullptr, &response_offset_);
447 }
448
Ryan Hamilton8d9ee76e2018-05-29 23:52:52449 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23450 ConstructServerConnectAuthReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13451 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55452 block[":status"] = "407";
453 block["proxy-authenticate"] = "Basic realm=\"MyRealm1\"";
454 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09455 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55456 std::move(block), nullptr, &response_offset_);
457 }
458
Ryan Hamilton8d9ee76e2018-05-29 23:52:52459 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23460 ConstructServerConnectRedirectReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13461 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55462 block[":status"] = "302";
463 block["location"] = kRedirectUrl;
464 block["set-cookie"] = "foo=bar";
465 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09466 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55467 std::move(block), nullptr, &response_offset_);
468 }
469
Ryan Hamilton8d9ee76e2018-05-29 23:52:52470 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23471 ConstructServerConnectErrorReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13472 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55473 block[":status"] = "500";
474
475 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09476 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55477 std::move(block), nullptr, &response_offset_);
478 }
479
480 void AssertConnectSucceeds() {
481 TestCompletionCallback callback;
482 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
483 ASSERT_THAT(callback.WaitForResult(), IsOk());
484 }
485
486 void AssertConnectFails(int result) {
487 TestCompletionCallback callback;
488 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
489 ASSERT_EQ(result, callback.WaitForResult());
490 }
491
492 void ResumeAndRun() {
493 // Run until the pause, if the provider isn't paused yet.
494 SequencedSocketData* data = mock_quic_data_.GetSequencedSocketData();
495 data->RunUntilPaused();
496 data->Resume();
497 base::RunLoop().RunUntilIdle();
498 }
499
500 void AssertWriteReturns(const char* data, int len, int rv) {
Victor Costan9c7302b2018-08-27 16:39:44501 scoped_refptr<IOBufferWithSize> buf =
502 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55503 memcpy(buf->data(), data, len);
504 EXPECT_EQ(rv,
[email protected]578968d42017-12-13 15:39:32505 sock_->Write(buf.get(), buf->size(), write_callback_.callback(),
506 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55507 }
508
509 void AssertSyncWriteSucceeds(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44510 scoped_refptr<IOBufferWithSize> buf =
511 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55512 memcpy(buf->data(), data, len);
Bence Békyd8a21fc32018-06-27 18:29:58513 EXPECT_EQ(len,
514 sock_->Write(buf.get(), buf->size(), CompletionOnceCallback(),
515 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55516 }
517
518 void AssertSyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44519 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Bence Békyd8a21fc32018-06-27 18:29:58520 ASSERT_EQ(len, sock_->Read(buf.get(), len, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:13521 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55522 ASSERT_TRUE(sock_->IsConnected());
523 }
524
525 void AssertAsyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44526 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55527 ASSERT_EQ(ERR_IO_PENDING,
528 sock_->Read(buf.get(), len, read_callback_.callback()));
529 EXPECT_TRUE(sock_->IsConnected());
530
531 ResumeAndRun();
532
533 EXPECT_EQ(len, read_callback_.WaitForResult());
534 EXPECT_TRUE(sock_->IsConnected());
Ryan Hamilton0239aac2018-05-19 00:03:13535 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55536 }
537
538 void AssertReadStarts(const char* data, int len) {
539 // Issue the read, which will be completed asynchronously.
Victor Costan9c7302b2018-08-27 16:39:44540 read_buf_ = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55541 ASSERT_EQ(ERR_IO_PENDING,
542 sock_->Read(read_buf_.get(), len, read_callback_.callback()));
543 EXPECT_TRUE(sock_->IsConnected());
544 }
545
546 void AssertReadReturns(const char* data, int len) {
547 EXPECT_TRUE(sock_->IsConnected());
548
549 // Now the read will return.
550 EXPECT_EQ(len, read_callback_.WaitForResult());
Ryan Hamilton0239aac2018-05-19 00:03:13551 ASSERT_EQ(spdy::SpdyString(data, len),
552 spdy::SpdyString(read_buf_->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55553 }
554
Victor Vasiliev076657c2019-03-12 02:46:43555 std::string ConstructDataHeader(size_t body_len) {
Nick Harper23290b82019-05-02 00:02:56556 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41557 return "";
558 }
559 quic::HttpEncoder encoder;
560 std::unique_ptr<char[]> buffer;
561 auto header_length = encoder.SerializeDataFrameHeader(body_len, &buffer);
Victor Vasiliev076657c2019-03-12 02:46:43562 return std::string(buffer.get(), header_length);
Renjief49758b2019-01-11 23:32:41563 }
564
Nick Harper23290b82019-05-02 00:02:56565 const quic::ParsedQuicVersion version_;
Ryan Hamilton47cf9d12018-10-17 04:33:09566 const quic::QuicStreamId client_data_stream_id1_;
Yixin Wang079ad542018-01-11 04:06:05567 const bool client_headers_include_h2_stream_dependency_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55568
569 // order of destruction of these members matter
Ryan Hamilton8d9ee76e2018-05-29 23:52:52570 quic::MockClock clock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55571 MockQuicData mock_quic_data_;
572 std::unique_ptr<QuicChromiumConnectionHelper> helper_;
573 std::unique_ptr<QuicChromiumClientSession> session_;
574 std::unique_ptr<QuicChromiumClientSession::Handle> session_handle_;
575 std::unique_ptr<QuicProxyClientSocket> sock_;
576
577 BoundTestNetLog net_log_;
578
Ryan Hamilton8d9ee76e2018-05-29 23:52:52579 quic::test::MockSendAlgorithm* send_algorithm_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55580 scoped_refptr<TestTaskRunner> runner_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55581
582 std::unique_ptr<QuicChromiumAlarmFactory> alarm_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52583 testing::StrictMock<quic::test::MockQuicConnectionVisitor> visitor_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55584 TransportSecurityState transport_security_state_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52585 quic::QuicCryptoClientConfig crypto_config_;
586 quic::QuicClientPushPromiseIndex push_promise_index_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55587
Ryan Hamilton8d9ee76e2018-05-29 23:52:52588 const quic::QuicConnectionId connection_id_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55589 QuicTestPacketMaker client_maker_;
590 QuicTestPacketMaker server_maker_;
591 IPEndPoint peer_addr_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52592 quic::test::MockRandom random_generator_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55593 ProofVerifyDetailsChromium verify_details_;
594 MockCryptoClientStreamFactory crypto_client_stream_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52595 quic::QuicStreamOffset header_stream_offset_;
596 quic::QuicStreamOffset response_offset_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55597
598 std::string user_agent_;
599 HostPortPair proxy_host_port_;
600 HostPortPair endpoint_host_port_;
601 HttpAuthCache http_auth_cache_;
602 std::unique_ptr<MockHostResolverBase> host_resolver_;
603 std::unique_ptr<HttpAuthHandlerRegistryFactory> http_auth_handler_factory_;
604
605 TestCompletionCallback read_callback_;
606 scoped_refptr<IOBuffer> read_buf_;
607
608 TestCompletionCallback write_callback_;
609
610 DISALLOW_COPY_AND_ASSIGN(QuicProxyClientSocketTest);
611};
612
613TEST_P(QuicProxyClientSocketTest, ConnectSendsCorrectRequest) {
Zhongyi Shi32f2fd02018-04-16 18:23:43614 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
615 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
616 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55617 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
618 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52619 SYNCHRONOUS,
620 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55621
622 Initialize();
623
624 ASSERT_FALSE(sock_->IsConnected());
625
626 AssertConnectSucceeds();
627
628 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
629 ASSERT_TRUE(response != nullptr);
630 ASSERT_EQ(200, response->headers->response_code());
631}
632
633TEST_P(QuicProxyClientSocketTest, ConnectWithAuthRequested) {
Zhongyi Shi32f2fd02018-04-16 18:23:43634 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
635 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
636 mock_quic_data_.AddRead(ASYNC,
637 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55638 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
639 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52640 SYNCHRONOUS,
641 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55642
643 Initialize();
644
645 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
646
647 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
648 ASSERT_TRUE(response != nullptr);
649 ASSERT_EQ(407, response->headers->response_code());
650}
651
652TEST_P(QuicProxyClientSocketTest, ConnectWithAuthCredentials) {
Zhongyi Shi32f2fd02018-04-16 18:23:43653 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
654 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectAuthRequestPacket(2));
655 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(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
663 // Add auth to cache
664 const base::string16 kFoo(base::ASCIIToUTF16("foo"));
665 const base::string16 kBar(base::ASCIIToUTF16("bar"));
666 http_auth_cache_.Add(GURL(kProxyUrl), "MyRealm1", HttpAuth::AUTH_SCHEME_BASIC,
667 "Basic realm=MyRealm1", AuthCredentials(kFoo, kBar),
668 "/");
669
670 AssertConnectSucceeds();
671
672 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
673 ASSERT_TRUE(response != nullptr);
674 ASSERT_EQ(200, response->headers->response_code());
675}
676
Eric Roman96c5b292019-04-23 18:04:59677// Tests that a redirect response from a CONNECT fails.
Yixin Wang0d2c6b7e12017-08-16 21:12:55678TEST_P(QuicProxyClientSocketTest, ConnectRedirects) {
Zhongyi Shi32f2fd02018-04-16 18:23:43679 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
680 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
681 mock_quic_data_.AddRead(ASYNC,
682 ConstructServerConnectRedirectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55683 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
684 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52685 SYNCHRONOUS,
686 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55687
688 Initialize();
689
Eric Roman96c5b292019-04-23 18:04:59690 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
Yixin Wang0d2c6b7e12017-08-16 21:12:55691
692 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
693 ASSERT_TRUE(response != nullptr);
694
695 const HttpResponseHeaders* headers = response->headers.get();
696 ASSERT_EQ(302, headers->response_code());
Eric Roman96c5b292019-04-23 18:04:59697 ASSERT_TRUE(headers->HasHeader("set-cookie"));
Yixin Wang0d2c6b7e12017-08-16 21:12:55698
699 std::string location;
700 ASSERT_TRUE(headers->IsRedirect(&location));
701 ASSERT_EQ(location, kRedirectUrl);
702}
703
704TEST_P(QuicProxyClientSocketTest, ConnectFails) {
Zhongyi Shi32f2fd02018-04-16 18:23:43705 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
706 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Yixin Wang0d2c6b7e12017-08-16 21:12:55707 mock_quic_data_.AddRead(ASYNC, 0); // EOF
708
709 Initialize();
710
711 ASSERT_FALSE(sock_->IsConnected());
712
713 AssertConnectFails(ERR_QUIC_PROTOCOL_ERROR);
714
715 ASSERT_FALSE(sock_->IsConnected());
716}
717
718TEST_P(QuicProxyClientSocketTest, WasEverUsedReturnsCorrectValue) {
Zhongyi Shi32f2fd02018-04-16 18:23:43719 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
720 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
721 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55722 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
723 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52724 SYNCHRONOUS,
725 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55726
727 Initialize();
728
729 EXPECT_TRUE(sock_->WasEverUsed()); // Used due to crypto handshake
730 AssertConnectSucceeds();
731 EXPECT_TRUE(sock_->WasEverUsed());
732 sock_->Disconnect();
733 EXPECT_TRUE(sock_->WasEverUsed());
734}
735
736TEST_P(QuicProxyClientSocketTest, GetPeerAddressReturnsCorrectValues) {
Zhongyi Shi32f2fd02018-04-16 18:23:43737 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
738 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
739 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55740 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
741 mock_quic_data_.AddRead(ASYNC, 0); // EOF
742
743 Initialize();
744
745 IPEndPoint addr;
746 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
747
748 AssertConnectSucceeds();
749 EXPECT_TRUE(sock_->IsConnected());
750 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsOk());
751
752 ResumeAndRun();
753
754 EXPECT_FALSE(sock_->IsConnected());
755 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
756
757 sock_->Disconnect();
758
759 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
760}
761
762TEST_P(QuicProxyClientSocketTest, IsConnectedAndIdle) {
Zhongyi Shi32f2fd02018-04-16 18:23:43763 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
764 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
765 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55766 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
767
Victor Vasiliev076657c2019-03-12 02:46:43768 std::string header = ConstructDataHeader(kLen1);
769 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
770 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43771 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55772 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52773 mock_quic_data_.AddWrite(
774 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55775
776 Initialize();
777
778 EXPECT_FALSE(sock_->IsConnectedAndIdle());
779
780 AssertConnectSucceeds();
781
782 EXPECT_TRUE(sock_->IsConnectedAndIdle());
783
784 // The next read is consumed and buffered.
785 ResumeAndRun();
786
787 EXPECT_FALSE(sock_->IsConnectedAndIdle());
788
789 AssertSyncReadEquals(kMsg1, kLen1);
790
791 EXPECT_TRUE(sock_->IsConnectedAndIdle());
792}
793
794TEST_P(QuicProxyClientSocketTest, GetTotalReceivedBytes) {
Zhongyi Shi32f2fd02018-04-16 18:23:43795 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
796 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
797 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55798 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
799
Victor Vasiliev076657c2019-03-12 02:46:43800 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:41801 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:43802 ASYNC,
803 ConstructServerDataPacket(2, 0, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:43804 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55805 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52806 mock_quic_data_.AddWrite(
807 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55808
809 Initialize();
810
811 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
812
813 AssertConnectSucceeds();
814
815 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
816
817 // The next read is consumed and buffered.
818 ResumeAndRun();
819
820 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
821
822 // The payload from the single large data frame will be read across
823 // two different reads.
824 AssertSyncReadEquals(kMsg33, kLen33);
825
Renjief49758b2019-01-11 23:32:41826 EXPECT_EQ((int64_t)(kLen33 + header.length()),
827 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55828
829 AssertSyncReadEquals(kMsg3, kLen3);
830
Renjief49758b2019-01-11 23:32:41831 EXPECT_EQ((int64_t)(kLen333 + header.length()),
832 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55833}
834
Lily Chenf11e1292018-11-29 16:42:09835TEST_P(QuicProxyClientSocketTest, SetStreamPriority) {
836 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
Matt Menkeedaf3b82019-03-14 21:39:44837 // Despite setting the priority to HIGHEST, the requets initial priority of
838 // LOWEST is used.
Lily Chenf11e1292018-11-29 16:42:09839 mock_quic_data_.AddWrite(SYNCHRONOUS,
Matt Menkeedaf3b82019-03-14 21:39:44840 ConstructConnectRequestPacket(2, LOWEST));
Lily Chenf11e1292018-11-29 16:42:09841 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
842 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
843 mock_quic_data_.AddWrite(
844 SYNCHRONOUS,
845 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
846
847 Initialize();
848
849 sock_->SetStreamPriority(HIGHEST);
850 AssertConnectSucceeds();
851}
Yixin Wang0d2c6b7e12017-08-16 21:12:55852
853TEST_P(QuicProxyClientSocketTest, WriteSendsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43854 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
855 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
856 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55857 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:56858 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:43859 std::string header = ConstructDataHeader(kLen1);
860 mock_quic_data_.AddWrite(
861 SYNCHRONOUS, ConstructAckAndMultipleDataFramesPacket(
862 3, 1, 1, 1, 0, {header, std::string(kMsg1, kLen1)}));
863 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:41864 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:35865 SYNCHRONOUS,
Victor Vasiliev076657c2019-03-12 02:46:43866 ConstructMultipleDataFramesPacket(
867 4, kLen1 + header.length(), {header2, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:41868 mock_quic_data_.AddWrite(
869 SYNCHRONOUS,
Renjied172e812019-01-16 05:12:35870 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED,
Renjief49758b2019-01-11 23:32:41871 kLen1 + kLen2 + header.length() + header2.length()));
872 } else {
873 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41874 SYNCHRONOUS,
Victor Vasiliev076657c2019-03-12 02:46:43875 ConstructAckAndDataPacket(3, 1, 1, 1, 0, std::string(kMsg1, kLen1)));
876 mock_quic_data_.AddWrite(
877 SYNCHRONOUS, ConstructDataPacket(4, kLen1, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:41878 mock_quic_data_.AddWrite(
879 SYNCHRONOUS,
880 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED, kLen1 + kLen2));
881 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55882
883 Initialize();
884
885 AssertConnectSucceeds();
886
887 AssertSyncWriteSucceeds(kMsg1, kLen1);
888 AssertSyncWriteSucceeds(kMsg2, kLen2);
889}
890
891TEST_P(QuicProxyClientSocketTest, WriteSplitsLargeDataIntoMultiplePackets) {
Renjief49758b2019-01-11 23:32:41892 int write_packet_index = 1;
893 mock_quic_data_.AddWrite(SYNCHRONOUS,
894 ConstructSettingsPacket(write_packet_index++));
895 mock_quic_data_.AddWrite(SYNCHRONOUS,
896 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:43897 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55898 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:43899 std::string header = ConstructDataHeader(kLen1);
Nick Harper23290b82019-05-02 00:02:56900 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:41901 mock_quic_data_.AddWrite(
902 SYNCHRONOUS, ConstructAckAndDataPacket(write_packet_index++, 1, 1, 1, 0,
Victor Vasiliev076657c2019-03-12 02:46:43903 std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:41904 } else {
Renjied172e812019-01-16 05:12:35905 mock_quic_data_.AddWrite(SYNCHRONOUS,
906 ConstructAckAndMultipleDataFramesPacket(
907 write_packet_index++, 1, 1, 1, 0,
Victor Vasiliev076657c2019-03-12 02:46:43908 {header, std::string(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41909 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55910
911 // Expect |kNumDataPackets| data packets, each containing the max possible
912 // amount of data.
Renjied172e812019-01-16 05:12:35913 int numDataPackets = 3;
914 std::string data(numDataPackets * quic::kDefaultMaxPacketSize, 'x');
Renjief49758b2019-01-11 23:32:41915 quic::QuicStreamOffset offset = kLen1 + header.length();
Renjied172e812019-01-16 05:12:35916
Nick Harper23290b82019-05-02 00:02:56917 if (version_.transport_version == quic::QUIC_VERSION_99) {
Renjied172e812019-01-16 05:12:35918 numDataPackets++;
Renjief49758b2019-01-11 23:32:41919 }
Renjied172e812019-01-16 05:12:35920 size_t total_data_length = 0;
921 for (int i = 0; i < numDataPackets; ++i) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55922 size_t max_packet_data_length = GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52923 quic::kDefaultMaxPacketSize, version_, !kIncludeVersion,
924 !kIncludeDiversificationNonce, quic::PACKET_8BYTE_CONNECTION_ID,
925 quic::PACKET_1BYTE_PACKET_NUMBER, offset);
Nick Harper23290b82019-05-02 00:02:56926 if (version_.transport_version == quic::QUIC_VERSION_99 && i == 0) {
Renjied172e812019-01-16 05:12:35927 // 3973 is the data frame length from packet length.
Victor Vasiliev076657c2019-03-12 02:46:43928 std::string header2 = ConstructDataHeader(3973);
Renjied172e812019-01-16 05:12:35929 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43930 SYNCHRONOUS, ConstructMultipleDataFramesPacket(
931 write_packet_index++, offset,
932 {header2, std::string(data.c_str(),
933 max_packet_data_length - 7)}));
Renjied172e812019-01-16 05:12:35934 offset += max_packet_data_length - header2.length() - 1;
Nick Harper23290b82019-05-02 00:02:56935 } else if (version_.transport_version == quic::QUIC_VERSION_99 &&
936 i == numDataPackets - 1) {
Renjied172e812019-01-16 05:12:35937 mock_quic_data_.AddWrite(
938 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, offset,
Victor Vasiliev076657c2019-03-12 02:46:43939 std::string(data.c_str(), 7)));
Renjied172e812019-01-16 05:12:35940 offset += 7;
941 } else {
942 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:43943 SYNCHRONOUS, ConstructDataPacket(
944 write_packet_index++, offset,
945 std::string(data.c_str(), max_packet_data_length)));
Renjied172e812019-01-16 05:12:35946 offset += max_packet_data_length;
947 }
948 if (i != 3) {
949 total_data_length += max_packet_data_length;
950 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55951 }
Renjied172e812019-01-16 05:12:35952
Yixin Wang0d2c6b7e12017-08-16 21:12:55953 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41954 SYNCHRONOUS, ConstructRstPacket(write_packet_index++,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52955 quic::QUIC_STREAM_CANCELLED, offset));
Yixin Wang0d2c6b7e12017-08-16 21:12:55956
957 Initialize();
958
959 AssertConnectSucceeds();
960
961 // Make a small write. An ACK and STOP_WAITING will be bundled. This prevents
962 // ACK and STOP_WAITING from being bundled with the subsequent large write.
963 // This allows the test code for computing the size of data sent in each
964 // packet to not become too complicated.
965 AssertSyncWriteSucceeds(kMsg1, kLen1);
966
967 // Make large write that should be split up
968 AssertSyncWriteSucceeds(data.c_str(), total_data_length);
969}
970
971// ----------- Read
972
973TEST_P(QuicProxyClientSocketTest, ReadReadsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43974 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
975 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
976 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55977 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
978
Victor Vasiliev076657c2019-03-12 02:46:43979 std::string header = ConstructDataHeader(kLen1);
980 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
981 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43982 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55983 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52984 mock_quic_data_.AddWrite(
985 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55986
987 Initialize();
988
989 AssertConnectSucceeds();
990
991 ResumeAndRun();
992 AssertSyncReadEquals(kMsg1, kLen1);
993}
994
995TEST_P(QuicProxyClientSocketTest, ReadDataFromBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:43996 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
997 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
998 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55999 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1000
Victor Vasiliev076657c2019-03-12 02:46:431001 std::string header = ConstructDataHeader(kLen1);
1002 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1003 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431004 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551005 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1006
Victor Vasiliev076657c2019-03-12 02:46:431007 std::string header2 = ConstructDataHeader(kLen2);
1008 mock_quic_data_.AddRead(
1009 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1010 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551011 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1012
1013 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521014 SYNCHRONOUS,
1015 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551016
1017 Initialize();
1018
1019 AssertConnectSucceeds();
1020
1021 ResumeAndRun();
1022 AssertSyncReadEquals(kMsg1, kLen1);
1023
1024 ResumeAndRun();
1025 AssertSyncReadEquals(kMsg2, kLen2);
1026}
1027
1028TEST_P(QuicProxyClientSocketTest, ReadDataMultipleBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431029 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1030 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1031 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551032 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1033
Victor Vasiliev076657c2019-03-12 02:46:431034 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411035 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
Victor Vasiliev076657c2019-03-12 02:46:431036 2, 0, header + std::string(kMsg1, kLen1)));
1037 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1038 std::string header2 = ConstructDataHeader(kLen2);
1039 mock_quic_data_.AddRead(
1040 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1041 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551042 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1043
1044 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521045 SYNCHRONOUS,
1046 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551047
1048 Initialize();
1049
1050 AssertConnectSucceeds();
1051
1052 // The next two reads are consumed and buffered.
1053 ResumeAndRun();
1054
1055 AssertSyncReadEquals(kMsg1, kLen1);
1056 AssertSyncReadEquals(kMsg2, kLen2);
1057}
1058
1059TEST_P(QuicProxyClientSocketTest, LargeReadWillMergeDataFromDifferentFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431060 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1061 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1062 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551063 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1064
Victor Vasiliev076657c2019-03-12 02:46:431065 std::string header = ConstructDataHeader(kLen3);
Renjief49758b2019-01-11 23:32:411066 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
Victor Vasiliev076657c2019-03-12 02:46:431067 2, 0, header + std::string(kMsg3, kLen3)));
1068 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1069 std::string header2 = ConstructDataHeader(kLen3);
1070 mock_quic_data_.AddRead(
1071 ASYNC, ConstructServerDataPacket(3, kLen3 + header.length(),
1072 header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551073 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1074
1075 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521076 SYNCHRONOUS,
1077 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551078
1079 Initialize();
1080
1081 AssertConnectSucceeds();
1082
1083 // The next two reads are consumed and buffered.
1084 ResumeAndRun();
1085 // The payload from two data frames, each with kMsg3 will be combined
1086 // together into a single read().
1087 AssertSyncReadEquals(kMsg33, kLen33);
1088}
1089
1090TEST_P(QuicProxyClientSocketTest, MultipleShortReadsThenMoreRead) {
Zhongyi Shi32f2fd02018-04-16 18:23:431091 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1092 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1093 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551094 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1095
1096 int offset = 0;
1097
Victor Vasiliev076657c2019-03-12 02:46:431098 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411099 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431100 ASYNC,
1101 ConstructServerDataPacket(2, offset, header + std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411102 offset += kLen1 + header.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431103 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551104
Victor Vasiliev076657c2019-03-12 02:46:431105 std::string header2 = ConstructDataHeader(kLen3);
Renjief49758b2019-01-11 23:32:411106 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431107 ASYNC, ConstructServerDataPacket(3, offset,
1108 header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411109 offset += kLen3 + header2.length();
1110 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431111 ASYNC, ConstructServerDataPacket(4, offset,
1112 header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411113 offset += kLen3 + header2.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431114 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(4, 4, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551115
Victor Vasiliev076657c2019-03-12 02:46:431116 std::string header3 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411117 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431118 ASYNC, ConstructServerDataPacket(5, offset,
1119 header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411120 offset += kLen2 + header3.length();
Yixin Wang0d2c6b7e12017-08-16 21:12:551121 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1122
1123 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521124 SYNCHRONOUS,
1125 ConstructAckAndRstPacket(5, quic::QUIC_STREAM_CANCELLED, 5, 5, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551126
1127 Initialize();
1128
1129 AssertConnectSucceeds();
1130
1131 // The next 4 reads are consumed and buffered.
1132 ResumeAndRun();
1133
1134 AssertSyncReadEquals(kMsg1, kLen1);
1135 // The payload from two data frames, each with kMsg3 will be combined
1136 // together into a single read().
1137 AssertSyncReadEquals(kMsg33, kLen33);
1138 AssertSyncReadEquals(kMsg2, kLen2);
1139}
1140
1141TEST_P(QuicProxyClientSocketTest, ReadWillSplitDataFromLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431142 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1143 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1144 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551145 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1146
Victor Vasiliev076657c2019-03-12 02:46:431147 std::string header = ConstructDataHeader(kLen1);
1148 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1149 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431150 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431151 std::string header2 = ConstructDataHeader(kLen33);
Renjief49758b2019-01-11 23:32:411152 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431153 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1154 header2 + std::string(kMsg33, kLen33)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551155 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1156
1157 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521158 SYNCHRONOUS,
1159 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551160
1161 Initialize();
1162
1163 AssertConnectSucceeds();
1164
1165 // The next 2 reads are consumed and buffered.
1166 ResumeAndRun();
1167
1168 AssertSyncReadEquals(kMsg1, kLen1);
1169 // The payload from the single large data frame will be read across
1170 // two different reads.
1171 AssertSyncReadEquals(kMsg3, kLen3);
1172 AssertSyncReadEquals(kMsg3, kLen3);
1173}
1174
1175TEST_P(QuicProxyClientSocketTest, MultipleReadsFromSameLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431176 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1177 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1178 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551179 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1180
Victor Vasiliev076657c2019-03-12 02:46:431181 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:411182 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431183 ASYNC,
1184 ConstructServerDataPacket(2, 0, header + std::string(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:431185 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551186 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1187
Ryan Hamilton8d9ee76e2018-05-29 23:52:521188 mock_quic_data_.AddWrite(
1189 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551190
1191 Initialize();
1192
1193 AssertConnectSucceeds();
1194
1195 // The next read is consumed and buffered.
1196 ResumeAndRun();
1197
1198 // The payload from the single large data frame will be read across
1199 // two different reads.
1200 AssertSyncReadEquals(kMsg33, kLen33);
1201
1202 // Now attempt to do a read of more data than remains buffered
Victor Costan9c7302b2018-08-27 16:39:441203 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kLen33);
Bence Békyd8a21fc32018-06-27 18:29:581204 ASSERT_EQ(kLen3, sock_->Read(buf.get(), kLen33, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:131205 ASSERT_EQ(spdy::SpdyString(kMsg3, kLen3),
1206 spdy::SpdyString(buf->data(), kLen3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551207 ASSERT_TRUE(sock_->IsConnected());
1208}
1209
1210TEST_P(QuicProxyClientSocketTest, ReadAuthResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431211 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1212 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1213 mock_quic_data_.AddRead(ASYNC,
1214 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551215 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1216
Victor Vasiliev076657c2019-03-12 02:46:431217 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411218 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
Victor Vasiliev076657c2019-03-12 02:46:431219 2, 0, header + std::string(kMsg1, kLen1)));
1220 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
1221 std::string header2 = ConstructDataHeader(kLen2);
1222 mock_quic_data_.AddRead(
1223 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1224 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551225 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1226
1227 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521228 SYNCHRONOUS,
1229 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551230
1231 Initialize();
1232
1233 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
1234
1235 // The next two reads are consumed and buffered.
1236 ResumeAndRun();
1237
1238 AssertSyncReadEquals(kMsg1, kLen1);
1239 AssertSyncReadEquals(kMsg2, kLen2);
1240}
1241
1242TEST_P(QuicProxyClientSocketTest, ReadErrorResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431243 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1244 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1245 mock_quic_data_.AddRead(ASYNC,
1246 ConstructServerConnectErrorReplyPacket(1, !kFin));
Victor Vasiliev076657c2019-03-12 02:46:431247 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411248 mock_quic_data_.AddRead(
1249 SYNCHRONOUS,
Victor Vasiliev076657c2019-03-12 02:46:431250 ConstructServerDataPacket(2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431251 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Victor Vasiliev076657c2019-03-12 02:46:431252 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411253 mock_quic_data_.AddRead(
1254 SYNCHRONOUS,
1255 ConstructServerDataPacket(3, kLen1 + header.length(),
Victor Vasiliev076657c2019-03-12 02:46:431256 header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551257 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1258
1259 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521260 SYNCHRONOUS,
1261 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551262 Initialize();
1263
1264 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
1265}
1266
1267// ----------- Reads and Writes
1268
1269TEST_P(QuicProxyClientSocketTest, AsyncReadAroundWrite) {
Renjief49758b2019-01-11 23:32:411270 int write_packet_index = 1;
1271 mock_quic_data_.AddWrite(SYNCHRONOUS,
1272 ConstructSettingsPacket(write_packet_index++));
1273 mock_quic_data_.AddWrite(SYNCHRONOUS,
1274 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431275 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551276 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1277
Victor Vasiliev076657c2019-03-12 02:46:431278 std::string header = ConstructDataHeader(kLen1);
1279 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1280 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431281 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjief49758b2019-01-11 23:32:411282 ConstructAckPacket(write_packet_index++, 2, 1, 1));
1283
Victor Vasiliev076657c2019-03-12 02:46:431284 std::string header2 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561285 if (version_.transport_version == quic::QUIC_VERSION_99) {
Victor Vasiliev076657c2019-03-12 02:46:431286 mock_quic_data_.AddWrite(
1287 SYNCHRONOUS,
1288 ConstructMultipleDataFramesPacket(
1289 write_packet_index++, 0, {header2, std::string(kMsg2, kLen2)}));
Renjied172e812019-01-16 05:12:351290 } else {
Renjief49758b2019-01-11 23:32:411291 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431292 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, header2.length(),
1293 std::string(kMsg2, kLen2)));
Renjied172e812019-01-16 05:12:351294 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551295
1296 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1297
Victor Vasiliev076657c2019-03-12 02:46:431298 std::string header3 = ConstructDataHeader(kLen3);
1299 mock_quic_data_.AddRead(
1300 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1301 header3 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551302 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1303
1304 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:411305 SYNCHRONOUS, ConstructAckAndRstPacket(write_packet_index++,
1306 quic::QUIC_STREAM_CANCELLED, 3, 3,
1307 1, kLen2 + header2.length()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551308
1309 Initialize();
1310
1311 AssertConnectSucceeds();
1312
1313 ResumeAndRun();
1314
1315 AssertSyncReadEquals(kMsg1, kLen1);
1316
1317 AssertReadStarts(kMsg3, kLen3);
1318 // Read should block until after the write succeeds.
1319
1320 AssertSyncWriteSucceeds(kMsg2, kLen2);
1321
1322 ASSERT_FALSE(read_callback_.have_result());
1323 ResumeAndRun();
1324
1325 // Now the read will return.
1326 AssertReadReturns(kMsg3, kLen3);
1327}
1328
1329TEST_P(QuicProxyClientSocketTest, AsyncWriteAroundReads) {
Renjied172e812019-01-16 05:12:351330 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1331 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Zhongyi Shi32f2fd02018-04-16 18:23:431332 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551333 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1334
Victor Vasiliev076657c2019-03-12 02:46:431335 std::string header = ConstructDataHeader(kLen1);
1336 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1337 2, 0, header + std::string(kMsg1, kLen1)));
Renjied172e812019-01-16 05:12:351338 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551339 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1340
Victor Vasiliev076657c2019-03-12 02:46:431341 std::string header2 = ConstructDataHeader(kLen3);
1342 mock_quic_data_.AddRead(
1343 ASYNC, ConstructServerDataPacket(3, kLen1 + header.length(),
1344 header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551345 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1346
1347 mock_quic_data_.AddWrite(ASYNC, ERR_IO_PENDING); // Pause
1348
Victor Vasiliev076657c2019-03-12 02:46:431349 std::string header3 = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561350 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:411351 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431352 ASYNC, ConstructDataPacket(4, 0, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411353 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351354 SYNCHRONOUS, ConstructAckAndDataPacket(5, 3, 3, 1, kLen2,
Victor Vasiliev076657c2019-03-12 02:46:431355 std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411356 } else {
Victor Vasiliev076657c2019-03-12 02:46:431357 mock_quic_data_.AddWrite(ASYNC,
1358 ConstructMultipleDataFramesPacket(
1359 4, 0, {header3, std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:411360 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431361 ASYNC, ConstructAckAndDataPacket(5, 3, 3, 1, header3.length() + kLen2,
1362 header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411363 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551364
Renjied172e812019-01-16 05:12:351365 mock_quic_data_.AddWrite(
1366 SYNCHRONOUS, ConstructRstPacket(6, quic::QUIC_STREAM_CANCELLED,
1367 kLen2 + kLen2 + 2 * header3.length()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551368
1369 Initialize();
1370
1371 AssertConnectSucceeds();
1372
1373 ResumeAndRun();
1374 AssertSyncReadEquals(kMsg1, kLen1);
1375
1376 // Write should block until the next read completes.
1377 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1378 // asynchronous starting with the second time it's called while the UDP socket
1379 // is write-blocked. Therefore, at least two writes need to be called on
1380 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351381 AssertWriteReturns(kMsg2, kLen2, kLen2);
Yixin Wang0d2c6b7e12017-08-16 21:12:551382 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1383
1384 AssertAsyncReadEquals(kMsg3, kLen3);
1385
1386 ASSERT_FALSE(write_callback_.have_result());
1387
1388 // Now the write will complete
1389 ResumeAndRun();
Yixin Wangdbbd8752018-01-17 21:50:021390 EXPECT_EQ(kLen2, write_callback_.WaitForResult());
Yixin Wang0d2c6b7e12017-08-16 21:12:551391}
1392
1393// ----------- Reading/Writing on Closed socket
1394
1395// Reading from an already closed socket should return 0
1396TEST_P(QuicProxyClientSocketTest, ReadOnClosedSocketReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431397 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1398 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1399 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551400 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1401
1402 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1403
1404 Initialize();
1405
1406 AssertConnectSucceeds();
1407
1408 ResumeAndRun();
1409
1410 ASSERT_FALSE(sock_->IsConnected());
Raul Tambre94493c652019-03-11 17:18:351411 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1412 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1413 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551414 ASSERT_FALSE(sock_->IsConnectedAndIdle());
1415}
1416
1417// Read pending when socket is closed should return 0
1418TEST_P(QuicProxyClientSocketTest, PendingReadOnCloseReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431419 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1420 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1421 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551422 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1423
1424 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1425
1426 Initialize();
1427
1428 AssertConnectSucceeds();
1429
1430 AssertReadStarts(kMsg1, kLen1);
1431
1432 ResumeAndRun();
1433
1434 ASSERT_EQ(0, read_callback_.WaitForResult());
1435}
1436
1437// Reading from a disconnected socket is an error
1438TEST_P(QuicProxyClientSocketTest, ReadOnDisconnectSocketReturnsNotConnected) {
Zhongyi Shi32f2fd02018-04-16 18:23:431439 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1440 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1441 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551442 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1443 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521444 SYNCHRONOUS,
1445 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551446
1447 Initialize();
1448
1449 AssertConnectSucceeds();
1450
1451 sock_->Disconnect();
1452
1453 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581454 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551455}
1456
1457// Reading data after receiving FIN should return buffered data received before
1458// FIN, then 0.
1459TEST_P(QuicProxyClientSocketTest, ReadAfterFinReceivedReturnsBufferedData) {
Zhongyi Shi32f2fd02018-04-16 18:23:431460 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1461 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1462 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551463 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1464
Victor Vasiliev076657c2019-03-12 02:46:431465 std::string header = ConstructDataHeader(kLen1);
1466 mock_quic_data_.AddRead(ASYNC, ConstructServerDataFinPacket(
1467 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431468 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551469 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521470 mock_quic_data_.AddWrite(
1471 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551472
1473 Initialize();
1474
1475 AssertConnectSucceeds();
1476
1477 ResumeAndRun();
1478
1479 AssertSyncReadEquals(kMsg1, kLen1);
Raul Tambre94493c652019-03-11 17:18:351480 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1481 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551482
1483 sock_->Disconnect();
1484 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581485 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551486}
1487
1488// Calling Write() on a closed socket is an error.
1489TEST_P(QuicProxyClientSocketTest, WriteOnClosedStream) {
Zhongyi Shi32f2fd02018-04-16 18:23:431490 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1491 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1492 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551493 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1494
1495 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1496
1497 Initialize();
1498
1499 AssertConnectSucceeds();
1500
1501 ResumeAndRun();
1502
1503 AssertWriteReturns(kMsg1, kLen1, ERR_QUIC_PROTOCOL_ERROR);
1504}
1505
1506// Calling Write() on a disconnected socket is an error.
1507TEST_P(QuicProxyClientSocketTest, WriteOnDisconnectedSocket) {
Zhongyi Shi32f2fd02018-04-16 18:23:431508 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1509 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1510 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551511 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1512 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521513 SYNCHRONOUS,
1514 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551515
1516 Initialize();
1517
1518 AssertConnectSucceeds();
1519
1520 sock_->Disconnect();
1521
1522 AssertWriteReturns(kMsg1, kLen1, ERR_SOCKET_NOT_CONNECTED);
1523}
1524
1525// If the socket is closed with a pending Write(), the callback should be called
1526// with the same error the session was closed with.
1527TEST_P(QuicProxyClientSocketTest, WritePendingOnClose) {
Zhongyi Shi32f2fd02018-04-16 18:23:431528 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1529 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1530 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551531 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1532 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1533
1534 Initialize();
1535
1536 AssertConnectSucceeds();
1537
1538 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1539 // asynchronous starting with the second time it's called while the UDP socket
1540 // is write-blocked. Therefore, at least two writes need to be called on
1541 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351542 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551543
1544 // This second write will be async. This is the pending write that's being
1545 // tested.
1546 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1547
1548 // Make sure the write actually starts.
1549 base::RunLoop().RunUntilIdle();
1550
Ryan Hamilton8d9ee76e2018-05-29 23:52:521551 session_->CloseSessionOnError(ERR_CONNECTION_CLOSED,
Renjieba55fae2018-09-20 03:05:161552 quic::QUIC_INTERNAL_ERROR,
1553 quic::ConnectionCloseBehavior::SILENT_CLOSE);
Yixin Wang0d2c6b7e12017-08-16 21:12:551554
1555 EXPECT_THAT(write_callback_.WaitForResult(), IsError(ERR_CONNECTION_CLOSED));
1556}
1557
1558TEST_P(QuicProxyClientSocketTest, DisconnectWithWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431559 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1560 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1561 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551562 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1563 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1564
1565 Initialize();
1566
1567 AssertConnectSucceeds();
1568
1569 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1570 // asynchronous starting with the second time it's called while the UDP socket
1571 // is write-blocked. Therefore, at least two writes need to be called on
1572 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351573 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551574
1575 // This second write will be async. This is the pending write that's being
1576 // tested.
1577 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1578
1579 // Make sure the write actually starts.
1580 base::RunLoop().RunUntilIdle();
1581
1582 sock_->Disconnect();
1583 EXPECT_FALSE(sock_->IsConnected());
1584
1585 base::RunLoop().RunUntilIdle();
1586
1587 EXPECT_FALSE(sock_->IsConnected());
1588 EXPECT_FALSE(write_callback_.have_result());
1589}
1590
1591// If the socket is Disconnected with a pending Read(), the callback
1592// should not be called.
1593TEST_P(QuicProxyClientSocketTest, DisconnectWithReadPending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431594 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1595 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1596 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551597 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1598 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521599 SYNCHRONOUS,
1600 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551601
1602 Initialize();
1603
1604 AssertConnectSucceeds();
1605
1606 EXPECT_TRUE(sock_->IsConnected());
1607
1608 AssertReadStarts(kMsg1, kLen1);
1609
1610 sock_->Disconnect();
1611 EXPECT_FALSE(sock_->IsConnected());
1612
1613 base::RunLoop().RunUntilIdle();
1614
1615 EXPECT_FALSE(sock_->IsConnected());
1616 EXPECT_FALSE(read_callback_.have_result());
1617}
1618
1619// If the socket is Reset when both a read and write are pending,
1620// both should be called back.
1621TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431622 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1623 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1624 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551625 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1626
1627 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521628 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang2bea3cf2017-11-09 18:11:031629 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:431630 std::string header = ConstructDataHeader(kLen2);
Nick Harper23290b82019-05-02 00:02:561631 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:411632 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431633 ASYNC,
1634 ConstructAckAndDataPacket(3, 1, 1, 1, 0, std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411635 mock_quic_data_.AddWrite(
1636 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1637 2, 2, 1, kLen2));
1638 } else {
Renjief49758b2019-01-11 23:32:411639 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351640 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Victor Vasiliev076657c2019-03-12 02:46:431641 3, 1, 1, 1, 0, {header, std::string(kMsg2, kLen2)}));
Frank Kastenholz684ea412019-02-13 18:48:181642 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
1643 4, quic::QUIC_STREAM_CANCELLED, 2,
1644 2, 1, header.length() + kLen2));
Renjief49758b2019-01-11 23:32:411645 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551646
1647 Initialize();
1648
1649 AssertConnectSucceeds();
1650
1651 EXPECT_TRUE(sock_->IsConnected());
1652
1653 AssertReadStarts(kMsg1, kLen1);
1654
1655 // Write should block until the next read completes.
1656 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1657 // asynchronous starting with the second time it's called while the UDP socket
1658 // is write-blocked. Therefore, at least two writes need to be called on
1659 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351660 AssertWriteReturns(kMsg2, kLen2, kLen2);
Renjief49758b2019-01-11 23:32:411661
Yixin Wang0d2c6b7e12017-08-16 21:12:551662 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1663
1664 ResumeAndRun();
1665
1666 EXPECT_TRUE(read_callback_.have_result());
1667 EXPECT_TRUE(write_callback_.have_result());
1668}
1669
1670// Makes sure the proxy client socket's source gets the expected NetLog events
1671// and only the expected NetLog events (No SpdySession events).
1672TEST_P(QuicProxyClientSocketTest, NetLog) {
Zhongyi Shi32f2fd02018-04-16 18:23:431673 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1674 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1675 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551676 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1677
Victor Vasiliev076657c2019-03-12 02:46:431678 std::string header = ConstructDataHeader(kLen1);
1679 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1680 2, 0, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431681 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang2bea3cf2017-11-09 18:11:031682 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521683 mock_quic_data_.AddWrite(
1684 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551685
1686 Initialize();
1687
1688 AssertConnectSucceeds();
1689
1690 ResumeAndRun();
1691 AssertSyncReadEquals(kMsg1, kLen1);
1692
1693 NetLogSource sock_source = sock_->NetLog().source();
1694 sock_.reset();
1695
1696 TestNetLogEntry::List entry_list;
1697 net_log_.GetEntriesForSource(sock_source, &entry_list);
1698
1699 ASSERT_EQ(entry_list.size(), 10u);
1700 EXPECT_TRUE(
1701 LogContainsBeginEvent(entry_list, 0, NetLogEventType::SOCKET_ALIVE));
1702 EXPECT_TRUE(LogContainsEvent(entry_list, 1,
1703 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
1704 NetLogEventPhase::NONE));
1705 EXPECT_TRUE(LogContainsBeginEvent(
1706 entry_list, 2, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1707 EXPECT_TRUE(LogContainsEvent(
1708 entry_list, 3, NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
1709 NetLogEventPhase::NONE));
1710 EXPECT_TRUE(LogContainsEndEvent(
1711 entry_list, 4, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1712 EXPECT_TRUE(LogContainsBeginEvent(
1713 entry_list, 5, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1714 EXPECT_TRUE(LogContainsEvent(
1715 entry_list, 6,
1716 NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
1717 NetLogEventPhase::NONE));
1718 EXPECT_TRUE(LogContainsEndEvent(
1719 entry_list, 7, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1720 EXPECT_TRUE(LogContainsEvent(entry_list, 8,
1721 NetLogEventType::SOCKET_BYTES_RECEIVED,
1722 NetLogEventPhase::NONE));
1723 EXPECT_TRUE(
1724 LogContainsEndEvent(entry_list, 9, NetLogEventType::SOCKET_ALIVE));
1725}
1726
Bence Béky8ddc2492018-06-13 01:02:041727// A helper class that will delete |sock| when the callback is invoked.
Yixin Wang0d2c6b7e12017-08-16 21:12:551728class DeleteSockCallback : public TestCompletionCallbackBase {
1729 public:
1730 explicit DeleteSockCallback(std::unique_ptr<QuicProxyClientSocket>* sock)
Bence Béky8ddc2492018-06-13 01:02:041731 : sock_(sock) {}
Yixin Wang0d2c6b7e12017-08-16 21:12:551732
1733 ~DeleteSockCallback() override {}
1734
Bence Béky8ddc2492018-06-13 01:02:041735 CompletionOnceCallback callback() {
1736 return base::BindOnce(&DeleteSockCallback::OnComplete,
1737 base::Unretained(this));
1738 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551739
1740 private:
1741 void OnComplete(int result) {
Raul Tambre94493c652019-03-11 17:18:351742 sock_->reset(nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:551743 SetResult(result);
1744 }
1745
1746 std::unique_ptr<QuicProxyClientSocket>* sock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:551747
1748 DISALLOW_COPY_AND_ASSIGN(DeleteSockCallback);
1749};
1750
1751// If the socket is reset when both a read and write are pending, and the
1752// read callback causes the socket to be deleted, the write callback should
1753// not be called.
1754TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePendingDelete) {
Zhongyi Shi32f2fd02018-04-16 18:23:431755 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1756 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1757 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551758 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1759
1760 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521761 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551762 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harper23290b82019-05-02 00:02:561763 if (version_.transport_version != quic::QUIC_VERSION_99) {
Renjief49758b2019-01-11 23:32:411764 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431765 ASYNC,
1766 ConstructAckAndDataPacket(3, 1, 1, 1, 0, std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411767 mock_quic_data_.AddWrite(
1768 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1769 2, 2, 1, kLen1));
1770 } else {
Victor Vasiliev076657c2019-03-12 02:46:431771 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411772 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351773 ASYNC, ConstructAckAndMultipleDataFramesPacket(
Victor Vasiliev076657c2019-03-12 02:46:431774 3, 1, 1, 1, 0, {header, std::string(kMsg1, kLen1)}));
Frank Kastenholz684ea412019-02-13 18:48:181775 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
1776 4, quic::QUIC_STREAM_CANCELLED, 2,
1777 2, 1, header.length() + kLen1));
Renjief49758b2019-01-11 23:32:411778 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551779
1780 Initialize();
1781
1782 AssertConnectSucceeds();
1783
1784 EXPECT_TRUE(sock_->IsConnected());
1785
1786 DeleteSockCallback read_callback(&sock_);
Victor Costan9c7302b2018-08-27 16:39:441787 scoped_refptr<IOBuffer> read_buf = base::MakeRefCounted<IOBuffer>(kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551788 ASSERT_EQ(ERR_IO_PENDING,
1789 sock_->Read(read_buf.get(), kLen1, read_callback.callback()));
1790
1791 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1792 // asynchronous starting with the second time it's called while the UDP socket
1793 // is write-blocked. Therefore, at least two writes need to be called on
1794 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351795 AssertWriteReturns(kMsg1, kLen1, kLen1);
Renjief49758b2019-01-11 23:32:411796
Yixin Wang0d2c6b7e12017-08-16 21:12:551797 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1798
1799 ResumeAndRun();
1800
1801 EXPECT_FALSE(sock_.get());
1802
1803 EXPECT_EQ(0, read_callback.WaitForResult());
1804 EXPECT_FALSE(write_callback_.have_result());
1805}
1806
Victor Costane635086f2019-01-27 05:20:301807INSTANTIATE_TEST_SUITE_P(
Bence Békyce380cb2018-04-26 23:39:551808 VersionIncludeStreamDependencySequence,
Yixin Wang079ad542018-01-11 04:06:051809 QuicProxyClientSocketTest,
Victor Vasiliev5d6cdc22019-05-28 20:37:431810 ::testing::Combine(::testing::ValuesIn(quic::AllVersionsExcept99()),
Nick Harper23290b82019-05-02 00:02:561811 ::testing::Bool()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551812
1813} // namespace test
Dan Zhangf11470172017-09-18 22:02:091814} // namespace net