blob: 6fd314fa809a77ed20b711fcd86b4ead301a94f6 [file] [log] [blame]
[email protected]e92fa7e12012-02-16 23:31:221// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]fe2f62a2010-10-01 03:34:072// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Bence Béky94658bf2018-05-11 19:22:585#include "net/spdy/spdy_proxy_client_socket.h"
[email protected]fe2f62a2010-10-01 03:34:076
7#include <algorithm> // min
dchengc7eeda422015-12-26 03:56:488#include <utility>
[email protected]fe2f62a2010-10-01 03:34:079
[email protected]49639fa2011-12-20 23:22:4110#include "base/bind.h"
[email protected]aa19cfc2013-05-23 16:41:3811#include "base/callback_helpers.h"
Hans Wennborg0924470b2020-04-27 21:08:0512#include "base/check_op.h"
skyostil4891b25b2015-06-11 11:43:4513#include "base/location.h"
Hans Wennborg0924470b2020-04-27 21:08:0514#include "base/notreached.h"
skyostil4891b25b2015-06-11 11:43:4515#include "base/single_thread_task_runner.h"
[email protected]fc9be5802013-06-11 10:56:5116#include "base/strings/string_util.h"
gabf767595f2016-05-11 18:50:3517#include "base/threading/thread_task_runner_handle.h"
[email protected]f6c63db52013-02-02 00:35:2218#include "base/values.h"
[email protected]fe2f62a2010-10-01 03:34:0719#include "net/base/auth.h"
20#include "net/base/io_buffer.h"
Robert Ogden78d4f9eb2020-03-17 20:56:3821#include "net/base/proxy_delegate.h"
[email protected]fe3b7dc2012-02-03 19:52:0922#include "net/http/http_auth_cache.h"
23#include "net/http/http_auth_handler_factory.h"
Eric Roman06bd9742019-07-13 15:19:1324#include "net/http/http_log_util.h"
rchc5c07de2015-04-08 07:28:1825#include "net/http/http_request_info.h"
[email protected]b104b502010-10-18 20:21:3126#include "net/http/http_response_headers.h"
mikecirone8b85c432016-09-08 19:11:0027#include "net/log/net_log_event_type.h"
28#include "net/log/net_log_source_type.h"
Bence Béky94658bf2018-05-11 19:22:5829#include "net/spdy/spdy_http_utils.h"
[email protected]a2b2cfc2017-12-06 09:06:0830#include "net/traffic_annotation/network_traffic_annotation.h"
[email protected]f89276a72013-07-12 06:41:5431#include "url/gurl.h"
[email protected]fe2f62a2010-10-01 03:34:0732
33namespace net {
34
35SpdyProxyClientSocket::SpdyProxyClientSocket(
[email protected]d26ff352013-05-13 08:48:2836 const base::WeakPtr<SpdyStream>& spdy_stream,
Robert Ogden78d4f9eb2020-03-17 20:56:3837 const ProxyServer& proxy_server,
Bence Béky4e83f492018-05-13 23:14:2538 const std::string& user_agent,
[email protected]fe2f62a2010-10-01 03:34:0739 const HostPortPair& endpoint,
tfarina42834112016-09-22 13:38:2040 const NetLogWithSource& source_net_log,
Robert Ogden78d4f9eb2020-03-17 20:56:3841 HttpAuthController* auth_controller,
42 ProxyDelegate* proxy_delegate)
[email protected]49639fa2011-12-20 23:22:4143 : next_state_(STATE_DISCONNECTED),
[email protected]fe2f62a2010-10-01 03:34:0744 spdy_stream_(spdy_stream),
[email protected]fe2f62a2010-10-01 03:34:0745 endpoint_(endpoint),
mmenke2a1781d2015-10-07 19:25:3346 auth_(auth_controller),
Robert Ogden78d4f9eb2020-03-17 20:56:3847 proxy_server_(proxy_server),
48 proxy_delegate_(proxy_delegate),
rchecd3c552015-04-07 20:53:5449 user_agent_(user_agent),
[email protected]ca690b02013-04-17 10:38:4350 user_buffer_len_(0),
[email protected]fe2f62a2010-10-01 03:34:0751 write_buffer_len_(0),
[email protected]57d2dfa2013-06-24 06:04:1252 was_ever_used_(false),
tfarina42834112016-09-22 13:38:2053 net_log_(NetLogWithSource::Make(spdy_stream->net_log().net_log(),
54 NetLogSourceType::PROXY_CLIENT_SOCKET)),
Jeremy Romand54000b22019-07-08 18:40:1655 source_dependency_(source_net_log.source()) {
[email protected]fe2f62a2010-10-01 03:34:0756 request_.method = "CONNECT";
rchecd3c552015-04-07 20:53:5457 request_.url = GURL("https://" + endpoint.ToString());
Eric Roman06bd9742019-07-13 15:19:1358 net_log_.BeginEventReferencingSource(NetLogEventType::SOCKET_ALIVE,
59 source_net_log.source());
60 net_log_.AddEventReferencingSource(
mikecirone8b85c432016-09-08 19:11:0061 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
Eric Roman06bd9742019-07-13 15:19:1362 spdy_stream->net_log().source());
[email protected]f6c63db52013-02-02 00:35:2263
[email protected]fe2f62a2010-10-01 03:34:0764 spdy_stream_->SetDelegate(this);
65 was_ever_used_ = spdy_stream_->WasEverUsed();
66}
67
68SpdyProxyClientSocket::~SpdyProxyClientSocket() {
69 Disconnect();
mikecirone8b85c432016-09-08 19:11:0070 net_log_.EndEvent(NetLogEventType::SOCKET_ALIVE);
[email protected]fe2f62a2010-10-01 03:34:0771}
72
[email protected]be1a48b2011-01-20 00:12:1373const HttpResponseInfo* SpdyProxyClientSocket::GetConnectResponseInfo() const {
Raul Tambre94493c652019-03-11 17:18:3574 return response_.headers.get() ? &response_ : nullptr;
[email protected]be1a48b2011-01-20 00:12:1375}
76
[email protected]c0fe941d2012-02-25 00:15:3277const scoped_refptr<HttpAuthController>&
78SpdyProxyClientSocket::GetAuthController() const {
79 return auth_;
80}
81
Bence Béky99832232018-02-11 04:21:0082int SpdyProxyClientSocket::RestartWithAuth(CompletionOnceCallback callback) {
[email protected]c0fe941d2012-02-25 00:15:3283 // A SPDY Stream can only handle a single request, so the underlying
84 // stream may not be reused and a new SpdyProxyClientSocket must be
85 // created (possibly on top of the same SPDY Session).
86 next_state_ = STATE_DISCONNECTED;
Matt Menkeae2cbb82019-02-21 20:20:1387 return ERR_UNABLE_TO_REUSE_CONNECTION_FOR_PROXY_AUTH;
[email protected]c0fe941d2012-02-25 00:15:3288}
89
90bool SpdyProxyClientSocket::IsUsingSpdy() const {
91 return true;
92}
93
bnc6227b26e2016-08-12 02:00:4394NextProto SpdyProxyClientSocket::GetProxyNegotiatedProtocol() const {
bnc4cb660c2016-08-11 19:43:1295 return spdy_stream_->GetNegotiatedProtocol();
[email protected]c0fe941d2012-02-25 00:15:3296}
97
Matt Menkeedaf3b82019-03-14 21:39:4498// Ignore priority changes, just use priority of initial request. Since multiple
99// requests are pooled on the SpdyProxyClientSocket, reprioritization doesn't
100// really work.
101//
102// TODO(mmenke): Use a single priority value for all SpdyProxyClientSockets,
103// regardless of what priority they're created with.
104void SpdyProxyClientSocket::SetStreamPriority(RequestPriority priority) {}
Lily Chenf11e1292018-11-29 16:42:09105
bnc42331402016-07-25 13:36:15106// Sends a HEADERS frame to the proxy with a CONNECT request
[email protected]fe2f62a2010-10-01 03:34:07107// for the specified endpoint. Waits for the server to send back
bnc42331402016-07-25 13:36:15108// a HEADERS frame. OK will be returned if the status is 200.
[email protected]fe2f62a2010-10-01 03:34:07109// ERR_TUNNEL_CONNECTION_FAILED will be returned for any other status.
110// In any of these cases, Read() may be called to retrieve the HTTP
111// response body. Any other return values should be considered fatal.
[email protected]fe3b7dc2012-02-03 19:52:09112// TODO(rch): handle 407 proxy auth requested correctly, perhaps
113// by creating a new stream for the subsequent request.
[email protected]fe2f62a2010-10-01 03:34:07114// TODO(rch): create a more appropriate error code to disambiguate
115// the HTTPS Proxy tunnel failure from an HTTP Proxy tunnel failure.
Brad Lassey3a814172018-04-26 03:30:21116int SpdyProxyClientSocket::Connect(CompletionOnceCallback callback) {
[email protected]83039bb2011-12-09 18:43:55117 DCHECK(read_callback_.is_null());
[email protected]d9da5fe2010-10-13 22:37:16118 if (next_state_ == STATE_OPEN)
[email protected]fe2f62a2010-10-01 03:34:07119 return OK;
120
[email protected]d9da5fe2010-10-13 22:37:16121 DCHECK_EQ(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07122 next_state_ = STATE_GENERATE_AUTH_TOKEN;
123
124 int rv = DoLoop(OK);
125 if (rv == ERR_IO_PENDING)
Bence Békya25e3f72018-02-13 21:13:39126 read_callback_ = std::move(callback);
[email protected]fe2f62a2010-10-01 03:34:07127 return rv;
128}
129
130void SpdyProxyClientSocket::Disconnect() {
[email protected]ca690b02013-04-17 10:38:43131 read_buffer_queue_.Clear();
Raul Tambre94493c652019-03-11 17:18:35132 user_buffer_ = nullptr;
[email protected]ca690b02013-04-17 10:38:43133 user_buffer_len_ = 0;
[email protected]dbf036f2011-12-06 23:33:24134 read_callback_.Reset();
[email protected]d9da5fe2010-10-13 22:37:16135
136 write_buffer_len_ = 0;
[email protected]83039bb2011-12-09 18:43:55137 write_callback_.Reset();
[email protected]0a428c22014-06-14 02:26:37138 write_callback_weak_factory_.InvalidateWeakPtrs();
[email protected]d9da5fe2010-10-13 22:37:16139
140 next_state_ = STATE_DISCONNECTED;
141
[email protected]11fbca0b2013-06-02 23:37:21142 if (spdy_stream_.get()) {
[email protected]fe2f62a2010-10-01 03:34:07143 // This will cause OnClose to be invoked, which takes care of
144 // cleaning up all the internal state.
Bence Béky6b9c1352018-05-10 11:51:25145 spdy_stream_->Cancel(ERR_ABORTED);
[email protected]11fbca0b2013-06-02 23:37:21146 DCHECK(!spdy_stream_.get());
[email protected]f6a78292013-03-09 14:36:34147 }
[email protected]fe2f62a2010-10-01 03:34:07148}
149
150bool SpdyProxyClientSocket::IsConnected() const {
[email protected]194c7a92011-12-03 04:54:18151 return next_state_ == STATE_OPEN;
[email protected]fe2f62a2010-10-01 03:34:07152}
153
154bool SpdyProxyClientSocket::IsConnectedAndIdle() const {
[email protected]ca690b02013-04-17 10:38:43155 return IsConnected() && read_buffer_queue_.IsEmpty() &&
[email protected]5ff0ed32014-02-12 17:48:51156 spdy_stream_->IsOpen();
[email protected]fe2f62a2010-10-01 03:34:07157}
158
tfarina42834112016-09-22 13:38:20159const NetLogWithSource& SpdyProxyClientSocket::NetLog() const {
[email protected]e4be2dd2010-12-14 00:44:39160 return net_log_;
161}
162
[email protected]fe2f62a2010-10-01 03:34:07163bool SpdyProxyClientSocket::WasEverUsed() const {
[email protected]11fbca0b2013-06-02 23:37:21164 return was_ever_used_ || (spdy_stream_.get() && spdy_stream_->WasEverUsed());
[email protected]fe2f62a2010-10-01 03:34:07165}
166
tfarina2846404c2016-12-25 14:31:37167bool SpdyProxyClientSocket::WasAlpnNegotiated() const {
[email protected]2d88e7d2012-07-19 17:55:17168 return false;
169}
170
[email protected]33661e482012-04-03 16:16:26171NextProto SpdyProxyClientSocket::GetNegotiatedProtocol() const {
172 return kProtoUnknown;
173}
174
[email protected]2d88e7d2012-07-19 17:55:17175bool SpdyProxyClientSocket::GetSSLInfo(SSLInfo* ssl_info) {
bnc4cb660c2016-08-11 19:43:12176 return spdy_stream_->GetSSLInfo(ssl_info);
[email protected]2d88e7d2012-07-19 17:55:17177}
178
ttuttle23fdb7b2015-05-15 01:28:03179void SpdyProxyClientSocket::GetConnectionAttempts(
180 ConnectionAttempts* out) const {
181 out->clear();
182}
183
tbansalf82cc8e2015-10-14 20:05:49184int64_t SpdyProxyClientSocket::GetTotalReceivedBytes() const {
185 NOTIMPLEMENTED();
186 return 0;
187}
188
Paul Jensen0f49dec2017-12-12 23:39:58189void SpdyProxyClientSocket::ApplySocketTag(const SocketTag& tag) {
Matt Menke5b7974fa2019-03-09 01:18:12190 // In the case of a connection to the proxy using HTTP/2 or HTTP/3 where the
191 // underlying socket may multiplex multiple streams, applying this request's
192 // socket tag to the multiplexed session would incorrectly apply the socket
193 // tag to all mutliplexed streams. Fortunately socket tagging is only
194 // supported on Android without the data reduction proxy, so only simple HTTP
195 // proxies are supported, so proxies won't be using HTTP/2 or HTTP/3. Enforce
196 // that a specific (non-default) tag isn't being applied.
197 CHECK(tag == SocketTag());
Paul Jensen0f49dec2017-12-12 23:39:58198}
199
Bence Békya25e3f72018-02-13 21:13:39200int SpdyProxyClientSocket::Read(IOBuffer* buf,
201 int buf_len,
Brad Lassey3a814172018-04-26 03:30:21202 CompletionOnceCallback callback) {
Helen Lid7141882018-08-22 21:41:17203 int rv = ReadIfReady(buf, buf_len, std::move(callback));
204 if (rv == ERR_IO_PENDING) {
205 user_buffer_ = buf;
206 user_buffer_len_ = static_cast<size_t>(buf_len);
207 }
208 return rv;
209}
210
211int SpdyProxyClientSocket::ReadIfReady(IOBuffer* buf,
212 int buf_len,
213 CompletionOnceCallback callback) {
214 DCHECK(!read_callback_);
215 DCHECK(!user_buffer_);
[email protected]3f55aa12011-12-07 02:03:33216
217 if (next_state_ == STATE_DISCONNECTED)
218 return ERR_SOCKET_NOT_CONNECTED;
219
[email protected]ca690b02013-04-17 10:38:43220 if (next_state_ == STATE_CLOSED && read_buffer_queue_.IsEmpty()) {
[email protected]3f55aa12011-12-07 02:03:33221 return 0;
222 }
223
224 DCHECK(next_state_ == STATE_OPEN || next_state_ == STATE_CLOSED);
225 DCHECK(buf);
[email protected]ca690b02013-04-17 10:38:43226 size_t result = PopulateUserReadBuffer(buf->data(), buf_len);
[email protected]3f55aa12011-12-07 02:03:33227 if (result == 0) {
Brad Lassey3a814172018-04-26 03:30:21228 read_callback_ = std::move(callback);
[email protected]3f55aa12011-12-07 02:03:33229 return ERR_IO_PENDING;
230 }
[email protected]3f55aa12011-12-07 02:03:33231 return result;
232}
[email protected]fe2f62a2010-10-01 03:34:07233
Helen Lid7141882018-08-22 21:41:17234int SpdyProxyClientSocket::CancelReadIfReady() {
235 // Only a pending ReadIfReady() can be canceled.
236 DCHECK(!user_buffer_) << "Pending Read() cannot be canceled";
237 read_callback_.Reset();
238 return OK;
239}
240
[email protected]ca690b02013-04-17 10:38:43241size_t SpdyProxyClientSocket::PopulateUserReadBuffer(char* data, size_t len) {
[email protected]09a8d9172013-04-17 19:23:49242 return read_buffer_queue_.Dequeue(data, len);
[email protected]fe2f62a2010-10-01 03:34:07243}
244
[email protected]a2b2cfc2017-12-06 09:06:08245int SpdyProxyClientSocket::Write(
246 IOBuffer* buf,
247 int buf_len,
Brad Lassey3a814172018-04-26 03:30:21248 CompletionOnceCallback callback,
[email protected]a2b2cfc2017-12-06 09:06:08249 const NetworkTrafficAnnotationTag& traffic_annotation) {
[email protected]83039bb2011-12-09 18:43:55250 DCHECK(write_callback_.is_null());
[email protected]194c7a92011-12-03 04:54:18251 if (next_state_ != STATE_OPEN)
[email protected]d9da5fe2010-10-13 22:37:16252 return ERR_SOCKET_NOT_CONNECTED;
253
[email protected]11fbca0b2013-06-02 23:37:21254 DCHECK(spdy_stream_.get());
[email protected]edbfa8c2013-05-29 00:22:33255 spdy_stream_->SendData(buf, buf_len, MORE_DATA_TO_SEND);
mikecirone8b85c432016-09-08 19:11:00256 net_log_.AddByteTransferEvent(NetLogEventType::SOCKET_BYTES_SENT, buf_len,
257 buf->data());
Brad Lassey3a814172018-04-26 03:30:21258 write_callback_ = std::move(callback);
[email protected]aa19cfc2013-05-23 16:41:38259 write_buffer_len_ = buf_len;
260 return ERR_IO_PENDING;
[email protected]fe2f62a2010-10-01 03:34:07261}
262
Avi Drissman13fc8932015-12-20 04:40:46263int SpdyProxyClientSocket::SetReceiveBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10264 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15265 // is not safe for callers to change this underlying socket.
266 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07267}
268
Avi Drissman13fc8932015-12-20 04:40:46269int SpdyProxyClientSocket::SetSendBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10270 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15271 // is not safe for callers to change this underlying socket.
272 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07273}
274
[email protected]a3528692012-06-08 00:11:42275int SpdyProxyClientSocket::GetPeerAddress(IPEndPoint* address) const {
[email protected]fe2f62a2010-10-01 03:34:07276 if (!IsConnected())
[email protected]88e03fa2010-10-05 03:09:04277 return ERR_SOCKET_NOT_CONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07278 return spdy_stream_->GetPeerAddress(address);
279}
280
[email protected]e7f74da2011-04-19 23:49:35281int SpdyProxyClientSocket::GetLocalAddress(IPEndPoint* address) const {
282 if (!IsConnected())
283 return ERR_SOCKET_NOT_CONNECTED;
284 return spdy_stream_->GetLocalAddress(address);
285}
286
Bence Békya25e3f72018-02-13 21:13:39287void SpdyProxyClientSocket::RunCallback(CompletionOnceCallback callback,
[email protected]0a428c22014-06-14 02:26:37288 int result) const {
Bence Békya25e3f72018-02-13 21:13:39289 std::move(callback).Run(result);
[email protected]0a428c22014-06-14 02:26:37290}
291
[email protected]fe2f62a2010-10-01 03:34:07292void SpdyProxyClientSocket::OnIOComplete(int result) {
[email protected]d9da5fe2010-10-13 22:37:16293 DCHECK_NE(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07294 int rv = DoLoop(result);
295 if (rv != ERR_IO_PENDING) {
Brad Lassey3a814172018-04-26 03:30:21296 std::move(read_callback_).Run(rv);
[email protected]fe2f62a2010-10-01 03:34:07297 }
298}
299
300int SpdyProxyClientSocket::DoLoop(int last_io_result) {
[email protected]d9da5fe2010-10-13 22:37:16301 DCHECK_NE(next_state_, STATE_DISCONNECTED);
[email protected]fe2f62a2010-10-01 03:34:07302 int rv = last_io_result;
303 do {
304 State state = next_state_;
[email protected]d9da5fe2010-10-13 22:37:16305 next_state_ = STATE_DISCONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07306 switch (state) {
307 case STATE_GENERATE_AUTH_TOKEN:
308 DCHECK_EQ(OK, rv);
309 rv = DoGenerateAuthToken();
310 break;
311 case STATE_GENERATE_AUTH_TOKEN_COMPLETE:
312 rv = DoGenerateAuthTokenComplete(rv);
313 break;
314 case STATE_SEND_REQUEST:
315 DCHECK_EQ(OK, rv);
mikecirone8b85c432016-09-08 19:11:00316 net_log_.BeginEvent(
317 NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST);
[email protected]fe2f62a2010-10-01 03:34:07318 rv = DoSendRequest();
319 break;
320 case STATE_SEND_REQUEST_COMPLETE:
[email protected]d7fd1782011-02-08 19:16:43321 net_log_.EndEventWithNetErrorCode(
mikecirone8b85c432016-09-08 19:11:00322 NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST, rv);
[email protected]fe2f62a2010-10-01 03:34:07323 rv = DoSendRequestComplete(rv);
[email protected]f6c63db52013-02-02 00:35:22324 if (rv >= 0 || rv == ERR_IO_PENDING) {
325 // Emit extra event so can use the same events as
326 // HttpProxyClientSocket.
327 net_log_.BeginEvent(
mikecirone8b85c432016-09-08 19:11:00328 NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS);
[email protected]f6c63db52013-02-02 00:35:22329 }
[email protected]fe2f62a2010-10-01 03:34:07330 break;
331 case STATE_READ_REPLY_COMPLETE:
332 rv = DoReadReplyComplete(rv);
[email protected]d7fd1782011-02-08 19:16:43333 net_log_.EndEventWithNetErrorCode(
mikecirone8b85c432016-09-08 19:11:00334 NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS, rv);
[email protected]fe2f62a2010-10-01 03:34:07335 break;
336 default:
337 NOTREACHED() << "bad state";
338 rv = ERR_UNEXPECTED;
339 break;
340 }
[email protected]d9da5fe2010-10-13 22:37:16341 } while (rv != ERR_IO_PENDING && next_state_ != STATE_DISCONNECTED &&
342 next_state_ != STATE_OPEN);
[email protected]fe2f62a2010-10-01 03:34:07343 return rv;
344}
345
346int SpdyProxyClientSocket::DoGenerateAuthToken() {
347 next_state_ = STATE_GENERATE_AUTH_TOKEN_COMPLETE;
[email protected]49639fa2011-12-20 23:22:41348 return auth_->MaybeGenerateAuthToken(
349 &request_,
Yannic Bonenberger00e09842019-08-31 20:46:19350 base::BindOnce(&SpdyProxyClientSocket::OnIOComplete,
351 weak_factory_.GetWeakPtr()),
[email protected]49639fa2011-12-20 23:22:41352 net_log_);
[email protected]fe2f62a2010-10-01 03:34:07353}
354
355int SpdyProxyClientSocket::DoGenerateAuthTokenComplete(int result) {
356 DCHECK_NE(ERR_IO_PENDING, result);
357 if (result == OK)
358 next_state_ = STATE_SEND_REQUEST;
359 return result;
360}
361
362int SpdyProxyClientSocket::DoSendRequest() {
363 next_state_ = STATE_SEND_REQUEST_COMPLETE;
364
365 // Add Proxy-Authentication header if necessary.
366 HttpRequestHeaders authorization_headers;
367 if (auth_->HaveAuth()) {
368 auth_->AddAuthorizationHeader(&authorization_headers);
369 }
370
Robert Ogden78d4f9eb2020-03-17 20:56:38371 if (proxy_delegate_) {
372 HttpRequestHeaders proxy_delegate_headers;
373 proxy_delegate_->OnBeforeTunnelRequest(proxy_server_,
374 &proxy_delegate_headers);
375 request_.extra_headers.MergeFrom(proxy_delegate_headers);
376 }
377
Bence Béky4e83f492018-05-13 23:14:25378 std::string request_line;
rchecd3c552015-04-07 20:53:54379 BuildTunnelRequest(endpoint_, authorization_headers, user_agent_,
380 &request_line, &request_.extra_headers);
[email protected]3abacd62012-06-10 20:20:32381
Eric Roman06bd9742019-07-13 15:19:13382 NetLogRequestHeaders(net_log_,
383 NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
384 request_line, &request_.extra_headers);
[email protected]fe2f62a2010-10-01 03:34:07385
Bence Béky4c325e52020-10-22 20:48:01386 spdy::Http2HeaderBlock headers;
Bence Béky1af94d6f2018-02-08 00:40:14387 CreateSpdyHeadersFromHttpRequest(request_, request_.extra_headers, &headers);
[email protected]fe2f62a2010-10-01 03:34:07388
dchengc7eeda422015-12-26 03:56:48389 return spdy_stream_->SendRequestHeaders(std::move(headers),
390 MORE_DATA_TO_SEND);
[email protected]fe2f62a2010-10-01 03:34:07391}
392
393int SpdyProxyClientSocket::DoSendRequestComplete(int result) {
394 if (result < 0)
395 return result;
396
bnc42331402016-07-25 13:36:15397 // Wait for HEADERS frame from the server
[email protected]fe2f62a2010-10-01 03:34:07398 next_state_ = STATE_READ_REPLY_COMPLETE;
399 return ERR_IO_PENDING;
400}
401
402int SpdyProxyClientSocket::DoReadReplyComplete(int result) {
403 // We enter this method directly from DoSendRequestComplete, since
bnc42331402016-07-25 13:36:15404 // we are notified by a callback when the HEADERS frame arrives.
[email protected]fe2f62a2010-10-01 03:34:07405
406 if (result < 0)
407 return result;
408
[email protected]fe2f62a2010-10-01 03:34:07409 // Require the "HTTP/1.x" status line for SSL CONNECT.
bncbe0f6af2015-10-15 17:49:56410 if (response_.headers->GetHttpVersion() < HttpVersion(1, 0))
[email protected]fe2f62a2010-10-01 03:34:07411 return ERR_TUNNEL_CONNECTION_FAILED;
412
Eric Roman06bd9742019-07-13 15:19:13413 NetLogResponseHeaders(
414 net_log_, NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
415 response_.headers.get());
[email protected]fe2f62a2010-10-01 03:34:07416
Robert Ogden78d4f9eb2020-03-17 20:56:38417 if (proxy_delegate_) {
418 int rv = proxy_delegate_->OnTunnelHeadersReceived(proxy_server_,
419 *response_.headers);
420 if (rv != OK) {
421 DCHECK_NE(ERR_IO_PENDING, rv);
422 return rv;
423 }
424 }
425
[email protected]4eddbc732012-08-09 05:40:17426 switch (response_.headers->response_code()) {
427 case 200: // OK
428 next_state_ = STATE_OPEN;
429 return OK;
430
[email protected]4eddbc732012-08-09 05:40:17431 case 407: // Proxy Authentication Required
432 next_state_ = STATE_OPEN;
Jeremy Roman4803d2332021-04-26 22:37:22433 SanitizeProxyAuth(response_);
[email protected]90499482013-06-01 00:39:50434 return HandleProxyAuthChallenge(auth_.get(), &response_, net_log_);
[email protected]4eddbc732012-08-09 05:40:17435
436 default:
437 // Ignore response to avoid letting the proxy impersonate the target
438 // server. (See http://crbug.com/137891.)
[email protected]4eddbc732012-08-09 05:40:17439 return ERR_TUNNEL_CONNECTION_FAILED;
[email protected]511f6f52010-12-17 03:58:29440 }
[email protected]fe2f62a2010-10-01 03:34:07441}
442
443// SpdyStream::Delegate methods:
444// Called when SYN frame has been sent.
445// Returns true if no more data to be sent after SYN frame.
bnc4c214312016-11-28 16:49:15446void SpdyProxyClientSocket::OnHeadersSent() {
[email protected]fe2f62a2010-10-01 03:34:07447 DCHECK_EQ(next_state_, STATE_SEND_REQUEST_COMPLETE);
448
[email protected]d46715c2013-04-15 00:21:42449 OnIOComplete(OK);
[email protected]fe2f62a2010-10-01 03:34:07450}
451
Kenichi Ishibashi74155532021-03-13 01:38:06452void SpdyProxyClientSocket::OnEarlyHintsReceived(
453 const spdy::Http2HeaderBlock& headers) {}
454
bnc4c214312016-11-28 16:49:15455void SpdyProxyClientSocket::OnHeadersReceived(
Bence Béky4c325e52020-10-22 20:48:01456 const spdy::Http2HeaderBlock& response_headers,
457 const spdy::Http2HeaderBlock* pushed_request_headers) {
[email protected]d08358502010-12-03 22:04:03458 // If we've already received the reply, existing headers are too late.
459 // TODO(mbelshe): figure out a way to make HEADERS frames useful after the
460 // initial response.
461 if (next_state_ != STATE_READ_REPLY_COMPLETE)
bnc4c214312016-11-28 16:49:15462 return;
[email protected]fe2f62a2010-10-01 03:34:07463
[email protected]d08358502010-12-03 22:04:03464 // Save the response
bnc4c214312016-11-28 16:49:15465 const bool headers_valid =
466 SpdyHeadersToHttpResponse(response_headers, &response_);
467 DCHECK(headers_valid);
[email protected]fe2f62a2010-10-01 03:34:07468
[email protected]6d116e1a2013-06-24 07:42:15469 OnIOComplete(OK);
[email protected]fe2f62a2010-10-01 03:34:07470}
471
[email protected]ca690b02013-04-17 10:38:43472// Called when data is received or on EOF (if |buffer| is NULL).
danakjaee3e1ec2016-04-16 00:23:18473void SpdyProxyClientSocket::OnDataReceived(std::unique_ptr<SpdyBuffer> buffer) {
[email protected]ca690b02013-04-17 10:38:43474 if (buffer) {
mikecirone8b85c432016-09-08 19:11:00475 net_log_.AddByteTransferEvent(NetLogEventType::SOCKET_BYTES_RECEIVED,
[email protected]ca690b02013-04-17 10:38:43476 buffer->GetRemainingSize(),
477 buffer->GetRemainingData());
dchengc7eeda422015-12-26 03:56:48478 read_buffer_queue_.Enqueue(std::move(buffer));
[email protected]ca690b02013-04-17 10:38:43479 } else {
mikecirone8b85c432016-09-08 19:11:00480 net_log_.AddByteTransferEvent(NetLogEventType::SOCKET_BYTES_RECEIVED, 0,
Raul Tambre94493c652019-03-11 17:18:35481 nullptr);
[email protected]fe2f62a2010-10-01 03:34:07482 }
483
Helen Lid7141882018-08-22 21:41:17484 if (read_callback_) {
485 if (user_buffer_) {
486 int rv = PopulateUserReadBuffer(user_buffer_->data(), user_buffer_len_);
487 user_buffer_ = nullptr;
488 user_buffer_len_ = 0;
489 std::move(read_callback_).Run(rv);
490 } else {
491 // If ReadIfReady() is used instead of Read(), tell the caller that data
492 // is available for reading.
493 std::move(read_callback_).Run(OK);
494 }
[email protected]fe2f62a2010-10-01 03:34:07495 }
496}
497
[email protected]aa19cfc2013-05-23 16:41:38498void SpdyProxyClientSocket::OnDataSent() {
[email protected]83039bb2011-12-09 18:43:55499 DCHECK(!write_callback_.is_null());
[email protected]fe2f62a2010-10-01 03:34:07500
[email protected]aa19cfc2013-05-23 16:41:38501 int rv = write_buffer_len_;
502 write_buffer_len_ = 0;
[email protected]89d4d792014-04-06 16:27:15503
504 // Proxy write callbacks result in deep callback chains. Post to allow the
505 // stream's write callback chain to unwind (see crbug.com/355511).
skyostil4891b25b2015-06-11 11:43:45506 base::ThreadTaskRunnerHandle::Get()->PostTask(
Bence Békya25e3f72018-02-13 21:13:39507 FROM_HERE, base::BindOnce(&SpdyProxyClientSocket::RunCallback,
508 write_callback_weak_factory_.GetWeakPtr(),
Brad Lassey3a814172018-04-26 03:30:21509 std::move(write_callback_), rv));
[email protected]fe2f62a2010-10-01 03:34:07510}
511
Bence Béky4c325e52020-10-22 20:48:01512void SpdyProxyClientSocket::OnTrailers(const spdy::Http2HeaderBlock& trailers) {
xunjieli294da722015-08-11 19:15:02513 // |spdy_stream_| is of type SPDY_BIDIRECTIONAL_STREAM, so trailers are
514 // combined with response headers and this method will not be calld.
515 NOTREACHED();
516}
517
[email protected]fe2f62a2010-10-01 03:34:07518void SpdyProxyClientSocket::OnClose(int status) {
[email protected]fe2f62a2010-10-01 03:34:07519 was_ever_used_ = spdy_stream_->WasEverUsed();
[email protected]d26ff352013-05-13 08:48:28520 spdy_stream_.reset();
[email protected]d9da5fe2010-10-13 22:37:16521
522 bool connecting = next_state_ != STATE_DISCONNECTED &&
523 next_state_ < STATE_OPEN;
524 if (next_state_ == STATE_OPEN)
525 next_state_ = STATE_CLOSED;
526 else
527 next_state_ = STATE_DISCONNECTED;
528
[email protected]6af4e412011-11-29 23:39:18529 base::WeakPtr<SpdyProxyClientSocket> weak_ptr = weak_factory_.GetWeakPtr();
Bence Békya25e3f72018-02-13 21:13:39530 CompletionOnceCallback write_callback = std::move(write_callback_);
[email protected]d9da5fe2010-10-13 22:37:16531 write_buffer_len_ = 0;
[email protected]d9da5fe2010-10-13 22:37:16532
533 // If we're in the middle of connecting, we need to make sure
534 // we invoke the connect callback.
535 if (connecting) {
[email protected]83039bb2011-12-09 18:43:55536 DCHECK(!read_callback_.is_null());
Brad Lassey3a814172018-04-26 03:30:21537 std::move(read_callback_).Run(status);
[email protected]83039bb2011-12-09 18:43:55538 } else if (!read_callback_.is_null()) {
[email protected]6af4e412011-11-29 23:39:18539 // If we have a read_callback_, the we need to make sure we call it back.
danakjaee3e1ec2016-04-16 00:23:18540 OnDataReceived(std::unique_ptr<SpdyBuffer>());
[email protected]d9da5fe2010-10-13 22:37:16541 }
[email protected]6af4e412011-11-29 23:39:18542 // This may have been deleted by read_callback_, so check first.
[email protected]11fbca0b2013-06-02 23:37:21543 if (weak_ptr.get() && !write_callback.is_null())
Bence Békya25e3f72018-02-13 21:13:39544 std::move(write_callback).Run(ERR_CONNECTION_CLOSED);
[email protected]fe2f62a2010-10-01 03:34:07545}
546
Bence Béky21755dd62019-11-19 13:47:35547bool SpdyProxyClientSocket::CanGreaseFrameType() const {
548 return false;
549}
550
bnc3d95ca92017-03-29 13:20:34551NetLogSource SpdyProxyClientSocket::source_dependency() const {
552 return source_dependency_;
553}
554
[email protected]fe2f62a2010-10-01 03:34:07555} // namespace net