blob: 354b2c8030a102b123ad210878f29f47d7c3db6b [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"
Keishi Hattori0e45c022021-11-27 09:25:5214#include "base/memory/raw_ptr.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5515#include "base/run_loop.h"
Victor Vasiliev62c09dc2020-11-06 18:18:2916#include "base/strings/strcat.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5517#include "base/strings/utf_string_conversions.h"
Lei Zhang52637ed2019-02-20 01:38:3718#include "base/threading/thread_task_runner_handle.h"
Zhongyi Shic16b4102019-02-12 00:37:4019#include "base/time/default_tick_clock.h"
Eric Orth5ccc3f02021-09-23 00:01:5720#include "net/base/proxy_server.h"
21#include "net/base/proxy_string_util.h"
Robert Ogden78d4f9eb2020-03-17 20:56:3822#include "net/base/test_proxy_delegate.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5523#include "net/dns/mock_host_resolver.h"
Ben Schwartz3ff4dc1e62021-04-27 21:15:2324#include "net/dns/public/secure_dns_policy.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5525#include "net/http/http_auth_cache.h"
26#include "net/http/http_auth_handler_factory.h"
27#include "net/http/http_response_headers.h"
28#include "net/http/transport_security_state.h"
Matt Reichhoff0049a0b72021-10-20 20:44:2629#include "net/log/net_log.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5530#include "net/log/test_net_log.h"
31#include "net/log/test_net_log_util.h"
Victor Vasiliev4f6fb892019-05-31 16:58:3132#include "net/quic/address_utils.h"
Ryan Hamiltona3ee93a72018-08-01 22:03:0833#include "net/quic/crypto/proof_verifier_chromium.h"
34#include "net/quic/mock_crypto_client_stream_factory.h"
35#include "net/quic/mock_quic_data.h"
36#include "net/quic/quic_chromium_alarm_factory.h"
37#include "net/quic/quic_chromium_client_session.h"
38#include "net/quic/quic_chromium_connection_helper.h"
39#include "net/quic/quic_chromium_packet_writer.h"
Matt Menkefca05b62019-09-20 23:15:5640#include "net/quic/quic_crypto_client_config_handle.h"
Ryan Hamiltona3ee93a72018-08-01 22:03:0841#include "net/quic/quic_http_utils.h"
42#include "net/quic/quic_server_info.h"
43#include "net/quic/quic_stream_factory.h"
44#include "net/quic/quic_test_packet_maker.h"
Matt Menkefca05b62019-09-20 23:15:5645#include "net/quic/test_quic_crypto_client_config_handle.h"
Ryan Hamiltona3ee93a72018-08-01 22:03:0846#include "net/quic/test_task_runner.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5547#include "net/socket/socket_test_util.h"
48#include "net/test/cert_test_util.h"
Victor Vasilievdcdb6192019-02-25 19:49:5649#include "net/test/gtest_util.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5550#include "net/test/test_data_directory.h"
Gabriel Charettec7108742019-08-23 03:31:4051#include "net/test/test_with_task_environment.h"
Victor Vasiliev6bb59d22019-03-08 21:34:5152#include "net/third_party/quiche/src/quic/core/crypto/null_encrypter.h"
53#include "net/third_party/quiche/src/quic/core/quic_utils.h"
Victor Vasiliev6bb59d22019-03-08 21:34:5154#include "net/third_party/quiche/src/quic/test_tools/crypto_test_utils.h"
55#include "net/third_party/quiche/src/quic/test_tools/mock_clock.h"
56#include "net/third_party/quiche/src/quic/test_tools/mock_random.h"
Bence Béky6e243aa2019-12-13 19:01:0757#include "net/third_party/quiche/src/quic/test_tools/qpack/qpack_test_utils.h"
Victor Vasiliev6bb59d22019-03-08 21:34:5158#include "net/third_party/quiche/src/quic/test_tools/quic_connection_peer.h"
59#include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h"
[email protected]578968d42017-12-13 15:39:3260#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5561#include "testing/gmock/include/gmock/gmock.h"
62#include "testing/gtest/include/gtest/gtest.h"
Matt Menke8db2ff52021-11-23 20:17:4663#include "url/gurl.h"
Dan McArdle68a5f622021-07-09 20:56:5364#include "url/scheme_host_port.h"
65#include "url/url_constants.h"
Yixin Wang0d2c6b7e12017-08-16 21:12:5566
67using testing::_;
68using testing::AnyNumber;
69using testing::Return;
70
David Schinazi09e9a6012019-10-03 17:37:5771namespace net {
72namespace test {
73
Yixin Wang0d2c6b7e12017-08-16 21:12:5574namespace {
75
76static const char kOriginHost[] = "www.google.com";
77static const int kOriginPort = 443;
78static const char kProxyUrl[] = "https://myproxy:6121/";
79static const char kProxyHost[] = "myproxy";
80static const int kProxyPort = 6121;
81static const char kUserAgent[] = "Mozilla/1.0";
82static const char kRedirectUrl[] = "https://example.com/";
83
84static const char kMsg1[] = "\0hello!\xff";
85static const int kLen1 = 8;
86static const char kMsg2[] = "\0a2345678\0";
87static const int kLen2 = 10;
88static const char kMsg3[] = "bye!";
89static const int kLen3 = 4;
90static const char kMsg33[] = "bye!bye!";
91static const int kLen33 = kLen3 + kLen3;
92static const char kMsg333[] = "bye!bye!bye!";
93static const int kLen333 = kLen3 + kLen3 + kLen3;
94
David Schinazi09e9a6012019-10-03 17:37:5795struct TestParams {
96 quic::ParsedQuicVersion version;
97 bool client_headers_include_h2_stream_dependency;
98};
99
100// Used by ::testing::PrintToStringParamName().
101std::string PrintToString(const TestParams& p) {
Victor Vasiliev62c09dc2020-11-06 18:18:29102 return base::StrCat(
103 {ParsedQuicVersionToString(p.version), "_",
104 (p.client_headers_include_h2_stream_dependency ? "" : "No"),
105 "Dependency"});
David Schinazi09e9a6012019-10-03 17:37:57106}
107
108std::vector<TestParams> GetTestParams() {
109 std::vector<TestParams> params;
110 quic::ParsedQuicVersionVector all_supported_versions =
111 quic::AllSupportedVersions();
112 for (const auto& version : all_supported_versions) {
Renjie Tang1085bf52019-11-20 00:10:53113 params.push_back(TestParams{version, false});
114 params.push_back(TestParams{version, true});
David Schinazi09e9a6012019-10-03 17:37:57115 }
116 return params;
117}
118
Yixin Wang0d2c6b7e12017-08-16 21:12:55119} // anonymous namespace
120
David Schinazi09e9a6012019-10-03 17:37:57121class QuicProxyClientSocketTest : public ::testing::TestWithParam<TestParams>,
122 public WithTaskEnvironment {
Yixin Wang0d2c6b7e12017-08-16 21:12:55123 protected:
124 static const bool kFin = true;
125 static const bool kIncludeVersion = true;
126 static const bool kIncludeDiversificationNonce = true;
Yixin Wang0d2c6b7e12017-08-16 21:12:55127
128 static size_t GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:52129 quic::QuicByteCount packet_length,
Nick Harper23290b82019-05-02 00:02:56130 quic::ParsedQuicVersion version,
Yixin Wang0d2c6b7e12017-08-16 21:12:55131 bool include_version,
132 bool include_diversification_nonce,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52133 quic::QuicConnectionIdLength connection_id_length,
134 quic::QuicPacketNumberLength packet_number_length,
135 quic::QuicStreamOffset offset) {
Michael Warres167db3e2019-03-01 21:38:03136 quic::QuicVariableLengthIntegerLength retry_token_length_length =
137 quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
138 quic::QuicVariableLengthIntegerLength length_length =
Nick Harper23290b82019-05-02 00:02:56139 quic::QuicVersionHasLongHeaderLengths(version.transport_version) &&
140 include_version
Michael Warres167db3e2019-03-01 21:38:03141 ? quic::VARIABLE_LENGTH_INTEGER_LENGTH_2
142 : quic::VARIABLE_LENGTH_INTEGER_LENGTH_0;
Yixin Wang0d2c6b7e12017-08-16 21:12:55143 size_t min_data_length = 1;
144 size_t min_packet_length =
Ryan Hamilton8d9ee76e2018-05-29 23:52:52145 quic::NullEncrypter(quic::Perspective::IS_CLIENT)
Yixin Wang0d2c6b7e12017-08-16 21:12:55146 .GetCiphertextSize(min_data_length) +
Ryan Hamilton8d9ee76e2018-05-29 23:52:52147 quic::QuicPacketCreator::StreamFramePacketOverhead(
Nick Harper23290b82019-05-02 00:02:56148 version.transport_version, quic::PACKET_8BYTE_CONNECTION_ID,
Michael Warres60637ae2018-06-05 20:03:11149 quic::PACKET_0BYTE_CONNECTION_ID, include_version,
Michael Warres167db3e2019-03-01 21:38:03150 include_diversification_nonce, packet_number_length,
151 retry_token_length_length, length_length, offset);
Yixin Wang0d2c6b7e12017-08-16 21:12:55152
153 DCHECK(packet_length >= min_packet_length);
154 return min_data_length + packet_length - min_packet_length;
155 }
156
157 QuicProxyClientSocketTest()
David Schinazi09e9a6012019-10-03 17:37:57158 : version_(GetParam().version),
Nick Harper23290b82019-05-02 00:02:56159 client_data_stream_id1_(
Renjie Tang1085bf52019-11-20 00:10:53160 quic::VersionUsesHttp3(version_.transport_version)
161 ? quic::QuicUtils::GetFirstBidirectionalStreamId(
162 version_.transport_version,
163 quic::Perspective::IS_CLIENT)
164 : quic::QuicUtils::GetFirstBidirectionalStreamId(
165 version_.transport_version,
166 quic::Perspective::IS_CLIENT) +
167 quic::QuicUtils::StreamIdDelta(
168 version_.transport_version)),
David Schinazi09e9a6012019-10-03 17:37:57169 client_headers_include_h2_stream_dependency_(
170 GetParam().client_headers_include_h2_stream_dependency),
Ryan Hamiltonabad59e2019-06-06 04:02:59171 mock_quic_data_(version_),
Nick Harpera598fc5f2019-06-21 08:46:50172 crypto_config_(
173 quic::test::crypto_test_utils::ProofVerifierForTesting()),
David Schinazic8281052019-01-24 06:14:17174 connection_id_(quic::test::TestConnectionId(2)),
Yixin Wang0d2c6b7e12017-08-16 21:12:55175 client_maker_(version_,
176 connection_id_,
177 &clock_,
178 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52179 quic::Perspective::IS_CLIENT,
Yixin Wang079ad542018-01-11 04:06:05180 client_headers_include_h2_stream_dependency_),
Yixin Wang0d2c6b7e12017-08-16 21:12:55181 server_maker_(version_,
182 connection_id_,
183 &clock_,
184 kProxyHost,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52185 quic::Perspective::IS_SERVER,
Yixin Wang079ad542018-01-11 04:06:05186 false),
Yixin Wang0d2c6b7e12017-08-16 21:12:55187 random_generator_(0),
Yixin Wang0d2c6b7e12017-08-16 21:12:55188 user_agent_(kUserAgent),
Dan McArdle68a5f622021-07-09 20:56:53189 proxy_endpoint_(url::kHttpsScheme, kProxyHost, kProxyPort),
190 destination_endpoint_(url::kHttpsScheme, kOriginHost, kOriginPort),
Matt Menkebe090422019-10-18 20:25:26191 http_auth_cache_(
192 false /* key_server_entries_by_network_isolation_key */),
Yixin Wang0d2c6b7e12017-08-16 21:12:55193 host_resolver_(new MockCachingHostResolver()),
Eric Orthbe2efac2019-03-06 01:11:11194 http_auth_handler_factory_(HttpAuthHandlerFactory::CreateDefault()) {
Renjie Tang98b4d512020-02-08 01:24:19195 FLAGS_quic_enable_http3_grease_randomness = false;
Yixin Wang0d2c6b7e12017-08-16 21:12:55196 IPAddress ip(192, 0, 2, 33);
197 peer_addr_ = IPEndPoint(ip, 443);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52198 clock_.AdvanceTime(quic::QuicTime::Delta::FromMilliseconds(20));
David Schinazi84c58bb2020-06-04 20:14:33199 quic::QuicEnableVersion(version_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55200 }
201
Peter Boström407869b2021-10-07 04:42:48202 QuicProxyClientSocketTest(const QuicProxyClientSocketTest&) = delete;
203 QuicProxyClientSocketTest& operator=(const QuicProxyClientSocketTest&) =
204 delete;
205
Yixin Wang0d2c6b7e12017-08-16 21:12:55206 void SetUp() override {}
207
Yixin Wang2bea3cf2017-11-09 18:11:03208 void TearDown() override {
209 sock_.reset();
210 EXPECT_TRUE(mock_quic_data_.AllReadDataConsumed());
211 EXPECT_TRUE(mock_quic_data_.AllWriteDataConsumed());
212 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55213
214 void Initialize() {
215 std::unique_ptr<MockUDPClientSocket> socket(new MockUDPClientSocket(
Matt Reichhoff0049a0b72021-10-20 20:44:26216 mock_quic_data_.InitializeAndGetSequencedSocketData(), NetLog::Get()));
Yixin Wang0d2c6b7e12017-08-16 21:12:55217 socket->Connect(peer_addr_);
218 runner_ = new TestTaskRunner(&clock_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52219 send_algorithm_ = new quic::test::MockSendAlgorithm();
Yixin Wang0d2c6b7e12017-08-16 21:12:55220 EXPECT_CALL(*send_algorithm_, InRecovery()).WillRepeatedly(Return(false));
221 EXPECT_CALL(*send_algorithm_, InSlowStart()).WillRepeatedly(Return(false));
222 EXPECT_CALL(*send_algorithm_, OnPacketSent(_, _, _, _, _))
Dan Zhangf11470172017-09-18 22:02:09223 .Times(testing::AtLeast(1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55224 EXPECT_CALL(*send_algorithm_, GetCongestionWindow())
Zhongyi Shica576df2019-04-12 17:43:40225 .WillRepeatedly(Return(quic::kMaxOutgoingPacketSize));
Yixin Wang0d2c6b7e12017-08-16 21:12:55226 EXPECT_CALL(*send_algorithm_, PacingRate(_))
Ryan Hamilton8d9ee76e2018-05-29 23:52:52227 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Michael Warres74ee3ce2017-10-09 15:26:37228 EXPECT_CALL(*send_algorithm_, CanSend(_)).WillRepeatedly(Return(true));
Yixin Wang0d2c6b7e12017-08-16 21:12:55229 EXPECT_CALL(*send_algorithm_, BandwidthEstimate())
Ryan Hamilton8d9ee76e2018-05-29 23:52:52230 .WillRepeatedly(Return(quic::QuicBandwidth::Zero()));
Yixin Wang0d2c6b7e12017-08-16 21:12:55231 EXPECT_CALL(*send_algorithm_, SetFromConfig(_, _)).Times(AnyNumber());
232 EXPECT_CALL(*send_algorithm_, OnApplicationLimited(_)).Times(AnyNumber());
233 EXPECT_CALL(*send_algorithm_, GetCongestionControlType())
234 .Times(AnyNumber());
Peter Boström8a7540692021-04-05 20:48:20235 helper_ = std::make_unique<QuicChromiumConnectionHelper>(
236 &clock_, &random_generator_);
Renjie Tang6ff9a9b2021-02-03 22:11:09237 alarm_factory_ =
238 std::make_unique<QuicChromiumAlarmFactory>(runner_.get(), &clock_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55239
Ryan Hamilton9edcf1a2017-11-22 05:55:17240 QuicChromiumPacketWriter* writer = new QuicChromiumPacketWriter(
241 socket.get(), base::ThreadTaskRunnerHandle::Get().get());
Ryan Hamilton8d9ee76e2018-05-29 23:52:52242 quic::QuicConnection* connection = new quic::QuicConnection(
Nick Harperd049f192020-10-02 02:56:10243 connection_id_, quic::QuicSocketAddress(),
244 net::ToQuicSocketAddress(peer_addr_), helper_.get(),
Victor Vasiliev4f6fb892019-05-31 16:58:31245 alarm_factory_.get(), writer, true /* owns_writer */,
Nick Harper23290b82019-05-02 00:02:56246 quic::Perspective::IS_CLIENT, quic::test::SupportedVersions(version_));
Yixin Wang0d2c6b7e12017-08-16 21:12:55247 connection->set_visitor(&visitor_);
Ryan Hamilton8d9ee76e2018-05-29 23:52:52248 quic::test::QuicConnectionPeer::SetSendAlgorithm(connection,
249 send_algorithm_);
Yixin Wang0d2c6b7e12017-08-16 21:12:55250
251 // Load a certificate that is valid for *.example.org
252 scoped_refptr<X509Certificate> test_cert(
253 ImportCertFromFile(GetTestCertsDirectory(), "wildcard.pem"));
254 EXPECT_TRUE(test_cert.get());
255
256 verify_details_.cert_verify_result.verified_cert = test_cert;
257 verify_details_.cert_verify_result.is_issued_by_known_root = true;
258 crypto_client_stream_factory_.AddProofVerifyDetails(&verify_details_);
259
260 base::TimeTicks dns_end = base::TimeTicks::Now();
Peter Kastinge5a38ed2021-10-02 03:06:35261 base::TimeTicks dns_start = dns_end - base::Milliseconds(1);
Yixin Wang0d2c6b7e12017-08-16 21:12:55262
Renjie Tang6ff9a9b2021-02-03 22:11:09263 session_ = std::make_unique<QuicChromiumClientSession>(
Yixin Wang0d2c6b7e12017-08-16 21:12:55264 connection, std::move(socket),
265 /*stream_factory=*/nullptr, &crypto_client_stream_factory_, &clock_,
Nick Harper89bc7212018-07-31 19:07:57266 &transport_security_state_, /*ssl_config_service=*/nullptr,
Yixin Wang0d2c6b7e12017-08-16 21:12:55267 base::WrapUnique(static_cast<QuicServerInfo*>(nullptr)),
Paul Jensen8e3c5d32018-02-19 17:06:33268 QuicSessionKey("mail.example.org", 80, PRIVACY_MODE_DISABLED,
dalyk51ab46b2019-10-15 15:14:34269 SocketTag(), NetworkIsolationKey(),
Ben Schwartz3ff4dc1e62021-04-27 21:15:23270 SecureDnsPolicy::kAllow),
Bence Béky1ceba552019-07-19 17:11:05271 /*require_confirmation=*/false,
Bence Béky1ceba552019-07-19 17:11:05272 /*migrate_session_early_v2=*/false,
Zhongyi Shi757fcce2018-06-27 05:41:27273 /*migrate_session_on_network_change_v2=*/false,
274 /*default_network=*/NetworkChangeNotifier::kInvalidNetworkHandle,
Zhongyi Shie01f2db2019-02-22 19:53:23275 quic::QuicTime::Delta::FromMilliseconds(
Ryan Sleevi2e8255b2019-07-17 21:02:21276 kDefaultRetransmittableOnWireTimeout.InMilliseconds()),
Zhongyi Shiaf38c4e42019-08-29 22:49:05277 /*migrate_idle_session=*/true, /*allow_port_migration=*/false,
278 kDefaultIdleSessionMigrationPeriod, kMaxTimeOnNonDefaultNetwork,
Zhongyi Shiee760762018-08-01 00:54:29279 kMaxMigrationsToNonDefaultNetworkOnWriteError,
Zhongyi Shi8b1e43f2017-12-13 20:46:30280 kMaxMigrationsToNonDefaultNetworkOnPathDegrading,
Zhongyi Shi5f587cc2017-11-21 23:24:17281 kQuicYieldAfterPacketsRead,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52282 quic::QuicTime::Delta::FromMilliseconds(
283 kQuicYieldAfterDurationMilliseconds),
Zhongyi Shidbce7f412019-02-01 23:16:29284 /*go_away_on_path_degrading*/ false,
Yixin Wang079ad542018-01-11 04:06:05285 client_headers_include_h2_stream_dependency_, /*cert_verify_flags=*/0,
Matt Menkefca05b62019-09-20 23:15:56286 quic::test::DefaultQuicConfig(),
287 std::make_unique<TestQuicCryptoClientConfigHandle>(&crypto_config_),
Matt Menked804aaf2020-07-21 21:25:48288 "CONNECTION_UNKNOWN", dns_start, dns_end,
289 std::make_unique<quic::QuicClientPushPromiseIndex>(), nullptr,
Zhongyi Shic16b4102019-02-12 00:37:40290 base::DefaultTickClock::GetInstance(),
Yixin Wang0d2c6b7e12017-08-16 21:12:55291 base::ThreadTaskRunnerHandle::Get().get(),
Matt Reichhoff0049a0b72021-10-20 20:44:26292 /*socket_performance_watcher=*/nullptr, NetLog::Get());
Yixin Wang0d2c6b7e12017-08-16 21:12:55293
294 writer->set_delegate(session_.get());
295
Yixin Wang0d2c6b7e12017-08-16 21:12:55296 session_->Initialize();
Bence Béky6e243aa2019-12-13 19:01:07297
298 // Blackhole QPACK decoder stream instead of constructing mock writes.
299 if (VersionUsesHttp3(version_.transport_version)) {
300 session_->qpack_decoder()->set_qpack_stream_sender_delegate(
301 &noop_qpack_stream_sender_delegate_);
302 }
303
Yixin Wang0d2c6b7e12017-08-16 21:12:55304 TestCompletionCallback callback;
305 EXPECT_THAT(session_->CryptoConnect(callback.callback()), IsOk());
Victor Vasiliev29197e22020-01-16 20:48:32306 EXPECT_TRUE(session_->OneRttKeysAvailable());
Yixin Wang0d2c6b7e12017-08-16 21:12:55307
David Benjamin854a9922022-01-22 01:25:10308 session_handle_ = session_->CreateHandle(
309 url::SchemeHostPort(url::kHttpsScheme, "mail.example.org", 80));
Ramin Halavati683bcaa92018-02-14 08:42:39310 EXPECT_THAT(session_handle_->RequestStream(true, callback.callback(),
311 TRAFFIC_ANNOTATION_FOR_TESTS),
Yixin Wang0d2c6b7e12017-08-16 21:12:55312 IsOk());
313 std::unique_ptr<QuicChromiumClientStream::Handle> stream_handle =
314 session_handle_->ReleaseStream();
315 EXPECT_TRUE(stream_handle->IsOpen());
316
Renjie Tang6ff9a9b2021-02-03 22:11:09317 sock_ = std::make_unique<QuicProxyClientSocket>(
Robert Ogden78d4f9eb2020-03-17 20:56:38318 std::move(stream_handle), std::move(session_handle_),
Dan McArdle68a5f622021-07-09 20:56:53319 // TODO(crbug.com/1206799) Construct `ProxyServer` with plain
320 // `proxy_endpoint_` once it supports `url::SchemeHostPort`.
321 ProxyServer(ProxyServer::SCHEME_HTTPS,
322 HostPortPair::FromSchemeHostPort(proxy_endpoint_)),
323 user_agent_,
324 // TODO(crbug.com/1206799) Construct `QuicProxyClientSocket` with plain
325 // `proxy_endpoint_` once it supports `url::SchemeHostPort`.
326 HostPortPair::FromSchemeHostPort(destination_endpoint_),
Matt Reichhoff0049a0b72021-10-20 20:44:26327 NetLogWithSource::Make(NetLogSourceType::NONE),
Dan McArdle68a5f622021-07-09 20:56:53328 new HttpAuthController(HttpAuth::AUTH_PROXY, proxy_endpoint_.GetURL(),
Rohit Agarwal2653f472019-11-12 19:39:27329 NetworkIsolationKey(), &http_auth_cache_,
330 http_auth_handler_factory_.get(),
Robert Ogden78d4f9eb2020-03-17 20:56:38331 host_resolver_.get()),
Renjie Tang6ff9a9b2021-02-03 22:11:09332 proxy_delegate_.get());
Yixin Wang0d2c6b7e12017-08-16 21:12:55333
334 session_->StartReading();
335 }
336
Bence Béky4c325e52020-10-22 20:48:01337 void PopulateConnectRequestIR(spdy::Http2HeaderBlock* block) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55338 (*block)[":method"] = "CONNECT";
Dan McArdle68a5f622021-07-09 20:56:53339 (*block)[":authority"] =
340 HostPortPair::FromSchemeHostPort(destination_endpoint_).ToString();
Yixin Wang0d2c6b7e12017-08-16 21:12:55341 (*block)["user-agent"] = kUserAgent;
342 }
343
344 // Helper functions for constructing packets sent by the client
345
Ryan Hamilton8d9ee76e2018-05-29 23:52:52346 std::unique_ptr<quic::QuicReceivedPacket> ConstructSettingsPacket(
Fan Yangac867502019-01-28 21:10:23347 uint64_t packet_number) {
Ryan Hamilton0d65a8c2019-06-07 00:46:02348 return client_maker_.MakeInitialSettingsPacket(packet_number);
Yixin Wang0d2c6b7e12017-08-16 21:12:55349 }
350
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41351 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstOnlyPacket(
Fan Yangac867502019-01-28 21:10:23352 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52353 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23354 uint64_t largest_received,
Renjie Tangcd594f32020-07-11 20:18:34355 uint64_t smallest_received) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55356 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09357 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Renjie Tangcd594f32020-07-11 20:18:34358 largest_received, smallest_received,
Frank Kastenholz684ea412019-02-13 18:48:18359 /*include_stop_sending=*/false);
360 }
361
Ryan Hamilton8d9ee76e2018-05-29 23:52:52362 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndRstPacket(
Fan Yangac867502019-01-28 21:10:23363 uint64_t packet_number,
Ryan Hamilton8d9ee76e2018-05-29 23:52:52364 quic::QuicRstStreamErrorCode error_code,
Fan Yangac867502019-01-28 21:10:23365 uint64_t largest_received,
Renjie Tangcd594f32020-07-11 20:18:34366 uint64_t smallest_received) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55367 return client_maker_.MakeAckAndRstPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09368 packet_number, !kIncludeVersion, client_data_stream_id1_, error_code,
Renjie Tangcd594f32020-07-11 20:18:34369 largest_received, smallest_received,
Frank Kastenholz684ea412019-02-13 18:48:18370 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55371 }
372
Ryan Hamilton8d9ee76e2018-05-29 23:52:52373 std::unique_ptr<quic::QuicReceivedPacket> ConstructRstPacket(
Fan Yangac867502019-01-28 21:10:23374 uint64_t packet_number,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41375 quic::QuicRstStreamErrorCode error_code) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55376 return client_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09377 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18378 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55379 }
380
Ryan Hamilton8d9ee76e2018-05-29 23:52:52381 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectRequestPacket(
Fan Yangac867502019-01-28 21:10:23382 uint64_t packet_number,
Lily Chenf11e1292018-11-29 16:42:09383 RequestPriority request_priority = LOWEST) {
Bence Béky4c325e52020-10-22 20:48:01384 spdy::Http2HeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55385 PopulateConnectRequestIR(&block);
386 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09387 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Lily Chenf11e1292018-11-29 16:42:09388 ConvertRequestPriorityToQuicPriority(request_priority),
Ryan Hamilton0d65a8c2019-06-07 00:46:02389 std::move(block), 0, nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55390 }
391
Robert Ogden78d4f9eb2020-03-17 20:56:38392 std::unique_ptr<quic::QuicReceivedPacket>
393 ConstructConnectRequestPacketWithExtraHeaders(
394 uint64_t packet_number,
395 std::vector<std::pair<std::string, std::string>> extra_headers,
396 RequestPriority request_priority = LOWEST) {
Bence Béky4c325e52020-10-22 20:48:01397 spdy::Http2HeaderBlock block;
Robert Ogden78d4f9eb2020-03-17 20:56:38398 block[":method"] = "CONNECT";
Dan McArdle68a5f622021-07-09 20:56:53399 block[":authority"] =
400 HostPortPair::FromSchemeHostPort(destination_endpoint_).ToString();
Robert Ogden78d4f9eb2020-03-17 20:56:38401 for (const auto& header : extra_headers) {
402 block[header.first] = header.second;
403 }
Robert Ogden78d4f9eb2020-03-17 20:56:38404 return client_maker_.MakeRequestHeadersPacket(
405 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
406 ConvertRequestPriorityToQuicPriority(request_priority),
407 std::move(block), 0, nullptr);
408 }
409
Ryan Hamilton8d9ee76e2018-05-29 23:52:52410 std::unique_ptr<quic::QuicReceivedPacket> ConstructConnectAuthRequestPacket(
Fan Yangac867502019-01-28 21:10:23411 uint64_t packet_number) {
Renjie Tangee921d12020-02-06 00:41:49412 RequestPriority request_priority = LOWEST;
Bence Béky4c325e52020-10-22 20:48:01413 spdy::Http2HeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55414 PopulateConnectRequestIR(&block);
415 block["proxy-authorization"] = "Basic Zm9vOmJhcg==";
416 return client_maker_.MakeRequestHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09417 packet_number, client_data_stream_id1_, kIncludeVersion, !kFin,
Renjie Tangee921d12020-02-06 00:41:49418 ConvertRequestPriorityToQuicPriority(request_priority),
419 std::move(block), 0, nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55420 }
421
Ryan Hamilton8d9ee76e2018-05-29 23:52:52422 std::unique_ptr<quic::QuicReceivedPacket> ConstructDataPacket(
Fan Yangac867502019-01-28 21:10:23423 uint64_t packet_number,
Victor Vasiliev47ecb6a2020-10-14 17:22:10424 absl::string_view data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09425 return client_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17426 !kIncludeVersion, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55427 }
428
Ryan Hamilton8d9ee76e2018-05-29 23:52:52429 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckAndDataPacket(
Fan Yangac867502019-01-28 21:10:23430 uint64_t packet_number,
431 uint64_t largest_received,
432 uint64_t smallest_received,
Victor Vasiliev47ecb6a2020-10-14 17:22:10433 absl::string_view data) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55434 return client_maker_.MakeAckAndDataPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09435 packet_number, !kIncludeVersion, client_data_stream_id1_,
Renjie Tangcd594f32020-07-11 20:18:34436 largest_received, smallest_received, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55437 }
438
Ryan Hamilton8d9ee76e2018-05-29 23:52:52439 std::unique_ptr<quic::QuicReceivedPacket> ConstructAckPacket(
Fan Yangac867502019-01-28 21:10:23440 uint64_t packet_number,
441 uint64_t largest_received,
Renjie Tangcd594f32020-07-11 20:18:34442 uint64_t smallest_received) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55443 return client_maker_.MakeAckPacket(packet_number, largest_received,
Renjie Tangcd594f32020-07-11 20:18:34444 smallest_received);
Yixin Wang0d2c6b7e12017-08-16 21:12:55445 }
446
447 // Helper functions for constructing packets sent by the server
448
Ryan Hamilton8d9ee76e2018-05-29 23:52:52449 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerRstPacket(
Fan Yangac867502019-01-28 21:10:23450 uint64_t packet_number,
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41451 quic::QuicRstStreamErrorCode error_code) {
Yixin Wang0d2c6b7e12017-08-16 21:12:55452 return server_maker_.MakeRstPacket(packet_number, !kIncludeVersion,
Ryan Hamilton47cf9d12018-10-17 04:33:09453 client_data_stream_id1_, error_code,
Frank Kastenholz684ea412019-02-13 18:48:18454 /*include_stop_sending_if_v99=*/true);
Yixin Wang0d2c6b7e12017-08-16 21:12:55455 }
456
Ryan Hamilton8d9ee76e2018-05-29 23:52:52457 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataPacket(
Fan Yangac867502019-01-28 21:10:23458 uint64_t packet_number,
Victor Vasiliev47ecb6a2020-10-14 17:22:10459 absl::string_view data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09460 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17461 !kIncludeVersion, !kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55462 }
463
Ryan Hamilton8d9ee76e2018-05-29 23:52:52464 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerDataFinPacket(
Fan Yangac867502019-01-28 21:10:23465 uint64_t packet_number,
Victor Vasiliev47ecb6a2020-10-14 17:22:10466 absl::string_view data) {
Ryan Hamilton47cf9d12018-10-17 04:33:09467 return server_maker_.MakeDataPacket(packet_number, client_data_stream_id1_,
Ryan Hamilton7505eb92019-06-08 00:22:17468 !kIncludeVersion, kFin, data);
Yixin Wang0d2c6b7e12017-08-16 21:12:55469 }
470
Ryan Hamilton8d9ee76e2018-05-29 23:52:52471 std::unique_ptr<quic::QuicReceivedPacket> ConstructServerConnectReplyPacket(
Fan Yangac867502019-01-28 21:10:23472 uint64_t packet_number,
Renjie Tang1085bf52019-11-20 00:10:53473 bool fin,
474 size_t* header_length = nullptr) {
Bence Béky4c325e52020-10-22 20:48:01475 spdy::Http2HeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55476 block[":status"] = "200";
477
478 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09479 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Renjie Tang1085bf52019-11-20 00:10:53480 std::move(block), header_length);
Yixin Wang0d2c6b7e12017-08-16 21:12:55481 }
482
Ryan Hamilton8d9ee76e2018-05-29 23:52:52483 std::unique_ptr<quic::QuicReceivedPacket>
Robert Ogden78d4f9eb2020-03-17 20:56:38484 ConstructServerConnectReplyPacketWithExtraHeaders(
485 uint64_t packet_number,
486 bool fin,
487 std::vector<std::pair<std::string, std::string>> extra_headers) {
Bence Béky4c325e52020-10-22 20:48:01488 spdy::Http2HeaderBlock block;
Robert Ogden78d4f9eb2020-03-17 20:56:38489 block[":status"] = "200";
490 for (const auto& header : extra_headers) {
491 block[header.first] = header.second;
492 }
493
494 return server_maker_.MakeResponseHeadersPacket(
495 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
496 std::move(block), nullptr);
497 }
498
499 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23500 ConstructServerConnectAuthReplyPacket(uint64_t packet_number, bool fin) {
Bence Béky4c325e52020-10-22 20:48:01501 spdy::Http2HeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55502 block[":status"] = "407";
503 block["proxy-authenticate"] = "Basic realm=\"MyRealm1\"";
504 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09505 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02506 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55507 }
508
Ryan Hamilton8d9ee76e2018-05-29 23:52:52509 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23510 ConstructServerConnectRedirectReplyPacket(uint64_t packet_number, bool fin) {
Bence Béky4c325e52020-10-22 20:48:01511 spdy::Http2HeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55512 block[":status"] = "302";
513 block["location"] = kRedirectUrl;
514 block["set-cookie"] = "foo=bar";
515 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09516 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02517 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55518 }
519
Ryan Hamilton8d9ee76e2018-05-29 23:52:52520 std::unique_ptr<quic::QuicReceivedPacket>
Fan Yangac867502019-01-28 21:10:23521 ConstructServerConnectErrorReplyPacket(uint64_t packet_number, bool fin) {
Bence Béky4c325e52020-10-22 20:48:01522 spdy::Http2HeaderBlock block;
Yixin Wang0d2c6b7e12017-08-16 21:12:55523 block[":status"] = "500";
524
525 return server_maker_.MakeResponseHeadersPacket(
Ryan Hamilton47cf9d12018-10-17 04:33:09526 packet_number, client_data_stream_id1_, !kIncludeVersion, fin,
Ryan Hamilton0d65a8c2019-06-07 00:46:02527 std::move(block), nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:55528 }
529
530 void AssertConnectSucceeds() {
531 TestCompletionCallback callback;
532 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
533 ASSERT_THAT(callback.WaitForResult(), IsOk());
534 }
535
536 void AssertConnectFails(int result) {
537 TestCompletionCallback callback;
538 ASSERT_THAT(sock_->Connect(callback.callback()), IsError(ERR_IO_PENDING));
539 ASSERT_EQ(result, callback.WaitForResult());
540 }
541
542 void ResumeAndRun() {
543 // Run until the pause, if the provider isn't paused yet.
544 SequencedSocketData* data = mock_quic_data_.GetSequencedSocketData();
545 data->RunUntilPaused();
546 data->Resume();
547 base::RunLoop().RunUntilIdle();
548 }
549
550 void AssertWriteReturns(const char* data, int len, int rv) {
Victor Costan9c7302b2018-08-27 16:39:44551 scoped_refptr<IOBufferWithSize> buf =
552 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55553 memcpy(buf->data(), data, len);
554 EXPECT_EQ(rv,
[email protected]578968d42017-12-13 15:39:32555 sock_->Write(buf.get(), buf->size(), write_callback_.callback(),
556 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55557 }
558
559 void AssertSyncWriteSucceeds(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44560 scoped_refptr<IOBufferWithSize> buf =
561 base::MakeRefCounted<IOBufferWithSize>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55562 memcpy(buf->data(), data, len);
Bence Békyd8a21fc32018-06-27 18:29:58563 EXPECT_EQ(len,
564 sock_->Write(buf.get(), buf->size(), CompletionOnceCallback(),
565 TRAFFIC_ANNOTATION_FOR_TESTS));
Yixin Wang0d2c6b7e12017-08-16 21:12:55566 }
567
568 void AssertSyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44569 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Bence Békyd8a21fc32018-06-27 18:29:58570 ASSERT_EQ(len, sock_->Read(buf.get(), len, CompletionOnceCallback()));
Ian Swett59044a642019-08-19 16:51:15571 ASSERT_EQ(std::string(data, len), std::string(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55572 ASSERT_TRUE(sock_->IsConnected());
573 }
574
575 void AssertAsyncReadEquals(const char* data, int len) {
Victor Costan9c7302b2018-08-27 16:39:44576 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55577 ASSERT_EQ(ERR_IO_PENDING,
578 sock_->Read(buf.get(), len, read_callback_.callback()));
579 EXPECT_TRUE(sock_->IsConnected());
580
581 ResumeAndRun();
582
583 EXPECT_EQ(len, read_callback_.WaitForResult());
584 EXPECT_TRUE(sock_->IsConnected());
Ian Swett59044a642019-08-19 16:51:15585 ASSERT_EQ(std::string(data, len), std::string(buf->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55586 }
587
588 void AssertReadStarts(const char* data, int len) {
589 // Issue the read, which will be completed asynchronously.
Victor Costan9c7302b2018-08-27 16:39:44590 read_buf_ = base::MakeRefCounted<IOBuffer>(len);
Yixin Wang0d2c6b7e12017-08-16 21:12:55591 ASSERT_EQ(ERR_IO_PENDING,
592 sock_->Read(read_buf_.get(), len, read_callback_.callback()));
593 EXPECT_TRUE(sock_->IsConnected());
594 }
595
596 void AssertReadReturns(const char* data, int len) {
597 EXPECT_TRUE(sock_->IsConnected());
598
599 // Now the read will return.
600 EXPECT_EQ(len, read_callback_.WaitForResult());
Ian Swett59044a642019-08-19 16:51:15601 ASSERT_EQ(std::string(data, len), std::string(read_buf_->data(), len));
Yixin Wang0d2c6b7e12017-08-16 21:12:55602 }
603
Victor Vasiliev076657c2019-03-12 02:46:43604 std::string ConstructDataHeader(size_t body_len) {
Nick Harperc6cb7a612020-02-24 20:03:32605 if (!version_.HasIetfQuicFrames()) {
Renjief49758b2019-01-11 23:32:41606 return "";
607 }
Ian Swett17d4d1c02021-06-08 19:52:41608 quic::QuicBuffer buffer = quic::HttpEncoder::SerializeDataFrameHeader(
609 body_len, quic::SimpleBufferAllocator::Get());
610 return std::string(buffer.data(), buffer.size());
Renjief49758b2019-01-11 23:32:41611 }
612
Matt Reichhoff0049a0b72021-10-20 20:44:26613 RecordingNetLogObserver net_log_observer_;
David Schinazi3f7465c2019-07-12 01:57:05614 QuicFlagSaver saver_;
Nick Harper23290b82019-05-02 00:02:56615 const quic::ParsedQuicVersion version_;
Ryan Hamilton47cf9d12018-10-17 04:33:09616 const quic::QuicStreamId client_data_stream_id1_;
Yixin Wang079ad542018-01-11 04:06:05617 const bool client_headers_include_h2_stream_dependency_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55618
619 // order of destruction of these members matter
Ryan Hamilton8d9ee76e2018-05-29 23:52:52620 quic::MockClock clock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55621 MockQuicData mock_quic_data_;
622 std::unique_ptr<QuicChromiumConnectionHelper> helper_;
623 std::unique_ptr<QuicChromiumClientSession> session_;
624 std::unique_ptr<QuicChromiumClientSession::Handle> session_handle_;
625 std::unique_ptr<QuicProxyClientSocket> sock_;
Robert Ogden78d4f9eb2020-03-17 20:56:38626 std::unique_ptr<TestProxyDelegate> proxy_delegate_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55627
Keishi Hattori0e45c022021-11-27 09:25:52628 raw_ptr<quic::test::MockSendAlgorithm> send_algorithm_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55629 scoped_refptr<TestTaskRunner> runner_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55630
631 std::unique_ptr<QuicChromiumAlarmFactory> alarm_factory_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52632 testing::StrictMock<quic::test::MockQuicConnectionVisitor> visitor_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55633 TransportSecurityState transport_security_state_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52634 quic::QuicCryptoClientConfig crypto_config_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55635
Ryan Hamilton8d9ee76e2018-05-29 23:52:52636 const quic::QuicConnectionId connection_id_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55637 QuicTestPacketMaker client_maker_;
638 QuicTestPacketMaker server_maker_;
639 IPEndPoint peer_addr_;
Ryan Hamilton8d9ee76e2018-05-29 23:52:52640 quic::test::MockRandom random_generator_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55641 ProofVerifyDetailsChromium verify_details_;
642 MockCryptoClientStreamFactory crypto_client_stream_factory_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55643
644 std::string user_agent_;
Dan McArdle68a5f622021-07-09 20:56:53645 url::SchemeHostPort proxy_endpoint_;
646 url::SchemeHostPort destination_endpoint_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55647 HttpAuthCache http_auth_cache_;
648 std::unique_ptr<MockHostResolverBase> host_resolver_;
649 std::unique_ptr<HttpAuthHandlerRegistryFactory> http_auth_handler_factory_;
650
651 TestCompletionCallback read_callback_;
652 scoped_refptr<IOBuffer> read_buf_;
653
654 TestCompletionCallback write_callback_;
655
Bence Béky6e243aa2019-12-13 19:01:07656 quic::test::NoopQpackStreamSenderDelegate noop_qpack_stream_sender_delegate_;
Yixin Wang0d2c6b7e12017-08-16 21:12:55657};
658
659TEST_P(QuicProxyClientSocketTest, ConnectSendsCorrectRequest) {
Renjie Tangaadb84b2019-08-31 01:00:23660 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25661 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23662 mock_quic_data_.AddWrite(SYNCHRONOUS,
663 ConstructSettingsPacket(packet_number++));
664 }
665 mock_quic_data_.AddWrite(SYNCHRONOUS,
666 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:43667 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55668 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
669 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:34670 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
671 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55672
673 Initialize();
674
675 ASSERT_FALSE(sock_->IsConnected());
676
677 AssertConnectSucceeds();
678
679 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
680 ASSERT_TRUE(response != nullptr);
681 ASSERT_EQ(200, response->headers->response_code());
David Benjamin854a9922022-01-22 01:25:10682
683 // Although the underlying HTTP/3 connection uses TLS and negotiates ALPN, the
684 // tunnel itself is a TCP connection to the origin and should not report these
685 // values.
686 net::SSLInfo ssl_info;
687 EXPECT_FALSE(sock_->GetSSLInfo(&ssl_info));
688 EXPECT_FALSE(sock_->WasAlpnNegotiated());
689 EXPECT_EQ(sock_->GetNegotiatedProtocol(), NextProto::kProtoUnknown);
Yixin Wang0d2c6b7e12017-08-16 21:12:55690}
691
Robert Ogden78d4f9eb2020-03-17 20:56:38692TEST_P(QuicProxyClientSocketTest, ProxyDelegateExtraHeaders) {
693 proxy_delegate_ = std::make_unique<TestProxyDelegate>();
Dan McArdle68a5f622021-07-09 20:56:53694 // TODO(crbug.com/1206799) Construct `ProxyServer` with plain
695 // `proxy_endpoint_` once it supports `url::SchemeHostPort`.
696 ProxyServer proxy_server(ProxyServer::SCHEME_HTTPS,
697 HostPortPair::FromSchemeHostPort(proxy_endpoint_));
Robert Ogden78d4f9eb2020-03-17 20:56:38698
699 const char kResponseHeaderName[] = "foo";
700 const char kResponseHeaderValue[] = "testing";
701
702 int packet_number = 1;
703 if (VersionUsesHttp3(version_.transport_version)) {
704 mock_quic_data_.AddWrite(SYNCHRONOUS,
705 ConstructSettingsPacket(packet_number++));
706 }
Eric Orth5ccc3f02021-09-23 00:01:57707 mock_quic_data_.AddWrite(SYNCHRONOUS,
708 ConstructConnectRequestPacketWithExtraHeaders(
709 packet_number++,
710 // Order matters! Keep these alphabetical.
711 {{"foo", ProxyServerToProxyUri(proxy_server)},
712 {"user-agent", kUserAgent}}));
Robert Ogden78d4f9eb2020-03-17 20:56:38713 mock_quic_data_.AddRead(
714 ASYNC, ConstructServerConnectReplyPacketWithExtraHeaders(
715 1, !kFin, {{kResponseHeaderName, kResponseHeaderValue}}));
716 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
717 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:34718 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
719 quic::QUIC_STREAM_CANCELLED, 1, 1));
Robert Ogden78d4f9eb2020-03-17 20:56:38720
721 Initialize();
722
723 ASSERT_FALSE(sock_->IsConnected());
724
725 AssertConnectSucceeds();
726
727 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
728 ASSERT_TRUE(response != nullptr);
729 ASSERT_EQ(200, response->headers->response_code());
730 proxy_delegate_->VerifyOnTunnelHeadersReceived(
731 proxy_server, kResponseHeaderName, kResponseHeaderValue);
732}
733
Yixin Wang0d2c6b7e12017-08-16 21:12:55734TEST_P(QuicProxyClientSocketTest, ConnectWithAuthRequested) {
Renjie Tangaadb84b2019-08-31 01:00:23735 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25736 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23737 mock_quic_data_.AddWrite(SYNCHRONOUS,
738 ConstructSettingsPacket(packet_number++));
739 }
740 mock_quic_data_.AddWrite(SYNCHRONOUS,
741 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:43742 mock_quic_data_.AddRead(ASYNC,
743 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55744 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
745 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:34746 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
747 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55748
749 Initialize();
750
751 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
752
753 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
754 ASSERT_TRUE(response != nullptr);
755 ASSERT_EQ(407, response->headers->response_code());
756}
757
758TEST_P(QuicProxyClientSocketTest, ConnectWithAuthCredentials) {
Renjie Tangaadb84b2019-08-31 01:00:23759 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25760 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23761 mock_quic_data_.AddWrite(SYNCHRONOUS,
762 ConstructSettingsPacket(packet_number++));
763 }
764 mock_quic_data_.AddWrite(SYNCHRONOUS,
765 ConstructConnectAuthRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:43766 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55767 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
768 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:34769 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
770 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55771
772 Initialize();
773
774 // Add auth to cache
Jan Wilken Dörriec92a6d7242021-03-23 17:43:48775 const std::u16string kFoo(u"foo");
776 const std::u16string kBar(u"bar");
Matt Menke8db2ff52021-11-23 20:17:46777 http_auth_cache_.Add(
778 url::SchemeHostPort(GURL(kProxyUrl)), HttpAuth::AUTH_PROXY, "MyRealm1",
779 HttpAuth::AUTH_SCHEME_BASIC, NetworkIsolationKey(),
780 "Basic realm=MyRealm1", AuthCredentials(kFoo, kBar), "/");
Yixin Wang0d2c6b7e12017-08-16 21:12:55781
782 AssertConnectSucceeds();
783
784 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
785 ASSERT_TRUE(response != nullptr);
786 ASSERT_EQ(200, response->headers->response_code());
787}
788
Eric Roman96c5b292019-04-23 18:04:59789// Tests that a redirect response from a CONNECT fails.
Yixin Wang0d2c6b7e12017-08-16 21:12:55790TEST_P(QuicProxyClientSocketTest, ConnectRedirects) {
Renjie Tangaadb84b2019-08-31 01:00:23791 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25792 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23793 mock_quic_data_.AddWrite(SYNCHRONOUS,
794 ConstructSettingsPacket(packet_number++));
795 }
796 mock_quic_data_.AddWrite(SYNCHRONOUS,
797 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:43798 mock_quic_data_.AddRead(ASYNC,
799 ConstructServerConnectRedirectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55800 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
801 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:34802 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
803 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55804
805 Initialize();
806
Eric Roman96c5b292019-04-23 18:04:59807 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
Yixin Wang0d2c6b7e12017-08-16 21:12:55808
809 const HttpResponseInfo* response = sock_->GetConnectResponseInfo();
810 ASSERT_TRUE(response != nullptr);
811
812 const HttpResponseHeaders* headers = response->headers.get();
813 ASSERT_EQ(302, headers->response_code());
Eric Roman96c5b292019-04-23 18:04:59814 ASSERT_TRUE(headers->HasHeader("set-cookie"));
Yixin Wang0d2c6b7e12017-08-16 21:12:55815
816 std::string location;
817 ASSERT_TRUE(headers->IsRedirect(&location));
818 ASSERT_EQ(location, kRedirectUrl);
819}
820
821TEST_P(QuicProxyClientSocketTest, ConnectFails) {
Renjie Tangaadb84b2019-08-31 01:00:23822 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25823 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23824 mock_quic_data_.AddWrite(SYNCHRONOUS,
825 ConstructSettingsPacket(packet_number++));
826 }
827 mock_quic_data_.AddWrite(SYNCHRONOUS,
828 ConstructConnectRequestPacket(packet_number++));
David Schinazi395918c2021-02-05 18:56:21829 mock_quic_data_.AddRead(ASYNC, ERR_CONNECTION_CLOSED);
Yixin Wang0d2c6b7e12017-08-16 21:12:55830
831 Initialize();
832
833 ASSERT_FALSE(sock_->IsConnected());
834
835 AssertConnectFails(ERR_QUIC_PROTOCOL_ERROR);
836
837 ASSERT_FALSE(sock_->IsConnected());
838}
839
840TEST_P(QuicProxyClientSocketTest, WasEverUsedReturnsCorrectValue) {
Renjie Tangaadb84b2019-08-31 01:00:23841 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25842 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23843 mock_quic_data_.AddWrite(SYNCHRONOUS,
844 ConstructSettingsPacket(packet_number++));
845 }
846 mock_quic_data_.AddWrite(SYNCHRONOUS,
847 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:43848 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55849 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
850 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:34851 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
852 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:55853
854 Initialize();
855
Victor Vasiliev7da08172019-10-14 06:04:25856 if (VersionUsesHttp3(version_.transport_version))
Renjie Tangaadb84b2019-08-31 01:00:23857 EXPECT_TRUE(sock_->WasEverUsed()); // Used due to crypto handshake
Yixin Wang0d2c6b7e12017-08-16 21:12:55858 AssertConnectSucceeds();
859 EXPECT_TRUE(sock_->WasEverUsed());
860 sock_->Disconnect();
861 EXPECT_TRUE(sock_->WasEverUsed());
862}
863
864TEST_P(QuicProxyClientSocketTest, GetPeerAddressReturnsCorrectValues) {
Renjie Tangaadb84b2019-08-31 01:00:23865 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25866 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23867 mock_quic_data_.AddWrite(SYNCHRONOUS,
868 ConstructSettingsPacket(packet_number++));
869 }
870 mock_quic_data_.AddWrite(SYNCHRONOUS,
871 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:43872 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55873 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
David Schinazi395918c2021-02-05 18:56:21874 mock_quic_data_.AddRead(ASYNC, ERR_CONNECTION_CLOSED);
Yixin Wang0d2c6b7e12017-08-16 21:12:55875
876 Initialize();
877
878 IPEndPoint addr;
879 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
880
881 AssertConnectSucceeds();
882 EXPECT_TRUE(sock_->IsConnected());
883 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsOk());
884
885 ResumeAndRun();
886
887 EXPECT_FALSE(sock_->IsConnected());
888 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
889
890 sock_->Disconnect();
891
892 EXPECT_THAT(sock_->GetPeerAddress(&addr), IsError(ERR_SOCKET_NOT_CONNECTED));
893}
894
895TEST_P(QuicProxyClientSocketTest, IsConnectedAndIdle) {
Renjie Tangaadb84b2019-08-31 01:00:23896 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25897 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23898 mock_quic_data_.AddWrite(SYNCHRONOUS,
899 ConstructSettingsPacket(packet_number++));
900 }
901 mock_quic_data_.AddWrite(SYNCHRONOUS,
902 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:43903 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:55904 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
905
Victor Vasiliev076657c2019-03-12 02:46:43906 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:17907 mock_quic_data_.AddRead(
908 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41909 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:34910 ConstructAckPacket(packet_number++, 2, 1));
Renjie Tangaadb84b2019-08-31 01:00:23911 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
912 mock_quic_data_.AddWrite(
913 SYNCHRONOUS,
914 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55915
916 Initialize();
917
918 EXPECT_FALSE(sock_->IsConnectedAndIdle());
919
920 AssertConnectSucceeds();
921
922 EXPECT_TRUE(sock_->IsConnectedAndIdle());
923
924 // The next read is consumed and buffered.
925 ResumeAndRun();
926
927 EXPECT_FALSE(sock_->IsConnectedAndIdle());
928
929 AssertSyncReadEquals(kMsg1, kLen1);
930
931 EXPECT_TRUE(sock_->IsConnectedAndIdle());
932}
933
934TEST_P(QuicProxyClientSocketTest, GetTotalReceivedBytes) {
Renjie Tangaadb84b2019-08-31 01:00:23935 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25936 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:23937 mock_quic_data_.AddWrite(SYNCHRONOUS,
938 ConstructSettingsPacket(packet_number++));
939 }
Renjie Tang1085bf52019-11-20 00:10:53940 size_t header_length;
Renjie Tangaadb84b2019-08-31 01:00:23941 mock_quic_data_.AddWrite(SYNCHRONOUS,
942 ConstructConnectRequestPacket(packet_number++));
Renjie Tang1085bf52019-11-20 00:10:53943 mock_quic_data_.AddRead(
944 ASYNC, ConstructServerConnectReplyPacket(1, !kFin, &header_length));
Yixin Wang0d2c6b7e12017-08-16 21:12:55945 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
946
Renjie Tang1085bf52019-11-20 00:10:53947 std::string data_header = ConstructDataHeader(kLen333);
948 mock_quic_data_.AddRead(ASYNC,
949 ConstructServerDataPacket(
950 2, data_header + std::string(kMsg333, kLen333)));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:41951 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:34952 ConstructAckPacket(packet_number++, 2, 1));
Renjie Tangaadb84b2019-08-31 01:00:23953 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
954 mock_quic_data_.AddWrite(
955 SYNCHRONOUS,
956 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:55957
958 Initialize();
959
960 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
961
962 AssertConnectSucceeds();
963
Renjie Tang1085bf52019-11-20 00:10:53964 if (!VersionUsesHttp3(version_.transport_version)) {
965 header_length = 0;
966 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
967 } else {
968 // HTTP/3 sends and receives HTTP headers on the request stream.
969 EXPECT_EQ((int64_t)(header_length), sock_->GetTotalReceivedBytes());
970 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55971
972 // The next read is consumed and buffered.
973 ResumeAndRun();
974
Renjie Tang1085bf52019-11-20 00:10:53975 if (!VersionUsesHttp3(version_.transport_version)) {
976 EXPECT_EQ(0, sock_->GetTotalReceivedBytes());
977 } else {
978 // HTTP/3 encodes data with DATA frame. The header is consumed.
979 EXPECT_EQ((int64_t)(header_length + data_header.length()),
980 sock_->GetTotalReceivedBytes());
981 }
Yixin Wang0d2c6b7e12017-08-16 21:12:55982
983 // The payload from the single large data frame will be read across
984 // two different reads.
985 AssertSyncReadEquals(kMsg33, kLen33);
986
Renjie Tang1085bf52019-11-20 00:10:53987 EXPECT_EQ((int64_t)(header_length + data_header.length() + kLen33),
Renjief49758b2019-01-11 23:32:41988 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55989
990 AssertSyncReadEquals(kMsg3, kLen3);
991
Renjie Tang1085bf52019-11-20 00:10:53992 EXPECT_EQ((int64_t)(header_length + kLen333 + data_header.length()),
Renjief49758b2019-01-11 23:32:41993 sock_->GetTotalReceivedBytes());
Yixin Wang0d2c6b7e12017-08-16 21:12:55994}
995
Lily Chenf11e1292018-11-29 16:42:09996TEST_P(QuicProxyClientSocketTest, SetStreamPriority) {
Renjie Tangaadb84b2019-08-31 01:00:23997 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:25998 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tang1085bf52019-11-20 00:10:53999 mock_quic_data_.AddWrite(SYNCHRONOUS,
1000 ConstructSettingsPacket(packet_number++));
Renjie Tangaadb84b2019-08-31 01:00:231001 }
Matt Menkeedaf3b82019-03-14 21:39:441002 // Despite setting the priority to HIGHEST, the requets initial priority of
1003 // LOWEST is used.
Renjie Tangaadb84b2019-08-31 01:00:231004 mock_quic_data_.AddWrite(
1005 SYNCHRONOUS, ConstructConnectRequestPacket(packet_number++, LOWEST));
Lily Chenf11e1292018-11-29 16:42:091006 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
1007 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1008 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341009 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1010 quic::QUIC_STREAM_CANCELLED, 1, 1));
Lily Chenf11e1292018-11-29 16:42:091011
1012 Initialize();
1013
1014 sock_->SetStreamPriority(HIGHEST);
1015 AssertConnectSucceeds();
1016}
Yixin Wang0d2c6b7e12017-08-16 21:12:551017
1018TEST_P(QuicProxyClientSocketTest, WriteSendsDataInDataFrame) {
Renjie Tangaadb84b2019-08-31 01:00:231019 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251020 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231021 mock_quic_data_.AddWrite(SYNCHRONOUS,
1022 ConstructSettingsPacket(packet_number++));
1023 }
1024 mock_quic_data_.AddWrite(SYNCHRONOUS,
1025 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431026 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551027 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harperc6cb7a612020-02-24 20:03:321028 if (version_.HasIetfQuicFrames()) {
Victor Vasiliev076657c2019-03-12 02:46:431029 std::string header = ConstructDataHeader(kLen1);
1030 mock_quic_data_.AddWrite(
Renjie Tangaadb84b2019-08-31 01:00:231031 SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341032 ConstructAckAndDataPacket(packet_number++, 1, 1,
Renjie Tangd5133972019-12-06 00:20:281033 {header + std::string(kMsg1, kLen1)}));
Victor Vasiliev076657c2019-03-12 02:46:431034 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411035 mock_quic_data_.AddWrite(
Renjief49758b2019-01-11 23:32:411036 SYNCHRONOUS,
Renjie Tangd5133972019-12-06 00:20:281037 ConstructDataPacket(packet_number++,
1038 {header2 + std::string(kMsg2, kLen2)}));
Renjief49758b2019-01-11 23:32:411039 mock_quic_data_.AddWrite(
Renjie Tangaadb84b2019-08-31 01:00:231040 SYNCHRONOUS,
1041 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
1042 } else {
1043 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341044 SYNCHRONOUS, ConstructAckAndDataPacket(packet_number++, 1, 1,
Renjie Tangaadb84b2019-08-31 01:00:231045 std::string(kMsg1, kLen1)));
1046 mock_quic_data_.AddWrite(
1047 SYNCHRONOUS,
1048 ConstructDataPacket(packet_number++, std::string(kMsg2, kLen2)));
1049 mock_quic_data_.AddWrite(
1050 SYNCHRONOUS,
1051 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Renjief49758b2019-01-11 23:32:411052 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551053
1054 Initialize();
1055
1056 AssertConnectSucceeds();
1057
1058 AssertSyncWriteSucceeds(kMsg1, kLen1);
1059 AssertSyncWriteSucceeds(kMsg2, kLen2);
1060}
1061
1062TEST_P(QuicProxyClientSocketTest, WriteSplitsLargeDataIntoMultiplePackets) {
Renjief49758b2019-01-11 23:32:411063 int write_packet_index = 1;
Victor Vasiliev7da08172019-10-14 06:04:251064 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231065 mock_quic_data_.AddWrite(SYNCHRONOUS,
1066 ConstructSettingsPacket(write_packet_index++));
1067 }
Renjief49758b2019-01-11 23:32:411068 mock_quic_data_.AddWrite(SYNCHRONOUS,
1069 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431070 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551071 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:431072 std::string header = ConstructDataHeader(kLen1);
Nick Harperc6cb7a612020-02-24 20:03:321073 if (!version_.HasIetfQuicFrames()) {
Renjief49758b2019-01-11 23:32:411074 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341075 SYNCHRONOUS, ConstructAckAndDataPacket(write_packet_index++, 1, 1,
Victor Vasiliev076657c2019-03-12 02:46:431076 std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:411077 } else {
Renjie Tangd5133972019-12-06 00:20:281078 mock_quic_data_.AddWrite(
1079 SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341080 ConstructAckAndDataPacket(write_packet_index++, 1, 1,
Renjie Tangd5133972019-12-06 00:20:281081 {header + std::string(kMsg1, kLen1)}));
Renjief49758b2019-01-11 23:32:411082 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551083
1084 // Expect |kNumDataPackets| data packets, each containing the max possible
1085 // amount of data.
Renjied172e812019-01-16 05:12:351086 int numDataPackets = 3;
1087 std::string data(numDataPackets * quic::kDefaultMaxPacketSize, 'x');
Renjief49758b2019-01-11 23:32:411088 quic::QuicStreamOffset offset = kLen1 + header.length();
Renjied172e812019-01-16 05:12:351089
Nick Harperc6cb7a612020-02-24 20:03:321090 if (version_.HasIetfQuicFrames()) {
Renjied172e812019-01-16 05:12:351091 numDataPackets++;
Renjief49758b2019-01-11 23:32:411092 }
Renjied172e812019-01-16 05:12:351093 size_t total_data_length = 0;
1094 for (int i = 0; i < numDataPackets; ++i) {
Yixin Wang0d2c6b7e12017-08-16 21:12:551095 size_t max_packet_data_length = GetStreamFrameDataLengthFromPacketLength(
Ryan Hamilton8d9ee76e2018-05-29 23:52:521096 quic::kDefaultMaxPacketSize, version_, !kIncludeVersion,
1097 !kIncludeDiversificationNonce, quic::PACKET_8BYTE_CONNECTION_ID,
1098 quic::PACKET_1BYTE_PACKET_NUMBER, offset);
Nick Harperc6cb7a612020-02-24 20:03:321099 if (version_.HasIetfQuicFrames() && i == 0) {
Victor Vasiliev39caecf2021-09-28 03:06:031100 // 3673 is the data frame length from packet length.
1101 std::string header2 = ConstructDataHeader(3673);
Renjied172e812019-01-16 05:12:351102 mock_quic_data_.AddWrite(
Renjie Tangd5133972019-12-06 00:20:281103 SYNCHRONOUS,
1104 ConstructDataPacket(
1105 write_packet_index++,
1106 {header2 +
1107 std::string(data.c_str(), max_packet_data_length - 7)}));
Renjied172e812019-01-16 05:12:351108 offset += max_packet_data_length - header2.length() - 1;
Nick Harperc6cb7a612020-02-24 20:03:321109 } else if (version_.HasIetfQuicFrames() && i == numDataPackets - 1) {
Renjied172e812019-01-16 05:12:351110 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171111 SYNCHRONOUS, ConstructDataPacket(write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:431112 std::string(data.c_str(), 7)));
Renjied172e812019-01-16 05:12:351113 offset += 7;
1114 } else {
1115 mock_quic_data_.AddWrite(
Victor Vasiliev076657c2019-03-12 02:46:431116 SYNCHRONOUS, ConstructDataPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171117 write_packet_index++,
Victor Vasiliev076657c2019-03-12 02:46:431118 std::string(data.c_str(), max_packet_data_length)));
Renjied172e812019-01-16 05:12:351119 offset += max_packet_data_length;
1120 }
1121 if (i != 3) {
1122 total_data_length += max_packet_data_length;
1123 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551124 }
Renjied172e812019-01-16 05:12:351125
Yixin Wang0d2c6b7e12017-08-16 21:12:551126 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411127 SYNCHRONOUS,
1128 ConstructRstPacket(write_packet_index++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551129
1130 Initialize();
1131
1132 AssertConnectSucceeds();
1133
1134 // Make a small write. An ACK and STOP_WAITING will be bundled. This prevents
1135 // ACK and STOP_WAITING from being bundled with the subsequent large write.
1136 // This allows the test code for computing the size of data sent in each
1137 // packet to not become too complicated.
1138 AssertSyncWriteSucceeds(kMsg1, kLen1);
1139
1140 // Make large write that should be split up
1141 AssertSyncWriteSucceeds(data.c_str(), total_data_length);
1142}
1143
1144// ----------- Read
1145
1146TEST_P(QuicProxyClientSocketTest, ReadReadsDataInDataFrame) {
Renjie Tangaadb84b2019-08-31 01:00:231147 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251148 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231149 mock_quic_data_.AddWrite(SYNCHRONOUS,
1150 ConstructSettingsPacket(packet_number++));
1151 }
1152 mock_quic_data_.AddWrite(SYNCHRONOUS,
1153 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431154 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551155 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1156
Victor Vasiliev076657c2019-03-12 02:46:431157 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171158 mock_quic_data_.AddRead(
1159 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411160 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341161 ConstructAckPacket(packet_number++, 2, 1));
Renjie Tangaadb84b2019-08-31 01:00:231162 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1163 mock_quic_data_.AddWrite(
1164 SYNCHRONOUS,
1165 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551166
1167 Initialize();
1168
1169 AssertConnectSucceeds();
1170
1171 ResumeAndRun();
1172 AssertSyncReadEquals(kMsg1, kLen1);
1173}
1174
1175TEST_P(QuicProxyClientSocketTest, ReadDataFromBufferedFrames) {
Renjie Tangaadb84b2019-08-31 01:00:231176 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251177 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231178 mock_quic_data_.AddWrite(SYNCHRONOUS,
1179 ConstructSettingsPacket(packet_number++));
1180 }
1181 mock_quic_data_.AddWrite(SYNCHRONOUS,
1182 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431183 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551184 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1185
Victor Vasiliev076657c2019-03-12 02:46:431186 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171187 mock_quic_data_.AddRead(
1188 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjie Tangaadb84b2019-08-31 01:00:231189 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341190 ConstructAckPacket(packet_number++, 2, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551191 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1192
Victor Vasiliev076657c2019-03-12 02:46:431193 std::string header2 = ConstructDataHeader(kLen2);
1194 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171195 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551196 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1197
1198 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341199 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1200 quic::QUIC_STREAM_CANCELLED, 3, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551201
1202 Initialize();
1203
1204 AssertConnectSucceeds();
1205
1206 ResumeAndRun();
1207 AssertSyncReadEquals(kMsg1, kLen1);
1208
1209 ResumeAndRun();
1210 AssertSyncReadEquals(kMsg2, kLen2);
1211}
1212
1213TEST_P(QuicProxyClientSocketTest, ReadDataMultipleBufferedFrames) {
Renjie Tangaadb84b2019-08-31 01:00:231214 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251215 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231216 mock_quic_data_.AddWrite(SYNCHRONOUS,
1217 ConstructSettingsPacket(packet_number++));
1218 }
1219 mock_quic_data_.AddWrite(SYNCHRONOUS,
1220 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431221 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551222 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1223
Victor Vasiliev076657c2019-03-12 02:46:431224 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171225 mock_quic_data_.AddRead(
1226 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjie Tangaadb84b2019-08-31 01:00:231227 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341228 ConstructAckPacket(packet_number++, 2, 1));
Victor Vasiliev076657c2019-03-12 02:46:431229 std::string header2 = ConstructDataHeader(kLen2);
1230 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171231 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551232 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1233
1234 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341235 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1236 quic::QUIC_STREAM_CANCELLED, 3, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551237
1238 Initialize();
1239
1240 AssertConnectSucceeds();
1241
1242 // The next two reads are consumed and buffered.
1243 ResumeAndRun();
1244
1245 AssertSyncReadEquals(kMsg1, kLen1);
1246 AssertSyncReadEquals(kMsg2, kLen2);
1247}
1248
1249TEST_P(QuicProxyClientSocketTest, LargeReadWillMergeDataFromDifferentFrames) {
Renjie Tangaadb84b2019-08-31 01:00:231250 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251251 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231252 mock_quic_data_.AddWrite(SYNCHRONOUS,
1253 ConstructSettingsPacket(packet_number++));
1254 }
1255 mock_quic_data_.AddWrite(SYNCHRONOUS,
1256 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431257 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551258 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1259
Victor Vasiliev076657c2019-03-12 02:46:431260 std::string header = ConstructDataHeader(kLen3);
Ryan Hamilton7505eb92019-06-08 00:22:171261 mock_quic_data_.AddRead(
1262 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg3, kLen3)));
Renjie Tangaadb84b2019-08-31 01:00:231263 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341264 ConstructAckPacket(packet_number++, 2, 1));
Victor Vasiliev076657c2019-03-12 02:46:431265 std::string header2 = ConstructDataHeader(kLen3);
1266 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171267 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551268 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1269
1270 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341271 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1272 quic::QUIC_STREAM_CANCELLED, 3, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551273
1274 Initialize();
1275
1276 AssertConnectSucceeds();
1277
1278 // The next two reads are consumed and buffered.
1279 ResumeAndRun();
1280 // The payload from two data frames, each with kMsg3 will be combined
1281 // together into a single read().
1282 AssertSyncReadEquals(kMsg33, kLen33);
1283}
1284
1285TEST_P(QuicProxyClientSocketTest, MultipleShortReadsThenMoreRead) {
Renjie Tangaadb84b2019-08-31 01:00:231286 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251287 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231288 mock_quic_data_.AddWrite(SYNCHRONOUS,
1289 ConstructSettingsPacket(packet_number++));
1290 }
1291 mock_quic_data_.AddWrite(SYNCHRONOUS,
1292 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431293 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551294 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1295
Victor Vasiliev076657c2019-03-12 02:46:431296 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411297 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171298 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjie Tangaadb84b2019-08-31 01:00:231299 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341300 ConstructAckPacket(packet_number++, 2, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551301
Victor Vasiliev076657c2019-03-12 02:46:431302 std::string header2 = ConstructDataHeader(kLen3);
Renjief49758b2019-01-11 23:32:411303 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171304 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Renjief49758b2019-01-11 23:32:411305 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171306 ASYNC, ConstructServerDataPacket(4, header2 + std::string(kMsg3, kLen3)));
Renjie Tangaadb84b2019-08-31 01:00:231307 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341308 ConstructAckPacket(packet_number++, 4, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551309
Victor Vasiliev076657c2019-03-12 02:46:431310 std::string header3 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411311 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171312 ASYNC, ConstructServerDataPacket(5, header3 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551313 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1314
1315 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341316 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1317 quic::QUIC_STREAM_CANCELLED, 5, 5));
Yixin Wang0d2c6b7e12017-08-16 21:12:551318
1319 Initialize();
1320
1321 AssertConnectSucceeds();
1322
1323 // The next 4 reads are consumed and buffered.
1324 ResumeAndRun();
1325
1326 AssertSyncReadEquals(kMsg1, kLen1);
1327 // The payload from two data frames, each with kMsg3 will be combined
1328 // together into a single read().
1329 AssertSyncReadEquals(kMsg33, kLen33);
1330 AssertSyncReadEquals(kMsg2, kLen2);
1331}
1332
1333TEST_P(QuicProxyClientSocketTest, ReadWillSplitDataFromLargeFrame) {
Renjie Tangaadb84b2019-08-31 01:00:231334 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251335 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231336 mock_quic_data_.AddWrite(SYNCHRONOUS,
1337 ConstructSettingsPacket(packet_number++));
1338 }
1339 mock_quic_data_.AddWrite(SYNCHRONOUS,
1340 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431341 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551342 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1343
Victor Vasiliev076657c2019-03-12 02:46:431344 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171345 mock_quic_data_.AddRead(
1346 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjie Tangaadb84b2019-08-31 01:00:231347 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341348 ConstructAckPacket(packet_number++, 2, 1));
Victor Vasiliev076657c2019-03-12 02:46:431349 std::string header2 = ConstructDataHeader(kLen33);
Ryan Hamilton7505eb92019-06-08 00:22:171350 mock_quic_data_.AddRead(ASYNC, ConstructServerDataPacket(
1351 3, header2 + std::string(kMsg33, kLen33)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551352 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1353
1354 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341355 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1356 quic::QUIC_STREAM_CANCELLED, 3, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551357
1358 Initialize();
1359
1360 AssertConnectSucceeds();
1361
1362 // The next 2 reads are consumed and buffered.
1363 ResumeAndRun();
1364
1365 AssertSyncReadEquals(kMsg1, kLen1);
1366 // The payload from the single large data frame will be read across
1367 // two different reads.
1368 AssertSyncReadEquals(kMsg3, kLen3);
1369 AssertSyncReadEquals(kMsg3, kLen3);
1370}
1371
1372TEST_P(QuicProxyClientSocketTest, MultipleReadsFromSameLargeFrame) {
Renjie Tangaadb84b2019-08-31 01:00:231373 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251374 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231375 mock_quic_data_.AddWrite(SYNCHRONOUS,
1376 ConstructSettingsPacket(packet_number++));
1377 }
1378 mock_quic_data_.AddWrite(SYNCHRONOUS,
1379 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431380 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551381 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1382
Victor Vasiliev076657c2019-03-12 02:46:431383 std::string header = ConstructDataHeader(kLen333);
Renjief49758b2019-01-11 23:32:411384 mock_quic_data_.AddRead(
Victor Vasiliev076657c2019-03-12 02:46:431385 ASYNC,
Ryan Hamilton7505eb92019-06-08 00:22:171386 ConstructServerDataPacket(2, header + std::string(kMsg333, kLen333)));
Renjie Tangaadb84b2019-08-31 01:00:231387 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341388 ConstructAckPacket(packet_number++, 2, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551389 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1390
Renjie Tangaadb84b2019-08-31 01:00:231391 mock_quic_data_.AddWrite(
1392 SYNCHRONOUS,
1393 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551394
1395 Initialize();
1396
1397 AssertConnectSucceeds();
1398
1399 // The next read is consumed and buffered.
1400 ResumeAndRun();
1401
1402 // The payload from the single large data frame will be read across
1403 // two different reads.
1404 AssertSyncReadEquals(kMsg33, kLen33);
1405
1406 // Now attempt to do a read of more data than remains buffered
Victor Costan9c7302b2018-08-27 16:39:441407 scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kLen33);
Bence Békyd8a21fc32018-06-27 18:29:581408 ASSERT_EQ(kLen3, sock_->Read(buf.get(), kLen33, CompletionOnceCallback()));
Ian Swett59044a642019-08-19 16:51:151409 ASSERT_EQ(std::string(kMsg3, kLen3), std::string(buf->data(), kLen3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551410 ASSERT_TRUE(sock_->IsConnected());
1411}
1412
1413TEST_P(QuicProxyClientSocketTest, ReadAuthResponseBody) {
Renjie Tangaadb84b2019-08-31 01:00:231414 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251415 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231416 mock_quic_data_.AddWrite(SYNCHRONOUS,
1417 ConstructSettingsPacket(packet_number++));
1418 }
1419 mock_quic_data_.AddWrite(SYNCHRONOUS,
1420 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431421 mock_quic_data_.AddRead(ASYNC,
1422 ConstructServerConnectAuthReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551423 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1424
Victor Vasiliev076657c2019-03-12 02:46:431425 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171426 mock_quic_data_.AddRead(
1427 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjie Tangaadb84b2019-08-31 01:00:231428 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341429 ConstructAckPacket(packet_number++, 2, 1));
Victor Vasiliev076657c2019-03-12 02:46:431430 std::string header2 = ConstructDataHeader(kLen2);
1431 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171432 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551433 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1434
1435 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341436 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1437 quic::QUIC_STREAM_CANCELLED, 3, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551438
1439 Initialize();
1440
1441 AssertConnectFails(ERR_PROXY_AUTH_REQUESTED);
1442
1443 // The next two reads are consumed and buffered.
1444 ResumeAndRun();
1445
1446 AssertSyncReadEquals(kMsg1, kLen1);
1447 AssertSyncReadEquals(kMsg2, kLen2);
1448}
1449
1450TEST_P(QuicProxyClientSocketTest, ReadErrorResponseBody) {
Renjie Tangaadb84b2019-08-31 01:00:231451 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251452 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231453 mock_quic_data_.AddWrite(SYNCHRONOUS,
1454 ConstructSettingsPacket(packet_number++));
1455 }
1456 mock_quic_data_.AddWrite(SYNCHRONOUS,
1457 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431458 mock_quic_data_.AddRead(ASYNC,
1459 ConstructServerConnectErrorReplyPacket(1, !kFin));
Victor Vasiliev076657c2019-03-12 02:46:431460 std::string header = ConstructDataHeader(kLen1);
Renjief49758b2019-01-11 23:32:411461 mock_quic_data_.AddRead(
1462 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:171463 ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjie Tangaadb84b2019-08-31 01:00:231464 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341465 ConstructAckPacket(packet_number++, 2, 1));
Victor Vasiliev076657c2019-03-12 02:46:431466 std::string header2 = ConstructDataHeader(kLen2);
Renjief49758b2019-01-11 23:32:411467 mock_quic_data_.AddRead(
1468 SYNCHRONOUS,
Ryan Hamilton7505eb92019-06-08 00:22:171469 ConstructServerDataPacket(3, header2 + std::string(kMsg2, kLen2)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551470 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1471
1472 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341473 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1474 quic::QUIC_STREAM_CANCELLED, 3, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551475 Initialize();
1476
1477 AssertConnectFails(ERR_TUNNEL_CONNECTION_FAILED);
1478}
1479
1480// ----------- Reads and Writes
1481
1482TEST_P(QuicProxyClientSocketTest, AsyncReadAroundWrite) {
Renjief49758b2019-01-11 23:32:411483 int write_packet_index = 1;
Victor Vasiliev7da08172019-10-14 06:04:251484 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231485 mock_quic_data_.AddWrite(SYNCHRONOUS,
1486 ConstructSettingsPacket(write_packet_index++));
1487 }
Renjief49758b2019-01-11 23:32:411488 mock_quic_data_.AddWrite(SYNCHRONOUS,
1489 ConstructConnectRequestPacket(write_packet_index++));
Zhongyi Shi32f2fd02018-04-16 18:23:431490 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551491 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1492
Victor Vasiliev076657c2019-03-12 02:46:431493 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171494 mock_quic_data_.AddRead(
1495 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Zhongyi Shi32f2fd02018-04-16 18:23:431496 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341497 ConstructAckPacket(write_packet_index++, 2, 1));
Renjief49758b2019-01-11 23:32:411498
Victor Vasiliev076657c2019-03-12 02:46:431499 std::string header2 = ConstructDataHeader(kLen2);
Nick Harperc6cb7a612020-02-24 20:03:321500 if (version_.HasIetfQuicFrames()) {
Victor Vasiliev076657c2019-03-12 02:46:431501 mock_quic_data_.AddWrite(
1502 SYNCHRONOUS,
Renjie Tangd5133972019-12-06 00:20:281503 ConstructDataPacket(write_packet_index++,
1504 {header2 + std::string(kMsg2, kLen2)}));
Renjied172e812019-01-16 05:12:351505 } else {
Renjief49758b2019-01-11 23:32:411506 mock_quic_data_.AddWrite(
Ryan Hamilton7505eb92019-06-08 00:22:171507 SYNCHRONOUS,
1508 ConstructDataPacket(write_packet_index++, std::string(kMsg2, kLen2)));
Renjied172e812019-01-16 05:12:351509 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551510
1511 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1512
Victor Vasiliev076657c2019-03-12 02:46:431513 std::string header3 = ConstructDataHeader(kLen3);
1514 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171515 ASYNC, ConstructServerDataPacket(3, header3 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551516 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1517
1518 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341519 SYNCHRONOUS, ConstructAckAndRstPacket(write_packet_index++,
1520 quic::QUIC_STREAM_CANCELLED, 3, 3));
Yixin Wang0d2c6b7e12017-08-16 21:12:551521
1522 Initialize();
1523
1524 AssertConnectSucceeds();
1525
1526 ResumeAndRun();
1527
1528 AssertSyncReadEquals(kMsg1, kLen1);
1529
1530 AssertReadStarts(kMsg3, kLen3);
1531 // Read should block until after the write succeeds.
1532
1533 AssertSyncWriteSucceeds(kMsg2, kLen2);
1534
1535 ASSERT_FALSE(read_callback_.have_result());
1536 ResumeAndRun();
1537
1538 // Now the read will return.
1539 AssertReadReturns(kMsg3, kLen3);
1540}
1541
1542TEST_P(QuicProxyClientSocketTest, AsyncWriteAroundReads) {
Renjie Tangaadb84b2019-08-31 01:00:231543 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251544 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231545 mock_quic_data_.AddWrite(SYNCHRONOUS,
1546 ConstructSettingsPacket(packet_number++));
1547 }
1548 mock_quic_data_.AddWrite(SYNCHRONOUS,
1549 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431550 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551551 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1552
Victor Vasiliev076657c2019-03-12 02:46:431553 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171554 mock_quic_data_.AddRead(
1555 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Renjie Tangaadb84b2019-08-31 01:00:231556 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341557 ConstructAckPacket(packet_number++, 2, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551558 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1559
Victor Vasiliev076657c2019-03-12 02:46:431560 std::string header2 = ConstructDataHeader(kLen3);
1561 mock_quic_data_.AddRead(
Ryan Hamilton7505eb92019-06-08 00:22:171562 ASYNC, ConstructServerDataPacket(3, header2 + std::string(kMsg3, kLen3)));
Yixin Wang0d2c6b7e12017-08-16 21:12:551563 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1564
1565 mock_quic_data_.AddWrite(ASYNC, ERR_IO_PENDING); // Pause
1566
Victor Vasiliev076657c2019-03-12 02:46:431567 std::string header3 = ConstructDataHeader(kLen2);
Nick Harperc6cb7a612020-02-24 20:03:321568 if (!version_.HasIetfQuicFrames()) {
Renjief49758b2019-01-11 23:32:411569 mock_quic_data_.AddWrite(
Renjie Tangaadb84b2019-08-31 01:00:231570 ASYNC, ConstructDataPacket(packet_number++, std::string(kMsg2, kLen2)));
1571 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341572 SYNCHRONOUS, ConstructAckAndDataPacket(packet_number++, 3, 3,
Renjie Tangaadb84b2019-08-31 01:00:231573 std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411574 } else {
1575 mock_quic_data_.AddWrite(
Renjie Tangd5133972019-12-06 00:20:281576 ASYNC, ConstructDataPacket(packet_number++,
1577 {header3 + std::string(kMsg2, kLen2)}));
Renjie Tangaadb84b2019-08-31 01:00:231578 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341579 ASYNC, ConstructAckAndDataPacket(packet_number++, 3, 3,
Victor Vasiliev076657c2019-03-12 02:46:431580 header3 + std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411581 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551582
Renjie Tangaadb84b2019-08-31 01:00:231583 mock_quic_data_.AddWrite(
1584 SYNCHRONOUS,
1585 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551586
1587 Initialize();
1588
1589 AssertConnectSucceeds();
1590
1591 ResumeAndRun();
1592 AssertSyncReadEquals(kMsg1, kLen1);
1593
1594 // Write should block until the next read completes.
1595 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1596 // asynchronous starting with the second time it's called while the UDP socket
1597 // is write-blocked. Therefore, at least two writes need to be called on
1598 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351599 AssertWriteReturns(kMsg2, kLen2, kLen2);
Yixin Wang0d2c6b7e12017-08-16 21:12:551600 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1601
1602 AssertAsyncReadEquals(kMsg3, kLen3);
1603
1604 ASSERT_FALSE(write_callback_.have_result());
1605
1606 // Now the write will complete
1607 ResumeAndRun();
Yixin Wangdbbd8752018-01-17 21:50:021608 EXPECT_EQ(kLen2, write_callback_.WaitForResult());
Yixin Wang0d2c6b7e12017-08-16 21:12:551609}
1610
1611// ----------- Reading/Writing on Closed socket
1612
1613// Reading from an already closed socket should return 0
1614TEST_P(QuicProxyClientSocketTest, ReadOnClosedSocketReturnsZero) {
Renjie Tangaadb84b2019-08-31 01:00:231615 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251616 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231617 mock_quic_data_.AddWrite(SYNCHRONOUS,
1618 ConstructSettingsPacket(packet_number++));
1619 }
1620 mock_quic_data_.AddWrite(SYNCHRONOUS,
1621 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431622 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551623 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
David Schinazi395918c2021-02-05 18:56:211624 mock_quic_data_.AddRead(ASYNC, ERR_CONNECTION_CLOSED);
Yixin Wang0d2c6b7e12017-08-16 21:12:551625
1626 Initialize();
1627
1628 AssertConnectSucceeds();
1629
1630 ResumeAndRun();
1631
1632 ASSERT_FALSE(sock_->IsConnected());
Raul Tambre94493c652019-03-11 17:18:351633 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1634 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1635 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551636 ASSERT_FALSE(sock_->IsConnectedAndIdle());
1637}
1638
1639// Read pending when socket is closed should return 0
1640TEST_P(QuicProxyClientSocketTest, PendingReadOnCloseReturnsZero) {
Renjie Tangaadb84b2019-08-31 01:00:231641 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251642 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231643 mock_quic_data_.AddWrite(SYNCHRONOUS,
1644 ConstructSettingsPacket(packet_number++));
1645 }
1646 mock_quic_data_.AddWrite(SYNCHRONOUS,
1647 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431648 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551649 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
David Schinazi395918c2021-02-05 18:56:211650 mock_quic_data_.AddRead(ASYNC, ERR_CONNECTION_CLOSED);
Yixin Wang0d2c6b7e12017-08-16 21:12:551651
1652 Initialize();
1653
1654 AssertConnectSucceeds();
1655
1656 AssertReadStarts(kMsg1, kLen1);
1657
1658 ResumeAndRun();
1659
1660 ASSERT_EQ(0, read_callback_.WaitForResult());
1661}
1662
1663// Reading from a disconnected socket is an error
1664TEST_P(QuicProxyClientSocketTest, ReadOnDisconnectSocketReturnsNotConnected) {
Renjie Tangaadb84b2019-08-31 01:00:231665 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251666 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231667 mock_quic_data_.AddWrite(SYNCHRONOUS,
1668 ConstructSettingsPacket(packet_number++));
1669 }
1670 mock_quic_data_.AddWrite(SYNCHRONOUS,
1671 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431672 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551673 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1674 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341675 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1676 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551677
1678 Initialize();
1679
1680 AssertConnectSucceeds();
1681
1682 sock_->Disconnect();
1683
1684 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581685 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551686}
1687
1688// Reading data after receiving FIN should return buffered data received before
1689// FIN, then 0.
1690TEST_P(QuicProxyClientSocketTest, ReadAfterFinReceivedReturnsBufferedData) {
Renjie Tangaadb84b2019-08-31 01:00:231691 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251692 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231693 mock_quic_data_.AddWrite(SYNCHRONOUS,
1694 ConstructSettingsPacket(packet_number++));
1695 }
1696 mock_quic_data_.AddWrite(SYNCHRONOUS,
1697 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431698 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551699 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1700
Victor Vasiliev076657c2019-03-12 02:46:431701 std::string header = ConstructDataHeader(kLen1);
1702 mock_quic_data_.AddRead(ASYNC, ConstructServerDataFinPacket(
Ryan Hamilton7505eb92019-06-08 00:22:171703 2, header + std::string(kMsg1, kLen1)));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411704 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341705 ConstructAckPacket(packet_number++, 2, 1));
Renjie Tangaadb84b2019-08-31 01:00:231706 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1707 mock_quic_data_.AddWrite(
1708 SYNCHRONOUS,
1709 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551710
1711 Initialize();
1712
1713 AssertConnectSucceeds();
1714
1715 ResumeAndRun();
1716
1717 AssertSyncReadEquals(kMsg1, kLen1);
Raul Tambre94493c652019-03-11 17:18:351718 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
1719 ASSERT_EQ(0, sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551720
1721 sock_->Disconnect();
1722 ASSERT_EQ(ERR_SOCKET_NOT_CONNECTED,
Bence Békyd8a21fc32018-06-27 18:29:581723 sock_->Read(nullptr, 1, CompletionOnceCallback()));
Yixin Wang0d2c6b7e12017-08-16 21:12:551724}
1725
1726// Calling Write() on a closed socket is an error.
1727TEST_P(QuicProxyClientSocketTest, WriteOnClosedStream) {
Renjie Tangaadb84b2019-08-31 01:00:231728 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251729 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231730 mock_quic_data_.AddWrite(SYNCHRONOUS,
1731 ConstructSettingsPacket(packet_number++));
1732 }
1733 mock_quic_data_.AddWrite(SYNCHRONOUS,
1734 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431735 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551736 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
David Schinazi395918c2021-02-05 18:56:211737 mock_quic_data_.AddRead(ASYNC, ERR_CONNECTION_CLOSED);
Yixin Wang0d2c6b7e12017-08-16 21:12:551738
1739 Initialize();
1740
1741 AssertConnectSucceeds();
1742
1743 ResumeAndRun();
1744
1745 AssertWriteReturns(kMsg1, kLen1, ERR_QUIC_PROTOCOL_ERROR);
1746}
1747
1748// Calling Write() on a disconnected socket is an error.
1749TEST_P(QuicProxyClientSocketTest, WriteOnDisconnectedSocket) {
Renjie Tangaadb84b2019-08-31 01:00:231750 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251751 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231752 mock_quic_data_.AddWrite(SYNCHRONOUS,
1753 ConstructSettingsPacket(packet_number++));
1754 }
1755 mock_quic_data_.AddWrite(SYNCHRONOUS,
1756 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431757 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551758 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1759 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341760 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1761 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551762
1763 Initialize();
1764
1765 AssertConnectSucceeds();
1766
1767 sock_->Disconnect();
1768
1769 AssertWriteReturns(kMsg1, kLen1, ERR_SOCKET_NOT_CONNECTED);
1770}
1771
1772// If the socket is closed with a pending Write(), the callback should be called
1773// with the same error the session was closed with.
1774TEST_P(QuicProxyClientSocketTest, WritePendingOnClose) {
Renjie Tangaadb84b2019-08-31 01:00:231775 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251776 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231777 mock_quic_data_.AddWrite(SYNCHRONOUS,
1778 ConstructSettingsPacket(packet_number++));
1779 }
1780 mock_quic_data_.AddWrite(SYNCHRONOUS,
1781 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431782 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551783 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1784 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1785
1786 Initialize();
1787
1788 AssertConnectSucceeds();
1789
1790 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1791 // asynchronous starting with the second time it's called while the UDP socket
1792 // is write-blocked. Therefore, at least two writes need to be called on
1793 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351794 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551795
1796 // This second write will be async. This is the pending write that's being
1797 // tested.
1798 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1799
1800 // Make sure the write actually starts.
1801 base::RunLoop().RunUntilIdle();
1802
Ryan Hamilton8d9ee76e2018-05-29 23:52:521803 session_->CloseSessionOnError(ERR_CONNECTION_CLOSED,
Renjieba55fae2018-09-20 03:05:161804 quic::QUIC_INTERNAL_ERROR,
1805 quic::ConnectionCloseBehavior::SILENT_CLOSE);
Yixin Wang0d2c6b7e12017-08-16 21:12:551806
1807 EXPECT_THAT(write_callback_.WaitForResult(), IsError(ERR_CONNECTION_CLOSED));
1808}
1809
1810TEST_P(QuicProxyClientSocketTest, DisconnectWithWritePending) {
Renjie Tangaadb84b2019-08-31 01:00:231811 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251812 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231813 mock_quic_data_.AddWrite(SYNCHRONOUS,
1814 ConstructSettingsPacket(packet_number++));
1815 }
1816 mock_quic_data_.AddWrite(SYNCHRONOUS,
1817 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431818 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551819 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1820 mock_quic_data_.AddWrite(SYNCHRONOUS, ERR_IO_PENDING);
1821
1822 Initialize();
1823
1824 AssertConnectSucceeds();
1825
1826 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1827 // asynchronous starting with the second time it's called while the UDP socket
1828 // is write-blocked. Therefore, at least two writes need to be called on
1829 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351830 AssertWriteReturns(kMsg1, kLen1, kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:551831
1832 // This second write will be async. This is the pending write that's being
1833 // tested.
1834 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
1835
1836 // Make sure the write actually starts.
1837 base::RunLoop().RunUntilIdle();
1838
1839 sock_->Disconnect();
1840 EXPECT_FALSE(sock_->IsConnected());
1841
1842 base::RunLoop().RunUntilIdle();
1843
1844 EXPECT_FALSE(sock_->IsConnected());
1845 EXPECT_FALSE(write_callback_.have_result());
1846}
1847
1848// If the socket is Disconnected with a pending Read(), the callback
1849// should not be called.
1850TEST_P(QuicProxyClientSocketTest, DisconnectWithReadPending) {
Renjie Tangaadb84b2019-08-31 01:00:231851 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251852 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231853 mock_quic_data_.AddWrite(SYNCHRONOUS,
1854 ConstructSettingsPacket(packet_number++));
1855 }
1856 mock_quic_data_.AddWrite(SYNCHRONOUS,
1857 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431858 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551859 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1860 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341861 SYNCHRONOUS, ConstructAckAndRstPacket(packet_number++,
1862 quic::QUIC_STREAM_CANCELLED, 1, 1));
Yixin Wang0d2c6b7e12017-08-16 21:12:551863
1864 Initialize();
1865
1866 AssertConnectSucceeds();
1867
1868 EXPECT_TRUE(sock_->IsConnected());
1869
1870 AssertReadStarts(kMsg1, kLen1);
1871
1872 sock_->Disconnect();
1873 EXPECT_FALSE(sock_->IsConnected());
1874
1875 base::RunLoop().RunUntilIdle();
1876
1877 EXPECT_FALSE(sock_->IsConnected());
1878 EXPECT_FALSE(read_callback_.have_result());
1879}
1880
1881// If the socket is Reset when both a read and write are pending,
1882// both should be called back.
1883TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePending) {
Renjie Tangaadb84b2019-08-31 01:00:231884 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251885 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231886 mock_quic_data_.AddWrite(SYNCHRONOUS,
1887 ConstructSettingsPacket(packet_number++));
1888 }
1889 mock_quic_data_.AddWrite(SYNCHRONOUS,
1890 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431891 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551892 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1893
1894 mock_quic_data_.AddRead(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411895 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED));
Yixin Wang2bea3cf2017-11-09 18:11:031896 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Victor Vasiliev076657c2019-03-12 02:46:431897 std::string header = ConstructDataHeader(kLen2);
Nick Harperc6cb7a612020-02-24 20:03:321898 if (!version_.HasIetfQuicFrames()) {
Renjie Tangaadb84b2019-08-31 01:00:231899 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341900 ASYNC, ConstructAckAndDataPacket(packet_number++, 1, 1,
Renjie Tangaadb84b2019-08-31 01:00:231901 std::string(kMsg2, kLen2)));
Renjief49758b2019-01-11 23:32:411902 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411903 SYNCHRONOUS,
Renjie Tangaadb84b2019-08-31 01:00:231904 ConstructAckAndRstPacket(packet_number++,
Renjie Tangcd594f32020-07-11 20:18:341905 quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2));
Renjief49758b2019-01-11 23:32:411906 } else {
Renjie Tangd5133972019-12-06 00:20:281907 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341908 ASYNC, ConstructAckAndDataPacket(packet_number++, 1, 1,
Renjie Tangd5133972019-12-06 00:20:281909 {header + std::string(kMsg2, kLen2)}));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411910 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:341911 SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
1912 packet_number++, quic::QUIC_STREAM_CANCELLED, 2, 2));
Renjief49758b2019-01-11 23:32:411913 }
Yixin Wang0d2c6b7e12017-08-16 21:12:551914
1915 Initialize();
1916
1917 AssertConnectSucceeds();
1918
1919 EXPECT_TRUE(sock_->IsConnected());
1920
1921 AssertReadStarts(kMsg1, kLen1);
1922
1923 // Write should block until the next read completes.
1924 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
1925 // asynchronous starting with the second time it's called while the UDP socket
1926 // is write-blocked. Therefore, at least two writes need to be called on
1927 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:351928 AssertWriteReturns(kMsg2, kLen2, kLen2);
Renjief49758b2019-01-11 23:32:411929
Yixin Wang0d2c6b7e12017-08-16 21:12:551930 AssertWriteReturns(kMsg2, kLen2, ERR_IO_PENDING);
1931
1932 ResumeAndRun();
1933
1934 EXPECT_TRUE(read_callback_.have_result());
1935 EXPECT_TRUE(write_callback_.have_result());
1936}
1937
1938// Makes sure the proxy client socket's source gets the expected NetLog events
1939// and only the expected NetLog events (No SpdySession events).
1940TEST_P(QuicProxyClientSocketTest, NetLog) {
Renjie Tangaadb84b2019-08-31 01:00:231941 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:251942 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:231943 mock_quic_data_.AddWrite(SYNCHRONOUS,
1944 ConstructSettingsPacket(packet_number++));
1945 }
1946 mock_quic_data_.AddWrite(SYNCHRONOUS,
1947 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:431948 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:551949 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
1950
Victor Vasiliev076657c2019-03-12 02:46:431951 std::string header = ConstructDataHeader(kLen1);
Ryan Hamilton7505eb92019-06-08 00:22:171952 mock_quic_data_.AddRead(
1953 ASYNC, ConstructServerDataPacket(2, header + std::string(kMsg1, kLen1)));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:411954 mock_quic_data_.AddWrite(SYNCHRONOUS,
Renjie Tangcd594f32020-07-11 20:18:341955 ConstructAckPacket(packet_number++, 2, 1));
Renjie Tangaadb84b2019-08-31 01:00:231956 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
1957 mock_quic_data_.AddWrite(
1958 SYNCHRONOUS,
1959 ConstructRstPacket(packet_number++, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:551960
1961 Initialize();
1962
1963 AssertConnectSucceeds();
1964
1965 ResumeAndRun();
1966 AssertSyncReadEquals(kMsg1, kLen1);
1967
1968 NetLogSource sock_source = sock_->NetLog().source();
1969 sock_.reset();
1970
Matt Reichhoff0049a0b72021-10-20 20:44:261971 auto entry_list = net_log_observer_.GetEntriesForSource(sock_source);
Yixin Wang0d2c6b7e12017-08-16 21:12:551972
1973 ASSERT_EQ(entry_list.size(), 10u);
1974 EXPECT_TRUE(
1975 LogContainsBeginEvent(entry_list, 0, NetLogEventType::SOCKET_ALIVE));
1976 EXPECT_TRUE(LogContainsEvent(entry_list, 1,
1977 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
1978 NetLogEventPhase::NONE));
1979 EXPECT_TRUE(LogContainsBeginEvent(
1980 entry_list, 2, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1981 EXPECT_TRUE(LogContainsEvent(
1982 entry_list, 3, NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
1983 NetLogEventPhase::NONE));
1984 EXPECT_TRUE(LogContainsEndEvent(
1985 entry_list, 4, NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST));
1986 EXPECT_TRUE(LogContainsBeginEvent(
1987 entry_list, 5, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1988 EXPECT_TRUE(LogContainsEvent(
1989 entry_list, 6,
1990 NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
1991 NetLogEventPhase::NONE));
1992 EXPECT_TRUE(LogContainsEndEvent(
1993 entry_list, 7, NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS));
1994 EXPECT_TRUE(LogContainsEvent(entry_list, 8,
1995 NetLogEventType::SOCKET_BYTES_RECEIVED,
1996 NetLogEventPhase::NONE));
1997 EXPECT_TRUE(
1998 LogContainsEndEvent(entry_list, 9, NetLogEventType::SOCKET_ALIVE));
1999}
2000
Bence Béky8ddc2492018-06-13 01:02:042001// A helper class that will delete |sock| when the callback is invoked.
Yixin Wang0d2c6b7e12017-08-16 21:12:552002class DeleteSockCallback : public TestCompletionCallbackBase {
2003 public:
2004 explicit DeleteSockCallback(std::unique_ptr<QuicProxyClientSocket>* sock)
Bence Béky8ddc2492018-06-13 01:02:042005 : sock_(sock) {}
Yixin Wang0d2c6b7e12017-08-16 21:12:552006
Peter Boström293b1342021-09-22 17:31:432007 DeleteSockCallback(const DeleteSockCallback&) = delete;
2008 DeleteSockCallback& operator=(const DeleteSockCallback&) = delete;
2009
Yixin Wang0d2c6b7e12017-08-16 21:12:552010 ~DeleteSockCallback() override {}
2011
Bence Béky8ddc2492018-06-13 01:02:042012 CompletionOnceCallback callback() {
2013 return base::BindOnce(&DeleteSockCallback::OnComplete,
2014 base::Unretained(this));
2015 }
Yixin Wang0d2c6b7e12017-08-16 21:12:552016
2017 private:
2018 void OnComplete(int result) {
Raul Tambre94493c652019-03-11 17:18:352019 sock_->reset(nullptr);
Yixin Wang0d2c6b7e12017-08-16 21:12:552020 SetResult(result);
2021 }
2022
Keishi Hattori0e45c022021-11-27 09:25:522023 raw_ptr<std::unique_ptr<QuicProxyClientSocket>> sock_;
Yixin Wang0d2c6b7e12017-08-16 21:12:552024};
2025
2026// If the socket is reset when both a read and write are pending, and the
2027// read callback causes the socket to be deleted, the write callback should
2028// not be called.
2029TEST_P(QuicProxyClientSocketTest, RstWithReadAndWritePendingDelete) {
Renjie Tangaadb84b2019-08-31 01:00:232030 int packet_number = 1;
Victor Vasiliev7da08172019-10-14 06:04:252031 if (VersionUsesHttp3(version_.transport_version)) {
Renjie Tangaadb84b2019-08-31 01:00:232032 mock_quic_data_.AddWrite(SYNCHRONOUS,
2033 ConstructSettingsPacket(packet_number++));
2034 }
2035 mock_quic_data_.AddWrite(SYNCHRONOUS,
2036 ConstructConnectRequestPacket(packet_number++));
Zhongyi Shi32f2fd02018-04-16 18:23:432037 mock_quic_data_.AddRead(ASYNC, ConstructServerConnectReplyPacket(1, !kFin));
Yixin Wang0d2c6b7e12017-08-16 21:12:552038 mock_quic_data_.AddRead(ASYNC, ERR_IO_PENDING); // Pause
2039
2040 mock_quic_data_.AddRead(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:412041 ASYNC, ConstructServerRstPacket(2, quic::QUIC_STREAM_CANCELLED));
Yixin Wang0d2c6b7e12017-08-16 21:12:552042 mock_quic_data_.AddRead(SYNCHRONOUS, ERR_IO_PENDING);
Nick Harperc6cb7a612020-02-24 20:03:322043 if (!version_.HasIetfQuicFrames()) {
Renjie Tangaadb84b2019-08-31 01:00:232044 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:342045 ASYNC, ConstructAckAndDataPacket(packet_number++, 1, 1,
Renjie Tangaadb84b2019-08-31 01:00:232046 std::string(kMsg1, kLen1)));
Renjief49758b2019-01-11 23:32:412047 mock_quic_data_.AddWrite(
Ryan Hamiltonb5d4c5a2019-06-21 22:08:412048 SYNCHRONOUS,
Renjie Tangaadb84b2019-08-31 01:00:232049 ConstructAckAndRstPacket(packet_number++,
Renjie Tangcd594f32020-07-11 20:18:342050 quic::QUIC_RST_ACKNOWLEDGEMENT, 2, 2));
Renjief49758b2019-01-11 23:32:412051 } else {
Victor Vasiliev076657c2019-03-12 02:46:432052 std::string header = ConstructDataHeader(kLen1);
Renjie Tangd5133972019-12-06 00:20:282053 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:342054 ASYNC, ConstructAckAndDataPacket(packet_number++, 1, 1,
Renjie Tangd5133972019-12-06 00:20:282055 {header + std::string(kMsg1, kLen1)}));
Ryan Hamiltonb5d4c5a2019-06-21 22:08:412056 mock_quic_data_.AddWrite(
Renjie Tangcd594f32020-07-11 20:18:342057 SYNCHRONOUS, ConstructAckAndRstOnlyPacket(
2058 packet_number++, quic::QUIC_STREAM_CANCELLED, 2, 2));
Renjief49758b2019-01-11 23:32:412059 }
Yixin Wang0d2c6b7e12017-08-16 21:12:552060
2061 Initialize();
2062
2063 AssertConnectSucceeds();
2064
2065 EXPECT_TRUE(sock_->IsConnected());
2066
2067 DeleteSockCallback read_callback(&sock_);
Victor Costan9c7302b2018-08-27 16:39:442068 scoped_refptr<IOBuffer> read_buf = base::MakeRefCounted<IOBuffer>(kLen1);
Yixin Wang0d2c6b7e12017-08-16 21:12:552069 ASSERT_EQ(ERR_IO_PENDING,
2070 sock_->Read(read_buf.get(), kLen1, read_callback.callback()));
2071
2072 // QuicChromiumClientStream::Handle::WriteStreamData() will only be
2073 // asynchronous starting with the second time it's called while the UDP socket
2074 // is write-blocked. Therefore, at least two writes need to be called on
2075 // |sock_| to get an asynchronous one.
Renjied172e812019-01-16 05:12:352076 AssertWriteReturns(kMsg1, kLen1, kLen1);
Renjief49758b2019-01-11 23:32:412077
Yixin Wang0d2c6b7e12017-08-16 21:12:552078 AssertWriteReturns(kMsg1, kLen1, ERR_IO_PENDING);
2079
2080 ResumeAndRun();
2081
2082 EXPECT_FALSE(sock_.get());
2083
2084 EXPECT_EQ(0, read_callback.WaitForResult());
2085 EXPECT_FALSE(write_callback_.have_result());
2086}
2087
David Schinazi09e9a6012019-10-03 17:37:572088INSTANTIATE_TEST_SUITE_P(VersionIncludeStreamDependencySequence,
2089 QuicProxyClientSocketTest,
2090 ::testing::ValuesIn(GetTestParams()),
2091 ::testing::PrintToStringParamName());
Yixin Wang0d2c6b7e12017-08-16 21:12:552092
2093} // namespace test
Dan Zhangf11470172017-09-18 22:02:092094} // namespace net