blob: 056d001647b22f82c74f9a06b8d3270bbfa931bc [file] [log] [blame]
Yixin Wang0d2c6b7e12017-08-16 21:12:551// Copyright (c) 2017 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Ryan Hamiltona3ee93a72018-08-01 22:03:085#include "net/quic/quic_proxy_client_socket.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:556
Bence Béky8f9d7d3952017-10-09 19:58:047#include <memory>
Bence Békyd8a21fc32018-06-27 18:29:588#include <utility>
Bence Béky8f9d7d3952017-10-09 19:58:049
Sebastien Marchand6d0558fd2019-01-25 16:49:3710#include "base/bind.h"
Bence Béky8f9d7d3952017-10-09 19:58:0411#include "base/memory/ptr_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5512#include "base/run_loop.h"
13#include "base/strings/utf_string_conversions.h"
Zhongyi Shic16b4102019-02-12 00:37:4014#include "base/time/default_tick_clock.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5515#include "net/dns/mock_host_resolver.h"
16#include "net/http/http_auth_cache.h"
17#include "net/http/http_auth_handler_factory.h"
18#include "net/http/http_response_headers.h"
19#include "net/http/transport_security_state.h"
20#include "net/log/test_net_log.h"
21#include "net/log/test_net_log_util.h"
Ryan Hamiltona3ee93a72018-08-01 22:03:0822#include "net/quic/crypto/proof_verifier_chromium.h"
23#include "net/quic/mock_crypto_client_stream_factory.h"
24#include "net/quic/mock_quic_data.h"
25#include "net/quic/quic_chromium_alarm_factory.h"
26#include "net/quic/quic_chromium_client_session.h"
27#include "net/quic/quic_chromium_connection_helper.h"
28#include "net/quic/quic_chromium_packet_writer.h"
29#include "net/quic/quic_http_utils.h"
30#include "net/quic/quic_server_info.h"
31#include "net/quic/quic_stream_factory.h"
32#include "net/quic/quic_test_packet_maker.h"
33#include "net/quic/test_task_runner.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5534#include "net/socket/socket_test_util.h"
35#include "net/test/cert_test_util.h"
36#include "net/test/test_data_directory.h"
Bence Béky98447b12018-05-08 03:14:0137#include "net/test/test_with_scoped_task_environment.h"
Ryan Hamilton56b10c5d2018-05-11 13:40:1638#include "net/third_party/quic/core/crypto/null_encrypter.h"
Ryan Hamilton47cf9d12018-10-17 04:33:0939#include "net/third_party/quic/core/quic_utils.h"
Ryan Hamilton56b10c5d2018-05-11 13:40:1640#include "net/third_party/quic/core/tls_client_handshaker.h"
41#include "net/third_party/quic/test_tools/crypto_test_utils.h"
42#include "net/third_party/quic/test_tools/mock_clock.h"
43#include "net/third_party/quic/test_tools/mock_random.h"
44#include "net/third_party/quic/test_tools/quic_connection_peer.h"
45#include "net/third_party/quic/test_tools/quic_test_utils.h"
[email protected]578968d42017-12-13 15:39:3246#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5547#include "testing/gmock/include/gmock/gmock.h"
48#include "testing/gtest/include/gtest/gtest.h"
49
50using testing::_;
51using testing::AnyNumber;
52using testing::Return;
53
54namespace {
55
56static const char kOriginHost[] = "www.google.com";
57static const int kOriginPort = 443;
58static const char kProxyUrl[] = "https://myproxy:6121/";
59static const char kProxyHost[] = "myproxy";
60static const int kProxyPort = 6121;
61static const char kUserAgent[] = "Mozilla/1.0";
62static const char kRedirectUrl[] = "https://example.com/";
63
64static const char kMsg1[] = "\0hello!\xff";
65static const int kLen1 = 8;
66static const char kMsg2[] = "\0a2345678\0";
67static const int kLen2 = 10;
68static const char kMsg3[] = "bye!";
69static const int kLen3 = 4;
70static const char kMsg33[] = "bye!bye!";
71static const int kLen33 = kLen3 + kLen3;
72static const char kMsg333[] = "bye!bye!bye!";
73static const int kLen333 = kLen3 + kLen3 + kLen3;
74
75} // anonymous namespace
76
77namespace net {
78namespace test {
79
Michael Warres74ee3ce2017-10-09 15:26:3780class QuicProxyClientSocketTest
Ryan Hamilton8d9ee76e2018-05-29 23:52:5281 : public ::testing::TestWithParam<
82 std::tuple<quic::QuicTransportVersion, bool>>,
Bence Béky98447b12018-05-08 03:14:0183 public WithScopedTaskEnvironment {
Yixin Wang0d2c6b7e12017-08-16 21:12:5584 protected:
85 static const bool kFin = true;
86 static const bool kIncludeVersion = true;
87 static const bool kIncludeDiversificationNonce = true;
88 static const bool kIncludeCongestionFeedback = true;
89 static const bool kSendFeedback = true;
90
91 static size_t GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:5292 quic::QuicByteCount packet_length,
93 quic::QuicTransportVersion version,
Yixin Wang0d2c6b7e12017-08-16 21:12:5594 bool include_version,
95 bool include_diversification_nonce,
Ryan Hamilton8d9ee76e2018-05-29 23:52:5296 quic::QuicConnectionIdLength connection_id_length,
97 quic::QuicPacketNumberLength packet_number_length,
98 quic::QuicStreamOffset offset) {
Yixin Wang0d2c6b7e12017-08-16 21:12:5599 size_t min_data_length = 1;
100 size_t min_packet_length =
Ryan Hamilton8d9ee76e2018-05-29 23:52:52101 quic::NullEncrypter(quic::Perspective::IS_CLIENT)
Yixin Wang0d2c6b7e12017-08-16 21:12:55102 .GetCiphertextSize(min_data_length) +
Ryan Hamilton8d9ee76e2018-05-29 23:52:52103 quic::QuicPacketCreator::StreamFramePacketOverhead(
Michael Warres60637ae2018-06-05 20:03:11104 version, quic::PACKET_8BYTE_CONNECTION_ID,
105 quic::PACKET_0BYTE_CONNECTION_ID, include_version,
Yixin Wang0d2c6b7e12017-08-16 21:12:55106 include_diversification_nonce, packet_number_length, offset);
107
108 DCHECK(packet_length >= min_packet_length);
109 return min_data_length + packet_length - min_packet_length;
110 }
111
112 QuicProxyClientSocketTest()
Yixin Wang079ad542018-01-11 04:06:05113 : version_(std::get<0>(GetParam())),
Ryan Hamilton47cf9d12018-10-17 04:33:09114 client_data_stream_id1_(quic::QuicUtils::GetHeadersStreamId(version_) +
Fan Yang32c5a112018-12-10 20:06:33115 quic::QuicUtils::StreamIdDelta(version_)),
Yixin Wang079ad542018-01-11 04:06:05116 client_headers_include_h2_stream_dependency_(std::get<1>(GetParam())),
Ryan Hamilton8d9ee76e2018-05-29 23:52:52117 crypto_config_(quic::test::crypto_test_utils::ProofVerifierForTesting(),
118 quic::TlsClientHandshaker::CreateSslCtx()),
David Schinazic8281052019-01-24 06:14:17119 connection_id_(quic::test::TestConnectionId(2)),
Yixin Wang0d2c6b7e12017-08-16 21:12:55120 client_maker_(version_,
121 connection_id_,
122 &clock_,
123 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52124 quic::Perspective::IS_CLIENT,
Yixin Wang079ad542018-01-11 04:06:05125 client_headers_include_h2_stream_dependency_),
Yixin Wang0d2c6b7e12017-08-16 21:12:55126 server_maker_(version_,
127 connection_id_,
128 &clock_,
129 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52130 quic::Perspective::IS_SERVER,
Yixin Wang079ad542018-01-11 04:06:05131 false),
Yixin Wang0d2c6b7e12017-08-16 21:12:55132 random_generator_(0),
133 header_stream_offset_(0),
134 response_offset_(0),
135 user_agent_(kUserAgent),
136 proxy_host_port_(kProxyHost, kProxyPort),
137 endpoint_host_port_(kOriginHost, kOriginPort),
138 host_resolver_(new MockCachingHostResolver()),
139 http_auth_handler_factory_(
140 HttpAuthHandlerFactory::CreateDefault(host_resolver_.get())) {
141 IPAddress ip(192, 0, 2, 33);
142 peer_addr_ = IPEndPoint(ip, 443);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52143 clock_.AdvanceTime(quic::QuicTime::Delta::FromMilliseconds(20));
Yixin Wang0d2c6b7e12017-08-16 21:12:55144 }
145
146 void SetUp() override {}
147
Yixin Wang2bea3cf2017-11-09 18:11:03148 void TearDown() override {
149 sock_.reset();
150 EXPECT_TRUE(mock_quic_data_.AllReadDataConsumed());
151 EXPECT_TRUE(mock_quic_data_.AllWriteDataConsumed());
152 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55153
154 void Initialize() {
155 std::unique_ptr<MockUDPClientSocket> socket(new MockUDPClientSocket(
156 mock_quic_data_.InitializeAndGetSequencedSocketData(),
157 net_log_.bound().net_log()));
158 socket->Connect(peer_addr_);
159 runner_ = new TestTaskRunner(&clock_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52160 send_algorithm_ = new quic::test::MockSendAlgorithm();
Yixin Wang0d2c6b7e12017-08-16 21:12:55161 EXPECT_CALL(*send_algorithm_, InRecovery()).WillRepeatedly(Return(false));
162 EXPECT_CALL(*send_algorithm_, InSlowStart()).WillRepeatedly(Return(false));
163 EXPECT_CALL(*send_algorithm_, OnPacketSent(_, _, _, _, _))
Dan Zhangf11470172017-09-18 22:02:09164 .Times(testing::AtLeast(1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55165 EXPECT_CALL(*send_algorithm_, GetCongestionWindow())
Ryan Hamilton8d9ee76e2018-05-29 23:52:52166 .WillRepeatedly(Return(quic::kMaxPacketSize));
Yixin Wang0d2c6b7e12017-08-16 21:12:55167 EXPECT_CALL(*send_algorithm_, PacingRate(_))
Ryan Hamilton8d9ee76e2018-05-29 23:52:52168 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Michael Warres74ee3ce2017-10-09 15:26:37169 EXPECT_CALL(*send_algorithm_, CanSend(_)).WillRepeatedly(Return(true));
Yixin Wang0d2c6b7e12017-08-16 21:12:55170 EXPECT_CALL(*send_algorithm_, BandwidthEstimate())
Ryan Hamilton8d9ee76e2018-05-29 23:52:52171 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Yixin Wang0d2c6b7e12017-08-16 21:12:55172 EXPECT_CALL(*send_algorithm_, SetFromConfig(_, _)).Times(AnyNumber());
173 EXPECT_CALL(*send_algorithm_, OnApplicationLimited(_)).Times(AnyNumber());
174 EXPECT_CALL(*send_algorithm_, GetCongestionControlType())
175 .Times(AnyNumber());
176 helper_.reset(
177 new QuicChromiumConnectionHelper(&clock_, &random_generator_));
178 alarm_factory_.reset(new QuicChromiumAlarmFactory(runner_.get(), &clock_));
179
Ryan Hamilton9edcf1a2017-11-22 05:55:17180 QuicChromiumPacketWriter* writer = new QuicChromiumPacketWriter(
181 socket.get(), base::ThreadTaskRunnerHandle::Get().get());
Ryan Hamilton8d9ee76e2018-05-29 23:52:52182 quic::QuicConnection* connection = new quic::QuicConnection(
183 connection_id_,
184 quic::QuicSocketAddress(quic::QuicSocketAddressImpl(peer_addr_)),
Yixin Wang0d2c6b7e12017-08-16 21:12:55185 helper_.get(), alarm_factory_.get(), writer, true /* owns_writer */,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52186 quic::Perspective::IS_CLIENT,
187 quic::test::SupportedVersions(
188 quic::ParsedQuicVersion(quic::PROTOCOL_QUIC_CRYPTO, version_)));
Yixin Wang0d2c6b7e12017-08-16 21:12:55189 connection->set_visitor(&visitor_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52190 quic::test::QuicConnectionPeer::SetSendAlgorithm(connection,
191 send_algorithm_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55192
193 // Load a certificate that is valid for *.example.org
194 scoped_refptr<X509Certificate> test_cert(
195 ImportCertFromFile(GetTestCertsDirectory(), "wildcard.pem"));
196 EXPECT_TRUE(test_cert.get());
197
198 verify_details_.cert_verify_result.verified_cert = test_cert;
199 verify_details_.cert_verify_result.is_issued_by_known_root = true;
200 crypto_client_stream_factory_.AddProofVerifyDetails(&verify_details_);
201
202 base::TimeTicks dns_end = base::TimeTicks::Now();
203 base::TimeTicks dns_start = dns_end - base::TimeDelta::FromMilliseconds(1);
204
205 session_.reset(new QuicChromiumClientSession(
206 connection, std::move(socket),
207 /*stream_factory=*/nullptr, &crypto_client_stream_factory_, &clock_,
Nick Harper89bc7212018-07-31 19:07:57208 &transport_security_state_, /*ssl_config_service=*/nullptr,
Yixin Wang0d2c6b7e12017-08-16 21:12:55209 base::WrapUnique(static_cast<QuicServerInfo*>(nullptr)),
Paul Jensen8e3c5d32018-02-19 17:06:33210 QuicSessionKey("mail.example.org", 80, PRIVACY_MODE_DISABLED,
211 SocketTag()),
Zhongyi Shi757fcce2018-06-27 05:41:27212 /*require_confirmation=*/false, /*migrate_session_early_v2=*/false,
213 /*migrate_session_on_network_change_v2=*/false,
214 /*default_network=*/NetworkChangeNotifier::kInvalidNetworkHandle,
Zhongyi Shic16b4102019-02-12 00:37:40215 base::TimeDelta::FromSeconds(kDefaultIdleSessionMigrationPeriodSeconds),
Zhongyi Shi73f23ca872017-12-13 18:37:13216 base::TimeDelta::FromSeconds(kMaxTimeOnNonDefaultNetworkSecs),
Zhongyi Shiee760762018-08-01 00:54:29217 kMaxMigrationsToNonDefaultNetworkOnWriteError,
Zhongyi Shi8b1e43f2017-12-13 20:46:30218 kMaxMigrationsToNonDefaultNetworkOnPathDegrading,
Zhongyi Shi5f587cc2017-11-21 23:24:17219 kQuicYieldAfterPacketsRead,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52220 quic::QuicTime::Delta::FromMilliseconds(
221 kQuicYieldAfterDurationMilliseconds),
Zhongyi Shidbce7f412019-02-01 23:16:29222 /*go_away_on_path_degrading*/ false,
Yixin Wang079ad542018-01-11 04:06:05223 client_headers_include_h2_stream_dependency_, /*cert_verify_flags=*/0,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52224 quic::test::DefaultQuicConfig(), &crypto_config_, "CONNECTION_UNKNOWN",
225 dns_start, dns_end, &push_promise_index_, nullptr,
Zhongyi Shic16b4102019-02-12 00:37:40226 base::DefaultTickClock::GetInstance(),
Yixin Wang0d2c6b7e12017-08-16 21:12:55227 base::ThreadTaskRunnerHandle::Get().get(),
228 /*socket_performance_watcher=*/nullptr, net_log_.bound().net_log()));
229
230 writer->set_delegate(session_.get());
231
Ryan Hamilton6c2a2a82017-12-15 02:06:28232 session_handle_ =
233 session_->CreateHandle(HostPortPair("mail.example.org", 80));
Yixin Wang0d2c6b7e12017-08-16 21:12:55234
235 session_->Initialize();
236 TestCompletionCallback callback;
237 EXPECT_THAT(session_->CryptoConnect(callback.callback()), IsOk());
238 EXPECT_TRUE(session_->IsCryptoHandshakeConfirmed());
239
Ramin Halavati683bcaa92018-02-14 08:42:39240 EXPECT_THAT(session_handle_->RequestStream(true, callback.callback(),
241 TRAFFIC_ANNOTATION_FOR_TESTS),
Yixin Wang0d2c6b7e12017-08-16 21:12:55242 IsOk());
243 std::unique_ptr<QuicChromiumClientStream::Handle> stream_handle =
244 session_handle_->ReleaseStream();
245 EXPECT_TRUE(stream_handle->IsOpen());
246
247 sock_.reset(new QuicProxyClientSocket(
248 std::move(stream_handle), std::move(session_handle_), user_agent_,
249 endpoint_host_port_, net_log_.bound(),
250 new HttpAuthController(HttpAuth::AUTH_PROXY,
251 GURL("https://" + proxy_host_port_.ToString()),
252 &http_auth_cache_,
253 http_auth_handler_factory_.get())));
254
255 session_->StartReading();
256 }
257
Ryan Hamilton0239aac2018-05-19 00:03:13258 void PopulateConnectRequestIR(spdy::SpdyHeaderBlock* block) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55259 (*block)[":method"] = "CONNECT";
260 (*block)[":authority"] = endpoint_host_port_.ToString();
261 (*block)["user-agent"] = kUserAgent;
262 }
263
264 // Helper functions for constructing packets sent by the client
265
Ryan Hamilton8d9ee76e2018-05-29 23:52:52266 std::unique_ptr<quic::QuicReceivedPacket> ConstructSettingsPacket(
Fan Yangac867502019-01-28 21:10:23267 uint64_t packet_number) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55268 return client_maker_.MakeInitialSettingsPacket(packet_number,
269 &header_stream_offset_);
270 }
271
Ryan Hamilton8d9ee76e2018-05-29 23:52:52272 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23273 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52274 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23275 uint64_t largest_received,
276 uint64_t smallest_received,
277 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55278 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09279 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55280 largest_received, smallest_received, least_unacked, kSendFeedback);
281 }
282
Ryan Hamilton8d9ee76e2018-05-29 23:52:52283 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23284 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52285 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23286 uint64_t largest_received,
287 uint64_t smallest_received,
288 uint64_t least_unacked,
Yixin Wang0d2c6b7e12017-08-16 21:12:55289 size_t bytes_written) {
290 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09291 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55292 largest_received, smallest_received, least_unacked, kSendFeedback,
293 bytes_written);
294 }
295
Ryan Hamilton8d9ee76e2018-05-29 23:52:52296 std::unique_ptr<quic::QuicReceivedPacket> ConstructRstPacket(
Fan Yangac867502019-01-28 21:10:23297 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52298 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55299 size_t bytes_written) {
300 return client_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09301 client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55302 bytes_written);
303 }
304
Ryan Hamilton8d9ee76e2018-05-29 23:52:52305 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectRequestPacket(
Fan Yangac867502019-01-28 21:10:23306 uint64_t packet_number,
Lily Chenf11e1292018-11-29 16:42:09307 RequestPriority request_priority = LOWEST) {
Ryan Hamilton0239aac2018-05-19 00:03:13308 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55309 PopulateConnectRequestIR(&block);
310 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09311 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Lily Chenf11e1292018-11-29 16:42:09312 ConvertRequestPriorityToQuicPriority(request_priority),
313 std::move(block), 0, nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55314 }
315
Ryan Hamilton8d9ee76e2018-05-29 23:52:52316 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectAuthRequestPacket(
Fan Yangac867502019-01-28 21:10:23317 uint64_t packet_number) {
Ryan Hamilton0239aac2018-05-19 00:03:13318 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55319 PopulateConnectRequestIR(&block);
320 block["proxy-authorization"] = "Basic Zm9vOmJhcg==";
321 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09322 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Yixin Wang7a3f1b8d2018-01-17 21:40:48323 ConvertRequestPriorityToQuicPriority(LOWEST), std::move(block), 0,
324 nullptr, &header_stream_offset_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55325 }
326
Ryan Hamilton8d9ee76e2018-05-29 23:52:52327 std::unique_ptr<quic::QuicReceivedPacket> ConstructDataPacket(
Fan Yangac867502019-01-28 21:10:23328 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52329 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41330 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09331 return client_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41332 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55333 }
334
Renjied172e812019-01-16 05:12:35335 std::unique_ptr<quic::QuicReceivedPacket> ConstructMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23336 uint64_t packet_number,
Renjied172e812019-01-16 05:12:35337 quic::QuicStreamOffset offset,
338 const std::vector<std::string> data_writes) {
339 return client_maker_.MakeMultipleDataFramesPacket(
340 packet_number, client_data_stream_id1_, !kIncludeVersion, !kFin, offset,
341 data_writes);
342 }
343
Ryan Hamilton8d9ee76e2018-05-29 23:52:52344 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndDataPacket(
Fan Yangac867502019-01-28 21:10:23345 uint64_t packet_number,
346 uint64_t largest_received,
347 uint64_t smallest_received,
348 uint64_t least_unacked,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52349 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41350 quic::QuicStringPiece data) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55351 return client_maker_.MakeAckAndDataPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09352 packet_number, !kIncludeVersion, client_data_stream_id1_,
353 largest_received, smallest_received, least_unacked, !kFin, offset,
Renjief49758b2019-01-11 23:32:41354 data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55355 }
356
Renjied172e812019-01-16 05:12:35357 std::unique_ptr<quic::QuicReceivedPacket>
358 ConstructAckAndMultipleDataFramesPacket(
Fan Yangac867502019-01-28 21:10:23359 uint64_t packet_number,
360 uint64_t largest_received,
361 uint64_t smallest_received,
362 uint64_t least_unacked,
Renjied172e812019-01-16 05:12:35363 quic::QuicStreamOffset offset,
364 const std::vector<std::string> data_writes) {
365 return client_maker_.MakeAckAndMultipleDataFramesPacket(
366 packet_number, !kIncludeVersion, client_data_stream_id1_,
367 largest_received, smallest_received, least_unacked, !kFin, offset,
368 data_writes);
369 }
370
Ryan Hamilton8d9ee76e2018-05-29 23:52:52371 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckPacket(
Fan Yangac867502019-01-28 21:10:23372 uint64_t packet_number,
373 uint64_t largest_received,
374 uint64_t smallest_received,
375 uint64_t least_unacked) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55376 return client_maker_.MakeAckPacket(packet_number, largest_received,
377 smallest_received, least_unacked,
378 kSendFeedback);
379 }
380
381 // Helper functions for constructing packets sent by the server
382
Ryan Hamilton8d9ee76e2018-05-29 23:52:52383 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerRstPacket(
Fan Yangac867502019-01-28 21:10:23384 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52385 quic::QuicRstStreamErrorCode error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55386 size_t bytes_written) {
387 return server_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09388 client_data_stream_id1_, error_code,
Yixin Wang0d2c6b7e12017-08-16 21:12:55389 bytes_written);
390 }
391
Ryan Hamilton8d9ee76e2018-05-29 23:52:52392 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataPacket(
Fan Yangac867502019-01-28 21:10:23393 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52394 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41395 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09396 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41397 !kIncludeVersion, !kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55398 }
399
Ryan Hamilton8d9ee76e2018-05-29 23:52:52400 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataFinPacket(
Fan Yangac867502019-01-28 21:10:23401 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52402 quic::QuicStreamOffset offset,
Renjief49758b2019-01-11 23:32:41403 quic::QuicStringPiece data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09404 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Renjief49758b2019-01-11 23:32:41405 !kIncludeVersion, kFin, offset, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55406 }
407
Ryan Hamilton8d9ee76e2018-05-29 23:52:52408 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerConnectReplyPacket(
Fan Yangac867502019-01-28 21:10:23409 uint64_t packet_number,
Yixin Wang0d2c6b7e12017-08-16 21:12:55410 bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13411 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55412 block[":status"] = "200";
413
414 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09415 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55416 std::move(block), nullptr, &response_offset_);
417 }
418
Ryan Hamilton8d9ee76e2018-05-29 23:52:52419 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23420 ConstructServerConnectAuthReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13421 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55422 block[":status"] = "407";
423 block["proxy-authenticate"] = "Basic realm=\"MyRealm1\"";
424 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09425 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55426 std::move(block), nullptr, &response_offset_);
427 }
428
Ryan Hamilton8d9ee76e2018-05-29 23:52:52429 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23430 ConstructServerConnectRedirectReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13431 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55432 block[":status"] = "302";
433 block["location"] = kRedirectUrl;
434 block["set-cookie"] = "foo=bar";
435 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09436 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55437 std::move(block), nullptr, &response_offset_);
438 }
439
Ryan Hamilton8d9ee76e2018-05-29 23:52:52440 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23441 ConstructServerConnectErrorReplyPacket(uint64_t packet_number, bool fin) {
Ryan Hamilton0239aac2018-05-19 00:03:13442 spdy::SpdyHeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55443 block[":status"] = "500";
444
445 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09446 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Yixin Wang0d2c6b7e12017-08-16 21:12:55447 std::move(block), nullptr, &response_offset_);
448 }
449
450 void AssertConnectSucceeds() {
451 TestCompletionCallback callback;
452 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
453 ASSERT_THAT(callback.WaitForResult(), IsOk());
454 }
455
456 void AssertConnectFails(int result) {
457 TestCompletionCallback callback;
458 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
459 ASSERT_EQ(result, callback.WaitForResult());
460 }
461
462 void ResumeAndRun() {
463 // Run until the pause, if the provider isn't paused yet.
464 SequencedSocketData* data = mock_quic_data_.GetSequencedSocketData();
465 data->RunUntilPaused();
466 data->Resume();
467 base::RunLoop().RunUntilIdle();
468 }
469
470 void AssertWriteReturns(const char* data, int len, int rv) {
Victor Costan9c7302b2018-08-27 16:39:44471 scoped_refptr<IOBufferWithSize> buf =
472 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55473 memcpy(buf->data(), data, len);
474 EXPECT_EQ(rv,
[email protected]578968d42017-12-13 15:39:32475 sock_->Write(buf.get(), buf->size(), write_callback_.callback(),
476 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55477 }
478
479 void AssertSyncWriteSucceeds(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44480 scoped_refptr<IOBufferWithSize> buf =
481 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55482 memcpy(buf->data(), data, len);
Bence Békyd8a21fc32018-06-27 18:29:58483 EXPECT_EQ(len,
484 sock_->Write(buf.get(), buf->size(), CompletionOnceCallback(),
485 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55486 }
487
488 void AssertSyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44489 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Bence Békyd8a21fc32018-06-27 18:29:58490 ASSERT_EQ(len, sock_->Read(buf.get(), len, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:13491 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55492 ASSERT_TRUE(sock_->IsConnected());
493 }
494
495 void AssertAsyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44496 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55497 ASSERT_EQ(ERR_IO_PENDING,
498 sock_->Read(buf.get(), len, read_callback_.callback()));
499 EXPECT_TRUE(sock_->IsConnected());
500
501 ResumeAndRun();
502
503 EXPECT_EQ(len, read_callback_.WaitForResult());
504 EXPECT_TRUE(sock_->IsConnected());
Ryan Hamilton0239aac2018-05-19 00:03:13505 ASSERT_EQ(spdy::SpdyString(data, len), spdy::SpdyString(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55506 }
507
508 void AssertReadStarts(const char* data, int len) {
509 // Issue the read, which will be completed asynchronously.
Victor Costan9c7302b2018-08-27 16:39:44510 read_buf_ = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55511 ASSERT_EQ(ERR_IO_PENDING,
512 sock_->Read(read_buf_.get(), len, read_callback_.callback()));
513 EXPECT_TRUE(sock_->IsConnected());
514 }
515
516 void AssertReadReturns(const char* data, int len) {
517 EXPECT_TRUE(sock_->IsConnected());
518
519 // Now the read will return.
520 EXPECT_EQ(len, read_callback_.WaitForResult());
Ryan Hamilton0239aac2018-05-19 00:03:13521 ASSERT_EQ(spdy::SpdyString(data, len),
522 spdy::SpdyString(read_buf_->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55523 }
524
Renjief49758b2019-01-11 23:32:41525 quic::QuicString ConstructDataHeader(size_t body_len) {
526 if (version_ != quic::QUIC_VERSION_99) {
527 return "";
528 }
529 quic::HttpEncoder encoder;
530 std::unique_ptr<char[]> buffer;
531 auto header_length = encoder.SerializeDataFrameHeader(body_len, &buffer);
532 return quic::QuicString(buffer.get(), header_length);
533 }
534
Ryan Hamilton8d9ee76e2018-05-29 23:52:52535 const quic::QuicTransportVersion version_;
Ryan Hamilton47cf9d12018-10-17 04:33:09536 const quic::QuicStreamId client_data_stream_id1_;
Yixin Wang079ad542018-01-11 04:06:05537 const bool client_headers_include_h2_stream_dependency_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55538
539 // order of destruction of these members matter
Ryan Hamilton8d9ee76e2018-05-29 23:52:52540 quic::MockClock clock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55541 MockQuicData mock_quic_data_;
542 std::unique_ptr<QuicChromiumConnectionHelper> helper_;
543 std::unique_ptr<QuicChromiumClientSession> session_;
544 std::unique_ptr<QuicChromiumClientSession::Handle> session_handle_;
545 std::unique_ptr<QuicProxyClientSocket> sock_;
546
547 BoundTestNetLog net_log_;
548
Ryan Hamilton8d9ee76e2018-05-29 23:52:52549 quic::test::MockSendAlgorithm* send_algorithm_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55550 scoped_refptr<TestTaskRunner> runner_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55551
552 std::unique_ptr<QuicChromiumAlarmFactory> alarm_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52553 testing::StrictMock<quic::test::MockQuicConnectionVisitor> visitor_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55554 TransportSecurityState transport_security_state_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52555 quic::QuicCryptoClientConfig crypto_config_;
556 quic::QuicClientPushPromiseIndex push_promise_index_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55557
Ryan Hamilton8d9ee76e2018-05-29 23:52:52558 const quic::QuicConnectionId connection_id_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55559 QuicTestPacketMaker client_maker_;
560 QuicTestPacketMaker server_maker_;
561 IPEndPoint peer_addr_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52562 quic::test::MockRandom random_generator_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55563 ProofVerifyDetailsChromium verify_details_;
564 MockCryptoClientStreamFactory crypto_client_stream_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52565 quic::QuicStreamOffset header_stream_offset_;
566 quic::QuicStreamOffset response_offset_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55567
568 std::string user_agent_;
569 HostPortPair proxy_host_port_;
570 HostPortPair endpoint_host_port_;
571 HttpAuthCache http_auth_cache_;
572 std::unique_ptr<MockHostResolverBase> host_resolver_;
573 std::unique_ptr<HttpAuthHandlerRegistryFactory> http_auth_handler_factory_;
574
575 TestCompletionCallback read_callback_;
576 scoped_refptr<IOBuffer> read_buf_;
577
578 TestCompletionCallback write_callback_;
579
580 DISALLOW_COPY_AND_ASSIGN(QuicProxyClientSocketTest);
581};
582
583TEST_P(QuicProxyClientSocketTest, ConnectSendsCorrectRequest) {
Zhongyi Shi32f2fd02018-04-16 18:23:43584 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
585 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
586 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55587 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
588 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52589 SYNCHRONOUS,
590 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55591
592 Initialize();
593
594 ASSERT_FALSE(sock_->IsConnected());
595
596 AssertConnectSucceeds();
597
598 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
599 ASSERT_TRUE(response != nullptr);
600 ASSERT_EQ(200, response->headers->response_code());
601}
602
603TEST_P(QuicProxyClientSocketTest, ConnectWithAuthRequested) {
Zhongyi Shi32f2fd02018-04-16 18:23:43604 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
605 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
606 mock_quic_data_.AddRead(ASYNC,
607 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55608 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
609 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52610 SYNCHRONOUS,
611 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55612
613 Initialize();
614
615 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
616
617 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
618 ASSERT_TRUE(response != nullptr);
619 ASSERT_EQ(407, response->headers->response_code());
620}
621
622TEST_P(QuicProxyClientSocketTest, ConnectWithAuthCredentials) {
Zhongyi Shi32f2fd02018-04-16 18:23:43623 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
624 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectAuthRequestPacket(2));
625 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55626 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
627 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52628 SYNCHRONOUS,
629 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55630
631 Initialize();
632
633 // Add auth to cache
634 const base::string16 kFoo(base::ASCIIToUTF16("foo"));
635 const base::string16 kBar(base::ASCIIToUTF16("bar"));
636 http_auth_cache_.Add(GURL(kProxyUrl), "MyRealm1", HttpAuth::AUTH_SCHEME_BASIC,
637 "Basic realm=MyRealm1", AuthCredentials(kFoo, kBar),
638 "/");
639
640 AssertConnectSucceeds();
641
642 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
643 ASSERT_TRUE(response != nullptr);
644 ASSERT_EQ(200, response->headers->response_code());
645}
646
647TEST_P(QuicProxyClientSocketTest, ConnectRedirects) {
Zhongyi Shi32f2fd02018-04-16 18:23:43648 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
649 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
650 mock_quic_data_.AddRead(ASYNC,
651 ConstructServerConnectRedirectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55652 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
653 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52654 SYNCHRONOUS,
655 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55656
657 Initialize();
658
659 AssertConnectFails(ERR_HTTPS_PROXY_TUNNEL_RESPONSE);
660
661 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
662 ASSERT_TRUE(response != nullptr);
663
664 const HttpResponseHeaders* headers = response->headers.get();
665 ASSERT_EQ(302, headers->response_code());
666 ASSERT_FALSE(headers->HasHeader("set-cookie"));
667 ASSERT_TRUE(headers->HasHeaderValue("content-length", "0"));
668
669 std::string location;
670 ASSERT_TRUE(headers->IsRedirect(&location));
671 ASSERT_EQ(location, kRedirectUrl);
672}
673
674TEST_P(QuicProxyClientSocketTest, ConnectFails) {
Zhongyi Shi32f2fd02018-04-16 18:23:43675 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
676 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Yixin Wang0d2c6b7e12017-08-16 21:12:55677 mock_quic_data_.AddRead(ASYNC, 0); // EOF
678
679 Initialize();
680
681 ASSERT_FALSE(sock_->IsConnected());
682
683 AssertConnectFails(ERR_QUIC_PROTOCOL_ERROR);
684
685 ASSERT_FALSE(sock_->IsConnected());
686}
687
688TEST_P(QuicProxyClientSocketTest, WasEverUsedReturnsCorrectValue) {
Zhongyi Shi32f2fd02018-04-16 18:23:43689 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
690 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
691 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55692 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
693 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52694 SYNCHRONOUS,
695 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55696
697 Initialize();
698
699 EXPECT_TRUE(sock_->WasEverUsed()); // Used due to crypto handshake
700 AssertConnectSucceeds();
701 EXPECT_TRUE(sock_->WasEverUsed());
702 sock_->Disconnect();
703 EXPECT_TRUE(sock_->WasEverUsed());
704}
705
706TEST_P(QuicProxyClientSocketTest, GetPeerAddressReturnsCorrectValues) {
Zhongyi Shi32f2fd02018-04-16 18:23:43707 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
708 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
709 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55710 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
711 mock_quic_data_.AddRead(ASYNC, 0); // EOF
712
713 Initialize();
714
715 IPEndPoint addr;
716 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
717
718 AssertConnectSucceeds();
719 EXPECT_TRUE(sock_->IsConnected());
720 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsOk());
721
722 ResumeAndRun();
723
724 EXPECT_FALSE(sock_->IsConnected());
725 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
726
727 sock_->Disconnect();
728
729 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
730}
731
732TEST_P(QuicProxyClientSocketTest, IsConnectedAndIdle) {
Zhongyi Shi32f2fd02018-04-16 18:23:43733 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
734 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
735 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55736 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
737
Renjief49758b2019-01-11 23:32:41738 quic::QuicString header = ConstructDataHeader(kLen1);
739 mock_quic_data_.AddRead(
740 ASYNC,
741 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43742 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55743 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52744 mock_quic_data_.AddWrite(
745 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55746
747 Initialize();
748
749 EXPECT_FALSE(sock_->IsConnectedAndIdle());
750
751 AssertConnectSucceeds();
752
753 EXPECT_TRUE(sock_->IsConnectedAndIdle());
754
755 // The next read is consumed and buffered.
756 ResumeAndRun();
757
758 EXPECT_FALSE(sock_->IsConnectedAndIdle());
759
760 AssertSyncReadEquals(kMsg1, kLen1);
761
762 EXPECT_TRUE(sock_->IsConnectedAndIdle());
763}
764
765TEST_P(QuicProxyClientSocketTest, GetTotalReceivedBytes) {
Zhongyi Shi32f2fd02018-04-16 18:23:43766 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
767 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
768 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55769 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
770
Renjief49758b2019-01-11 23:32:41771 quic::QuicString header = ConstructDataHeader(kLen333);
772 mock_quic_data_.AddRead(
773 ASYNC, ConstructServerDataPacket(
774 2, 0, header + quic::QuicString(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:43775 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55776 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52777 mock_quic_data_.AddWrite(
778 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55779
780 Initialize();
781
782 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
783
784 AssertConnectSucceeds();
785
786 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
787
788 // The next read is consumed and buffered.
789 ResumeAndRun();
790
791 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
792
793 // The payload from the single large data frame will be read across
794 // two different reads.
795 AssertSyncReadEquals(kMsg33, kLen33);
796
Renjief49758b2019-01-11 23:32:41797 EXPECT_EQ((int64_t)(kLen33 + header.length()),
798 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55799
800 AssertSyncReadEquals(kMsg3, kLen3);
801
Renjief49758b2019-01-11 23:32:41802 EXPECT_EQ((int64_t)(kLen333 + header.length()),
803 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55804}
805
Lily Chenf11e1292018-11-29 16:42:09806TEST_P(QuicProxyClientSocketTest, SetStreamPriority) {
807 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
808 mock_quic_data_.AddWrite(SYNCHRONOUS,
809 ConstructConnectRequestPacket(2, HIGHEST));
810 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
811 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
812 mock_quic_data_.AddWrite(
813 SYNCHRONOUS,
814 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
815
816 Initialize();
817
818 sock_->SetStreamPriority(HIGHEST);
819 AssertConnectSucceeds();
820}
Yixin Wang0d2c6b7e12017-08-16 21:12:55821
822TEST_P(QuicProxyClientSocketTest, WriteSendsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43823 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
824 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
825 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55826 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:41827 if (version_ == quic::QUIC_VERSION_99) {
828 quic::QuicString header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:41829 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:35830 SYNCHRONOUS,
831 ConstructAckAndMultipleDataFramesPacket(
832 3, 1, 1, 1, 0, {header, quic::QuicString(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41833 quic::QuicString header2 = ConstructDataHeader(kLen2);
Renjied172e812019-01-16 05:12:35834 mock_quic_data_.AddWrite(SYNCHRONOUS,
835 ConstructMultipleDataFramesPacket(
836 4, kLen1 + header.length(),
837 {header2, quic::QuicString(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:41838 mock_quic_data_.AddWrite(
839 SYNCHRONOUS,
Renjied172e812019-01-16 05:12:35840 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED,
Renjief49758b2019-01-11 23:32:41841 kLen1 + kLen2 + header.length() + header2.length()));
842 } else {
843 mock_quic_data_.AddWrite(
844 SYNCHRONOUS, ConstructAckAndDataPacket(3, 1, 1, 1, 0,
845 quic::QuicString(kMsg1, kLen1)));
846 mock_quic_data_.AddWrite(
847 SYNCHRONOUS,
848 ConstructDataPacket(4, kLen1, quic::QuicString(kMsg2, kLen2)));
849 mock_quic_data_.AddWrite(
850 SYNCHRONOUS,
851 ConstructRstPacket(5, quic::QUIC_STREAM_CANCELLED, kLen1 + kLen2));
852 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55853
854 Initialize();
855
856 AssertConnectSucceeds();
857
858 AssertSyncWriteSucceeds(kMsg1, kLen1);
859 AssertSyncWriteSucceeds(kMsg2, kLen2);
860}
861
862TEST_P(QuicProxyClientSocketTest, WriteSplitsLargeDataIntoMultiplePackets) {
Renjief49758b2019-01-11 23:32:41863 int write_packet_index = 1;
864 mock_quic_data_.AddWrite(SYNCHRONOUS,
865 ConstructSettingsPacket(write_packet_index++));
866 mock_quic_data_.AddWrite(SYNCHRONOUS,
867 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:43868 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55869 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:41870 quic::QuicString header = ConstructDataHeader(kLen1);
871 if (version_ != quic::QUIC_VERSION_99) {
872 mock_quic_data_.AddWrite(
873 SYNCHRONOUS, ConstructAckAndDataPacket(write_packet_index++, 1, 1, 1, 0,
874 quic::QuicString(kMsg1, kLen1)));
875 } else {
Renjied172e812019-01-16 05:12:35876 mock_quic_data_.AddWrite(SYNCHRONOUS,
877 ConstructAckAndMultipleDataFramesPacket(
878 write_packet_index++, 1, 1, 1, 0,
879 {header, quic::QuicString(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:41880 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55881
882 // Expect |kNumDataPackets| data packets, each containing the max possible
883 // amount of data.
Renjied172e812019-01-16 05:12:35884 int numDataPackets = 3;
885 std::string data(numDataPackets * quic::kDefaultMaxPacketSize, 'x');
Renjief49758b2019-01-11 23:32:41886 quic::QuicStreamOffset offset = kLen1 + header.length();
Renjied172e812019-01-16 05:12:35887
Renjief49758b2019-01-11 23:32:41888 if (version_ == quic::QUIC_VERSION_99) {
Renjied172e812019-01-16 05:12:35889 numDataPackets++;
Renjief49758b2019-01-11 23:32:41890 }
Renjied172e812019-01-16 05:12:35891 size_t total_data_length = 0;
892 for (int i = 0; i < numDataPackets; ++i) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55893 size_t max_packet_data_length = GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52894 quic::kDefaultMaxPacketSize, version_, !kIncludeVersion,
895 !kIncludeDiversificationNonce, quic::PACKET_8BYTE_CONNECTION_ID,
896 quic::PACKET_1BYTE_PACKET_NUMBER, offset);
Renjied172e812019-01-16 05:12:35897 if (version_ == quic::QUIC_VERSION_99 && i == 0) {
898 // 3973 is the data frame length from packet length.
899 quic::QuicString header2 = ConstructDataHeader(3973);
900 mock_quic_data_.AddWrite(
901 SYNCHRONOUS,
902 ConstructMultipleDataFramesPacket(
903 write_packet_index++, offset,
904 {header2,
905 quic::QuicString(data.c_str(), max_packet_data_length - 7)}));
906 offset += max_packet_data_length - header2.length() - 1;
907 } else if (version_ == quic::QUIC_VERSION_99 && i == numDataPackets - 1) {
908 mock_quic_data_.AddWrite(
909 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, offset,
910 quic::QuicString(data.c_str(), 7)));
911 offset += 7;
912 } else {
913 mock_quic_data_.AddWrite(
914 SYNCHRONOUS,
915 ConstructDataPacket(
916 write_packet_index++, offset,
917 quic::QuicString(data.c_str(), max_packet_data_length)));
918 offset += max_packet_data_length;
919 }
920 if (i != 3) {
921 total_data_length += max_packet_data_length;
922 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55923 }
Renjied172e812019-01-16 05:12:35924
Yixin Wang0d2c6b7e12017-08-16 21:12:55925 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:41926 SYNCHRONOUS, ConstructRstPacket(write_packet_index++,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52927 quic::QUIC_STREAM_CANCELLED, offset));
Yixin Wang0d2c6b7e12017-08-16 21:12:55928
929 Initialize();
930
931 AssertConnectSucceeds();
932
933 // Make a small write. An ACK and STOP_WAITING will be bundled. This prevents
934 // ACK and STOP_WAITING from being bundled with the subsequent large write.
935 // This allows the test code for computing the size of data sent in each
936 // packet to not become too complicated.
937 AssertSyncWriteSucceeds(kMsg1, kLen1);
938
939 // Make large write that should be split up
940 AssertSyncWriteSucceeds(data.c_str(), total_data_length);
941}
942
943// ----------- Read
944
945TEST_P(QuicProxyClientSocketTest, ReadReadsDataInDataFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:43946 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
947 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
948 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55949 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
950
Renjief49758b2019-01-11 23:32:41951 quic::QuicString header = ConstructDataHeader(kLen1);
952 mock_quic_data_.AddRead(
953 ASYNC,
954 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43955 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55956 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52957 mock_quic_data_.AddWrite(
958 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:55959
960 Initialize();
961
962 AssertConnectSucceeds();
963
964 ResumeAndRun();
965 AssertSyncReadEquals(kMsg1, kLen1);
966}
967
968TEST_P(QuicProxyClientSocketTest, ReadDataFromBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:43969 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
970 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
971 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55972 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
973
Renjief49758b2019-01-11 23:32:41974 quic::QuicString header = ConstructDataHeader(kLen1);
975 mock_quic_data_.AddRead(
976 ASYNC,
977 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:43978 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55979 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
980
Renjief49758b2019-01-11 23:32:41981 quic::QuicString header2 = ConstructDataHeader(kLen2);
982 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
983 3, kLen1 + header.length(),
984 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:55985 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
986
987 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52988 SYNCHRONOUS,
989 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55990
991 Initialize();
992
993 AssertConnectSucceeds();
994
995 ResumeAndRun();
996 AssertSyncReadEquals(kMsg1, kLen1);
997
998 ResumeAndRun();
999 AssertSyncReadEquals(kMsg2, kLen2);
1000}
1001
1002TEST_P(QuicProxyClientSocketTest, ReadDataMultipleBufferedFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431003 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1004 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1005 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551006 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1007
Renjief49758b2019-01-11 23:32:411008 quic::QuicString header = ConstructDataHeader(kLen1);
1009 mock_quic_data_.AddRead(
1010 ASYNC,
1011 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431012 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411013 quic::QuicString header2 = ConstructDataHeader(kLen2);
1014 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1015 3, kLen1 + header.length(),
1016 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551017 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1018
1019 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521020 SYNCHRONOUS,
1021 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551022
1023 Initialize();
1024
1025 AssertConnectSucceeds();
1026
1027 // The next two reads are consumed and buffered.
1028 ResumeAndRun();
1029
1030 AssertSyncReadEquals(kMsg1, kLen1);
1031 AssertSyncReadEquals(kMsg2, kLen2);
1032}
1033
1034TEST_P(QuicProxyClientSocketTest, LargeReadWillMergeDataFromDifferentFrames) {
Zhongyi Shi32f2fd02018-04-16 18:23:431035 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1036 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1037 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551038 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1039
Renjief49758b2019-01-11 23:32:411040 quic::QuicString header = ConstructDataHeader(kLen3);
1041 mock_quic_data_.AddRead(
1042 ASYNC,
1043 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg3, kLen3)));
Zhongyi Shi32f2fd02018-04-16 18:23:431044 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411045 quic::QuicString header2 = ConstructDataHeader(kLen3);
1046 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1047 3, kLen3 + header.length(),
1048 header2 + quic::QuicString(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551049 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1050
1051 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521052 SYNCHRONOUS,
1053 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551054
1055 Initialize();
1056
1057 AssertConnectSucceeds();
1058
1059 // The next two reads are consumed and buffered.
1060 ResumeAndRun();
1061 // The payload from two data frames, each with kMsg3 will be combined
1062 // together into a single read().
1063 AssertSyncReadEquals(kMsg33, kLen33);
1064}
1065
1066TEST_P(QuicProxyClientSocketTest, MultipleShortReadsThenMoreRead) {
Zhongyi Shi32f2fd02018-04-16 18:23:431067 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1068 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1069 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551070 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1071
1072 int offset = 0;
1073
Renjief49758b2019-01-11 23:32:411074 quic::QuicString header = ConstructDataHeader(kLen1);
1075 mock_quic_data_.AddRead(
1076 ASYNC, ConstructServerDataPacket(
1077 2, offset, header + quic::QuicString(kMsg1, kLen1)));
1078 offset += kLen1 + header.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431079 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551080
Renjief49758b2019-01-11 23:32:411081 quic::QuicString header2 = ConstructDataHeader(kLen3);
1082 mock_quic_data_.AddRead(
1083 ASYNC, ConstructServerDataPacket(
1084 3, offset, header2 + quic::QuicString(kMsg3, kLen3)));
1085 offset += kLen3 + header2.length();
1086 mock_quic_data_.AddRead(
1087 ASYNC, ConstructServerDataPacket(
1088 4, offset, header2 + quic::QuicString(kMsg3, kLen3)));
1089 offset += kLen3 + header2.length();
Zhongyi Shi32f2fd02018-04-16 18:23:431090 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(4, 4, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551091
Renjief49758b2019-01-11 23:32:411092 quic::QuicString header3 = ConstructDataHeader(kLen2);
1093 mock_quic_data_.AddRead(
1094 ASYNC, ConstructServerDataPacket(
1095 5, offset, header3 + quic::QuicString(kMsg2, kLen2)));
1096 offset += kLen2 + header3.length();
Yixin Wang0d2c6b7e12017-08-16 21:12:551097 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1098
1099 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521100 SYNCHRONOUS,
1101 ConstructAckAndRstPacket(5, quic::QUIC_STREAM_CANCELLED, 5, 5, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551102
1103 Initialize();
1104
1105 AssertConnectSucceeds();
1106
1107 // The next 4 reads are consumed and buffered.
1108 ResumeAndRun();
1109
1110 AssertSyncReadEquals(kMsg1, kLen1);
1111 // The payload from two data frames, each with kMsg3 will be combined
1112 // together into a single read().
1113 AssertSyncReadEquals(kMsg33, kLen33);
1114 AssertSyncReadEquals(kMsg2, kLen2);
1115}
1116
1117TEST_P(QuicProxyClientSocketTest, ReadWillSplitDataFromLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431118 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1119 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1120 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551121 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1122
Renjief49758b2019-01-11 23:32:411123 quic::QuicString header = ConstructDataHeader(kLen1);
1124 mock_quic_data_.AddRead(
1125 ASYNC,
1126 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431127 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411128 quic::QuicString header2 = ConstructDataHeader(kLen33);
1129 mock_quic_data_.AddRead(
1130 ASYNC,
1131 ConstructServerDataPacket(3, kLen1 + header.length(),
1132 header2 + quic::QuicString(kMsg33, kLen33)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551133 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1134
1135 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521136 SYNCHRONOUS,
1137 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551138
1139 Initialize();
1140
1141 AssertConnectSucceeds();
1142
1143 // The next 2 reads are consumed and buffered.
1144 ResumeAndRun();
1145
1146 AssertSyncReadEquals(kMsg1, kLen1);
1147 // The payload from the single large data frame will be read across
1148 // two different reads.
1149 AssertSyncReadEquals(kMsg3, kLen3);
1150 AssertSyncReadEquals(kMsg3, kLen3);
1151}
1152
1153TEST_P(QuicProxyClientSocketTest, MultipleReadsFromSameLargeFrame) {
Zhongyi Shi32f2fd02018-04-16 18:23:431154 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1155 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1156 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551157 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1158
Renjief49758b2019-01-11 23:32:411159 quic::QuicString header = ConstructDataHeader(kLen333);
1160 mock_quic_data_.AddRead(
1161 ASYNC, ConstructServerDataPacket(
1162 2, 0, header + quic::QuicString(kMsg333, kLen333)));
Zhongyi Shi32f2fd02018-04-16 18:23:431163 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551164 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1165
Ryan Hamilton8d9ee76e2018-05-29 23:52:521166 mock_quic_data_.AddWrite(
1167 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551168
1169 Initialize();
1170
1171 AssertConnectSucceeds();
1172
1173 // The next read is consumed and buffered.
1174 ResumeAndRun();
1175
1176 // The payload from the single large data frame will be read across
1177 // two different reads.
1178 AssertSyncReadEquals(kMsg33, kLen33);
1179
1180 // Now attempt to do a read of more data than remains buffered
Victor Costan9c7302b2018-08-27 16:39:441181 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kLen33);
Bence Békyd8a21fc32018-06-27 18:29:581182 ASSERT_EQ(kLen3, sock_->Read(buf.get(), kLen33, CompletionOnceCallback()));
Ryan Hamilton0239aac2018-05-19 00:03:131183 ASSERT_EQ(spdy::SpdyString(kMsg3, kLen3),
1184 spdy::SpdyString(buf->data(), kLen3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551185 ASSERT_TRUE(sock_->IsConnected());
1186}
1187
1188TEST_P(QuicProxyClientSocketTest, ReadAuthResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431189 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1190 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1191 mock_quic_data_.AddRead(ASYNC,
1192 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551193 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1194
Renjief49758b2019-01-11 23:32:411195 quic::QuicString header = ConstructDataHeader(kLen1);
1196 mock_quic_data_.AddRead(
1197 ASYNC,
1198 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431199 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411200 quic::QuicString header2 = ConstructDataHeader(kLen2);
1201 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1202 3, kLen1 + header.length(),
1203 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551204 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1205
1206 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521207 SYNCHRONOUS,
1208 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551209
1210 Initialize();
1211
1212 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
1213
1214 // The next two reads are consumed and buffered.
1215 ResumeAndRun();
1216
1217 AssertSyncReadEquals(kMsg1, kLen1);
1218 AssertSyncReadEquals(kMsg2, kLen2);
1219}
1220
1221TEST_P(QuicProxyClientSocketTest, ReadErrorResponseBody) {
Zhongyi Shi32f2fd02018-04-16 18:23:431222 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1223 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1224 mock_quic_data_.AddRead(ASYNC,
1225 ConstructServerConnectErrorReplyPacket(1, !kFin));
Renjief49758b2019-01-11 23:32:411226 quic::QuicString header = ConstructDataHeader(kLen1);
1227 mock_quic_data_.AddRead(
1228 SYNCHRONOUS,
1229 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431230 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Renjief49758b2019-01-11 23:32:411231 quic::QuicString header2 = ConstructDataHeader(kLen2);
1232 mock_quic_data_.AddRead(
1233 SYNCHRONOUS,
1234 ConstructServerDataPacket(3, kLen1 + header.length(),
1235 header2 + quic::QuicString(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551236 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1237
1238 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521239 SYNCHRONOUS,
1240 ConstructAckAndRstPacket(4, quic::QUIC_STREAM_CANCELLED, 3, 3, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551241 Initialize();
1242
1243 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
1244}
1245
1246// ----------- Reads and Writes
1247
1248TEST_P(QuicProxyClientSocketTest, AsyncReadAroundWrite) {
Renjief49758b2019-01-11 23:32:411249 int write_packet_index = 1;
1250 mock_quic_data_.AddWrite(SYNCHRONOUS,
1251 ConstructSettingsPacket(write_packet_index++));
1252 mock_quic_data_.AddWrite(SYNCHRONOUS,
1253 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431254 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551255 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1256
Renjief49758b2019-01-11 23:32:411257 quic::QuicString header = ConstructDataHeader(kLen1);
1258 mock_quic_data_.AddRead(
1259 ASYNC,
1260 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431261 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjief49758b2019-01-11 23:32:411262 ConstructAckPacket(write_packet_index++, 2, 1, 1));
1263
1264 quic::QuicString header2 = ConstructDataHeader(kLen2);
1265 if (version_ == quic::QUIC_VERSION_99) {
Renjied172e812019-01-16 05:12:351266 mock_quic_data_.AddWrite(SYNCHRONOUS,
1267 ConstructMultipleDataFramesPacket(
1268 write_packet_index++, 0,
1269 {header2, quic::QuicString(kMsg2, kLen2)}));
1270 } else {
Renjief49758b2019-01-11 23:32:411271 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:411272 SYNCHRONOUS, ConstructDataPacket(write_packet_index++, header2.length(),
1273 quic::QuicString(kMsg2, kLen2)));
Renjied172e812019-01-16 05:12:351274 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551275
1276 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1277
Renjief49758b2019-01-11 23:32:411278 quic::QuicString header3 = ConstructDataHeader(kLen3);
1279 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1280 3, kLen1 + header.length(),
1281 header3 + quic::QuicString(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551282 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1283
1284 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:411285 SYNCHRONOUS, ConstructAckAndRstPacket(write_packet_index++,
1286 quic::QUIC_STREAM_CANCELLED, 3, 3,
1287 1, kLen2 + header2.length()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551288
1289 Initialize();
1290
1291 AssertConnectSucceeds();
1292
1293 ResumeAndRun();
1294
1295 AssertSyncReadEquals(kMsg1, kLen1);
1296
1297 AssertReadStarts(kMsg3, kLen3);
1298 // Read should block until after the write succeeds.
1299
1300 AssertSyncWriteSucceeds(kMsg2, kLen2);
1301
1302 ASSERT_FALSE(read_callback_.have_result());
1303 ResumeAndRun();
1304
1305 // Now the read will return.
1306 AssertReadReturns(kMsg3, kLen3);
1307}
1308
1309TEST_P(QuicProxyClientSocketTest, AsyncWriteAroundReads) {
Renjied172e812019-01-16 05:12:351310 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1311 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
Zhongyi Shi32f2fd02018-04-16 18:23:431312 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551313 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1314
Renjief49758b2019-01-11 23:32:411315 quic::QuicString header = ConstructDataHeader(kLen1);
1316 mock_quic_data_.AddRead(
1317 ASYNC,
1318 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Renjied172e812019-01-16 05:12:351319 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551320 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1321
Renjief49758b2019-01-11 23:32:411322 quic::QuicString header2 = ConstructDataHeader(kLen3);
1323 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1324 3, kLen1 + header.length(),
1325 header2 + quic::QuicString(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551326 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1327
1328 mock_quic_data_.AddWrite(ASYNC, ERR_IO_PENDING); // Pause
1329
Renjief49758b2019-01-11 23:32:411330 quic::QuicString header3 = ConstructDataHeader(kLen2);
1331 if (version_ != quic::QUIC_VERSION_99) {
1332 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351333 ASYNC, ConstructDataPacket(4, 0, quic::QuicString(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411334 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351335 SYNCHRONOUS, ConstructAckAndDataPacket(5, 3, 3, 1, kLen2,
1336 quic::QuicString(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411337 } else {
1338 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351339 ASYNC, ConstructMultipleDataFramesPacket(
1340 4, 0, {header3, quic::QuicString(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:411341 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351342 ASYNC,
1343 ConstructAckAndDataPacket(5, 3, 3, 1, header3.length() + kLen2,
1344 header3 + quic::QuicString(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411345 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551346
Renjied172e812019-01-16 05:12:351347 mock_quic_data_.AddWrite(
1348 SYNCHRONOUS, ConstructRstPacket(6, quic::QUIC_STREAM_CANCELLED,
1349 kLen2 + kLen2 + 2 * header3.length()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551350
1351 Initialize();
1352
1353 AssertConnectSucceeds();
1354
1355 ResumeAndRun();
1356 AssertSyncReadEquals(kMsg1, kLen1);
1357
1358 // Write should block until the next read completes.
1359 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1360 // asynchronous starting with the second time it's called while the UDP socket
1361 // is write-blocked. Therefore, at least two writes need to be called on
1362 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351363 AssertWriteReturns(kMsg2, kLen2, kLen2);
Yixin Wang0d2c6b7e12017-08-16 21:12:551364 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1365
1366 AssertAsyncReadEquals(kMsg3, kLen3);
1367
1368 ASSERT_FALSE(write_callback_.have_result());
1369
1370 // Now the write will complete
1371 ResumeAndRun();
Yixin Wangdbbd8752018-01-17 21:50:021372 EXPECT_EQ(kLen2, write_callback_.WaitForResult());
Yixin Wang0d2c6b7e12017-08-16 21:12:551373}
1374
1375// ----------- Reading/Writing on Closed socket
1376
1377// Reading from an already closed socket should return 0
1378TEST_P(QuicProxyClientSocketTest, ReadOnClosedSocketReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431379 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1380 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1381 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551382 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1383
1384 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1385
1386 Initialize();
1387
1388 AssertConnectSucceeds();
1389
1390 ResumeAndRun();
1391
1392 ASSERT_FALSE(sock_->IsConnected());
Bence Békyd8a21fc32018-06-27 18:29:581393 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
1394 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
1395 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551396 ASSERT_FALSE(sock_->IsConnectedAndIdle());
1397}
1398
1399// Read pending when socket is closed should return 0
1400TEST_P(QuicProxyClientSocketTest, PendingReadOnCloseReturnsZero) {
Zhongyi Shi32f2fd02018-04-16 18:23:431401 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1402 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1403 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551404 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1405
1406 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1407
1408 Initialize();
1409
1410 AssertConnectSucceeds();
1411
1412 AssertReadStarts(kMsg1, kLen1);
1413
1414 ResumeAndRun();
1415
1416 ASSERT_EQ(0, read_callback_.WaitForResult());
1417}
1418
1419// Reading from a disconnected socket is an error
1420TEST_P(QuicProxyClientSocketTest, ReadOnDisconnectSocketReturnsNotConnected) {
Zhongyi Shi32f2fd02018-04-16 18:23:431421 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1422 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1423 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551424 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1425 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521426 SYNCHRONOUS,
1427 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551428
1429 Initialize();
1430
1431 AssertConnectSucceeds();
1432
1433 sock_->Disconnect();
1434
1435 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581436 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551437}
1438
1439// Reading data after receiving FIN should return buffered data received before
1440// FIN, then 0.
1441TEST_P(QuicProxyClientSocketTest, ReadAfterFinReceivedReturnsBufferedData) {
Zhongyi Shi32f2fd02018-04-16 18:23:431442 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1443 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1444 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551445 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1446
Renjief49758b2019-01-11 23:32:411447 quic::QuicString header = ConstructDataHeader(kLen1);
Zhongyi Shi32f2fd02018-04-16 18:23:431448 mock_quic_data_.AddRead(ASYNC,
Renjief49758b2019-01-11 23:32:411449 ConstructServerDataFinPacket(
1450 2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431451 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551452 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521453 mock_quic_data_.AddWrite(
1454 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551455
1456 Initialize();
1457
1458 AssertConnectSucceeds();
1459
1460 ResumeAndRun();
1461
1462 AssertSyncReadEquals(kMsg1, kLen1);
Bence Békyd8a21fc32018-06-27 18:29:581463 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
1464 ASSERT_EQ(0, sock_->Read(NULL, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551465
1466 sock_->Disconnect();
1467 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581468 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551469}
1470
1471// Calling Write() on a closed socket is an error.
1472TEST_P(QuicProxyClientSocketTest, WriteOnClosedStream) {
Zhongyi Shi32f2fd02018-04-16 18:23:431473 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1474 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1475 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551476 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1477
1478 mock_quic_data_.AddRead(ASYNC, 0); // EOF
1479
1480 Initialize();
1481
1482 AssertConnectSucceeds();
1483
1484 ResumeAndRun();
1485
1486 AssertWriteReturns(kMsg1, kLen1, ERR_QUIC_PROTOCOL_ERROR);
1487}
1488
1489// Calling Write() on a disconnected socket is an error.
1490TEST_P(QuicProxyClientSocketTest, WriteOnDisconnectedSocket) {
Zhongyi Shi32f2fd02018-04-16 18:23:431491 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1492 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1493 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551494 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1495 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521496 SYNCHRONOUS,
1497 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551498
1499 Initialize();
1500
1501 AssertConnectSucceeds();
1502
1503 sock_->Disconnect();
1504
1505 AssertWriteReturns(kMsg1, kLen1, ERR_SOCKET_NOT_CONNECTED);
1506}
1507
1508// If the socket is closed with a pending Write(), the callback should be called
1509// with the same error the session was closed with.
1510TEST_P(QuicProxyClientSocketTest, WritePendingOnClose) {
Zhongyi Shi32f2fd02018-04-16 18:23:431511 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1512 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1513 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551514 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1515 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1516
1517 Initialize();
1518
1519 AssertConnectSucceeds();
1520
1521 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1522 // asynchronous starting with the second time it's called while the UDP socket
1523 // is write-blocked. Therefore, at least two writes need to be called on
1524 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351525 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551526
1527 // This second write will be async. This is the pending write that's being
1528 // tested.
1529 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1530
1531 // Make sure the write actually starts.
1532 base::RunLoop().RunUntilIdle();
1533
Ryan Hamilton8d9ee76e2018-05-29 23:52:521534 session_->CloseSessionOnError(ERR_CONNECTION_CLOSED,
Renjieba55fae2018-09-20 03:05:161535 quic::QUIC_INTERNAL_ERROR,
1536 quic::ConnectionCloseBehavior::SILENT_CLOSE);
Yixin Wang0d2c6b7e12017-08-16 21:12:551537
1538 EXPECT_THAT(write_callback_.WaitForResult(), IsError(ERR_CONNECTION_CLOSED));
1539}
1540
1541TEST_P(QuicProxyClientSocketTest, DisconnectWithWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431542 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1543 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1544 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551545 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1546 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1547
1548 Initialize();
1549
1550 AssertConnectSucceeds();
1551
1552 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1553 // asynchronous starting with the second time it's called while the UDP socket
1554 // is write-blocked. Therefore, at least two writes need to be called on
1555 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351556 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551557
1558 // This second write will be async. This is the pending write that's being
1559 // tested.
1560 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1561
1562 // Make sure the write actually starts.
1563 base::RunLoop().RunUntilIdle();
1564
1565 sock_->Disconnect();
1566 EXPECT_FALSE(sock_->IsConnected());
1567
1568 base::RunLoop().RunUntilIdle();
1569
1570 EXPECT_FALSE(sock_->IsConnected());
1571 EXPECT_FALSE(write_callback_.have_result());
1572}
1573
1574// If the socket is Disconnected with a pending Read(), the callback
1575// should not be called.
1576TEST_P(QuicProxyClientSocketTest, DisconnectWithReadPending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431577 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1578 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1579 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551580 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1581 mock_quic_data_.AddWrite(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521582 SYNCHRONOUS,
1583 ConstructAckAndRstPacket(3, quic::QUIC_STREAM_CANCELLED, 1, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551584
1585 Initialize();
1586
1587 AssertConnectSucceeds();
1588
1589 EXPECT_TRUE(sock_->IsConnected());
1590
1591 AssertReadStarts(kMsg1, kLen1);
1592
1593 sock_->Disconnect();
1594 EXPECT_FALSE(sock_->IsConnected());
1595
1596 base::RunLoop().RunUntilIdle();
1597
1598 EXPECT_FALSE(sock_->IsConnected());
1599 EXPECT_FALSE(read_callback_.have_result());
1600}
1601
1602// If the socket is Reset when both a read and write are pending,
1603// both should be called back.
1604TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePending) {
Zhongyi Shi32f2fd02018-04-16 18:23:431605 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1606 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1607 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551608 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1609
1610 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521611 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang2bea3cf2017-11-09 18:11:031612 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:411613 quic::QuicString header = ConstructDataHeader(kLen2);
1614 if (version_ != quic::QUIC_VERSION_99) {
1615 mock_quic_data_.AddWrite(
1616 ASYNC, ConstructAckAndDataPacket(3, 1, 1, 1, 0,
1617 quic::QuicString(kMsg2, kLen2)));
1618 mock_quic_data_.AddWrite(
1619 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1620 2, 2, 1, kLen2));
1621 } else {
Renjief49758b2019-01-11 23:32:411622 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351623 ASYNC, ConstructAckAndMultipleDataFramesPacket(
1624 3, 1, 1, 1, 0, {header, quic::QuicString(kMsg2, kLen2)}));
1625 mock_quic_data_.AddWrite(
1626 SYNCHRONOUS,
1627 ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2, 1,
1628 header.length() + kLen2));
Renjief49758b2019-01-11 23:32:411629 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551630
1631 Initialize();
1632
1633 AssertConnectSucceeds();
1634
1635 EXPECT_TRUE(sock_->IsConnected());
1636
1637 AssertReadStarts(kMsg1, kLen1);
1638
1639 // Write should block until the next read completes.
1640 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1641 // asynchronous starting with the second time it's called while the UDP socket
1642 // is write-blocked. Therefore, at least two writes need to be called on
1643 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351644 AssertWriteReturns(kMsg2, kLen2, kLen2);
Renjief49758b2019-01-11 23:32:411645
Yixin Wang0d2c6b7e12017-08-16 21:12:551646 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1647
1648 ResumeAndRun();
1649
1650 EXPECT_TRUE(read_callback_.have_result());
1651 EXPECT_TRUE(write_callback_.have_result());
1652}
1653
1654// Makes sure the proxy client socket's source gets the expected NetLog events
1655// and only the expected NetLog events (No SpdySession events).
1656TEST_P(QuicProxyClientSocketTest, NetLog) {
Zhongyi Shi32f2fd02018-04-16 18:23:431657 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1658 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1659 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551660 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1661
Renjief49758b2019-01-11 23:32:411662 quic::QuicString header = ConstructDataHeader(kLen1);
1663 mock_quic_data_.AddRead(
1664 ASYNC,
1665 ConstructServerDataPacket(2, 0, header + quic::QuicString(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431666 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructAckPacket(3, 2, 1, 1));
Yixin Wang2bea3cf2017-11-09 18:11:031667 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Ryan Hamilton8d9ee76e2018-05-29 23:52:521668 mock_quic_data_.AddWrite(
1669 SYNCHRONOUS, ConstructRstPacket(4, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551670
1671 Initialize();
1672
1673 AssertConnectSucceeds();
1674
1675 ResumeAndRun();
1676 AssertSyncReadEquals(kMsg1, kLen1);
1677
1678 NetLogSource sock_source = sock_->NetLog().source();
1679 sock_.reset();
1680
1681 TestNetLogEntry::List entry_list;
1682 net_log_.GetEntriesForSource(sock_source, &entry_list);
1683
1684 ASSERT_EQ(entry_list.size(), 10u);
1685 EXPECT_TRUE(
1686 LogContainsBeginEvent(entry_list, 0, NetLogEventType::SOCKET_ALIVE));
1687 EXPECT_TRUE(LogContainsEvent(entry_list, 1,
1688 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
1689 NetLogEventPhase::NONE));
1690 EXPECT_TRUE(LogContainsBeginEvent(
1691 entry_list, 2, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1692 EXPECT_TRUE(LogContainsEvent(
1693 entry_list, 3, NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
1694 NetLogEventPhase::NONE));
1695 EXPECT_TRUE(LogContainsEndEvent(
1696 entry_list, 4, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1697 EXPECT_TRUE(LogContainsBeginEvent(
1698 entry_list, 5, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1699 EXPECT_TRUE(LogContainsEvent(
1700 entry_list, 6,
1701 NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
1702 NetLogEventPhase::NONE));
1703 EXPECT_TRUE(LogContainsEndEvent(
1704 entry_list, 7, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1705 EXPECT_TRUE(LogContainsEvent(entry_list, 8,
1706 NetLogEventType::SOCKET_BYTES_RECEIVED,
1707 NetLogEventPhase::NONE));
1708 EXPECT_TRUE(
1709 LogContainsEndEvent(entry_list, 9, NetLogEventType::SOCKET_ALIVE));
1710}
1711
Bence Béky8ddc2492018-06-13 01:02:041712// A helper class that will delete |sock| when the callback is invoked.
Yixin Wang0d2c6b7e12017-08-16 21:12:551713class DeleteSockCallback : public TestCompletionCallbackBase {
1714 public:
1715 explicit DeleteSockCallback(std::unique_ptr<QuicProxyClientSocket>* sock)
Bence Béky8ddc2492018-06-13 01:02:041716 : sock_(sock) {}
Yixin Wang0d2c6b7e12017-08-16 21:12:551717
1718 ~DeleteSockCallback() override {}
1719
Bence Béky8ddc2492018-06-13 01:02:041720 CompletionOnceCallback callback() {
1721 return base::BindOnce(&DeleteSockCallback::OnComplete,
1722 base::Unretained(this));
1723 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551724
1725 private:
1726 void OnComplete(int result) {
1727 sock_->reset(NULL);
1728 SetResult(result);
1729 }
1730
1731 std::unique_ptr<QuicProxyClientSocket>* sock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:551732
1733 DISALLOW_COPY_AND_ASSIGN(DeleteSockCallback);
1734};
1735
1736// If the socket is reset when both a read and write are pending, and the
1737// read callback causes the socket to be deleted, the write callback should
1738// not be called.
1739TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePendingDelete) {
Zhongyi Shi32f2fd02018-04-16 18:23:431740 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructSettingsPacket(1));
1741 mock_quic_data_.AddWrite(SYNCHRONOUS, ConstructConnectRequestPacket(2));
1742 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551743 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1744
1745 mock_quic_data_.AddRead(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521746 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED, 0));
Yixin Wang0d2c6b7e12017-08-16 21:12:551747 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Renjief49758b2019-01-11 23:32:411748 if (version_ != quic::QUIC_VERSION_99) {
1749 mock_quic_data_.AddWrite(
1750 ASYNC, ConstructAckAndDataPacket(3, 1, 1, 1, 0,
1751 quic::QuicString(kMsg1, kLen1)));
1752 mock_quic_data_.AddWrite(
1753 SYNCHRONOUS, ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT,
1754 2, 2, 1, kLen1));
1755 } else {
1756 quic::QuicString header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411757 mock_quic_data_.AddWrite(
Renjied172e812019-01-16 05:12:351758 ASYNC, ConstructAckAndMultipleDataFramesPacket(
1759 3, 1, 1, 1, 0, {header, quic::QuicString(kMsg1, kLen1)}));
1760 mock_quic_data_.AddWrite(
1761 SYNCHRONOUS,
1762 ConstructAckAndRstPacket(4, quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2, 1,
1763 header.length() + kLen1));
Renjief49758b2019-01-11 23:32:411764 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551765
1766 Initialize();
1767
1768 AssertConnectSucceeds();
1769
1770 EXPECT_TRUE(sock_->IsConnected());
1771
1772 DeleteSockCallback read_callback(&sock_);
Victor Costan9c7302b2018-08-27 16:39:441773 scoped_refptr<IOBuffer> read_buf = base::MakeRefCounted<IOBuffer>(kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551774 ASSERT_EQ(ERR_IO_PENDING,
1775 sock_->Read(read_buf.get(), kLen1, read_callback.callback()));
1776
1777 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1778 // asynchronous starting with the second time it's called while the UDP socket
1779 // is write-blocked. Therefore, at least two writes need to be called on
1780 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351781 AssertWriteReturns(kMsg1, kLen1, kLen1);
Renjief49758b2019-01-11 23:32:411782
Yixin Wang0d2c6b7e12017-08-16 21:12:551783 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1784
1785 ResumeAndRun();
1786
1787 EXPECT_FALSE(sock_.get());
1788
1789 EXPECT_EQ(0, read_callback.WaitForResult());
1790 EXPECT_FALSE(write_callback_.have_result());
1791}
1792
Victor Costane635086f2019-01-27 05:20:301793INSTANTIATE_TEST_SUITE_P(
Bence Békyce380cb2018-04-26 23:39:551794 VersionIncludeStreamDependencySequence,
Yixin Wang079ad542018-01-11 04:06:051795 QuicProxyClientSocketTest,
Ryan Hamilton8d9ee76e2018-05-29 23:52:521796 ::testing::Combine(
1797 ::testing::ValuesIn(quic::AllSupportedTransportVersions()),
1798 ::testing::Bool()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551799
1800} // namespace test
Dan Zhangf11470172017-09-18 22:02:091801} // namespace net