blob: 0aa3803a96e103d1570f362fb0b598fec82aedea [file] [log] [blame]
Avi Drissman64595482022-09-14 20:52:291// Copyright 2012 The Chromium Authors
[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
Hans Wennborg0924470b2020-04-27 21:08:0510#include "base/check_op.h"
Avi Drissman41c4a412023-01-11 22:45:3711#include "base/functional/bind.h"
12#include "base/functional/callback_helpers.h"
skyostil4891b25b2015-06-11 11:43:4513#include "base/location.h"
Hans Wennborg0924470b2020-04-27 21:08:0514#include "base/notreached.h"
[email protected]fc9be5802013-06-11 10:56:5115#include "base/strings/string_util.h"
Patrick Monette643cdf62021-10-15 19:13:4216#include "base/task/single_thread_task_runner.h"
[email protected]f6c63db52013-02-02 00:35:2217#include "base/values.h"
[email protected]fe2f62a2010-10-01 03:34:0718#include "net/base/auth.h"
19#include "net/base/io_buffer.h"
Robert Ogden78d4f9eb2020-03-17 20:56:3820#include "net/base/proxy_delegate.h"
[email protected]fe3b7dc2012-02-03 19:52:0921#include "net/http/http_auth_cache.h"
22#include "net/http/http_auth_handler_factory.h"
Eric Roman06bd9742019-07-13 15:19:1323#include "net/http/http_log_util.h"
rchc5c07de2015-04-08 07:28:1824#include "net/http/http_request_info.h"
[email protected]b104b502010-10-18 20:21:3125#include "net/http/http_response_headers.h"
mikecirone8b85c432016-09-08 19:11:0026#include "net/log/net_log_event_type.h"
27#include "net/log/net_log_source_type.h"
Bence Béky94658bf2018-05-11 19:22:5828#include "net/spdy/spdy_http_utils.h"
[email protected]a2b2cfc2017-12-06 09:06:0829#include "net/traffic_annotation/network_traffic_annotation.h"
[email protected]f89276a72013-07-12 06:41:5430#include "url/gurl.h"
[email protected]fe2f62a2010-10-01 03:34:0731
32namespace net {
33
34SpdyProxyClientSocket::SpdyProxyClientSocket(
[email protected]d26ff352013-05-13 08:48:2835 const base::WeakPtr<SpdyStream>& spdy_stream,
Robert Ogden78d4f9eb2020-03-17 20:56:3836 const ProxyServer& proxy_server,
Bence Béky4e83f492018-05-13 23:14:2537 const std::string& user_agent,
[email protected]fe2f62a2010-10-01 03:34:0738 const HostPortPair& endpoint,
tfarina42834112016-09-22 13:38:2039 const NetLogWithSource& source_net_log,
Tsuyoshi Horo5c935982022-07-11 02:01:4240 scoped_refptr<HttpAuthController> auth_controller,
Robert Ogden78d4f9eb2020-03-17 20:56:3841 ProxyDelegate* proxy_delegate)
Kenichi Ishibashib0207d72022-02-09 22:06:3642 : spdy_stream_(spdy_stream),
[email protected]fe2f62a2010-10-01 03:34:0743 endpoint_(endpoint),
Tsuyoshi Horo5c935982022-07-11 02:01:4244 auth_(std::move(auth_controller)),
Robert Ogden78d4f9eb2020-03-17 20:56:3845 proxy_server_(proxy_server),
46 proxy_delegate_(proxy_delegate),
rchecd3c552015-04-07 20:53:5447 user_agent_(user_agent),
tfarina42834112016-09-22 13:38:2048 net_log_(NetLogWithSource::Make(spdy_stream->net_log().net_log(),
49 NetLogSourceType::PROXY_CLIENT_SOCKET)),
Jeremy Romand54000b22019-07-08 18:40:1650 source_dependency_(source_net_log.source()) {
[email protected]fe2f62a2010-10-01 03:34:0751 request_.method = "CONNECT";
rchecd3c552015-04-07 20:53:5452 request_.url = GURL("https://" + endpoint.ToString());
Eric Roman06bd9742019-07-13 15:19:1353 net_log_.BeginEventReferencingSource(NetLogEventType::SOCKET_ALIVE,
54 source_net_log.source());
55 net_log_.AddEventReferencingSource(
mikecirone8b85c432016-09-08 19:11:0056 NetLogEventType::HTTP2_PROXY_CLIENT_SESSION,
Eric Roman06bd9742019-07-13 15:19:1357 spdy_stream->net_log().source());
[email protected]f6c63db52013-02-02 00:35:2258
[email protected]fe2f62a2010-10-01 03:34:0759 spdy_stream_->SetDelegate(this);
60 was_ever_used_ = spdy_stream_->WasEverUsed();
61}
62
63SpdyProxyClientSocket::~SpdyProxyClientSocket() {
64 Disconnect();
mikecirone8b85c432016-09-08 19:11:0065 net_log_.EndEvent(NetLogEventType::SOCKET_ALIVE);
[email protected]fe2f62a2010-10-01 03:34:0766}
67
[email protected]be1a48b2011-01-20 00:12:1368const HttpResponseInfo* SpdyProxyClientSocket::GetConnectResponseInfo() const {
Raul Tambre94493c652019-03-11 17:18:3569 return response_.headers.get() ? &response_ : nullptr;
[email protected]be1a48b2011-01-20 00:12:1370}
71
[email protected]c0fe941d2012-02-25 00:15:3272const scoped_refptr<HttpAuthController>&
73SpdyProxyClientSocket::GetAuthController() const {
74 return auth_;
75}
76
Bence Béky99832232018-02-11 04:21:0077int SpdyProxyClientSocket::RestartWithAuth(CompletionOnceCallback callback) {
[email protected]c0fe941d2012-02-25 00:15:3278 // A SPDY Stream can only handle a single request, so the underlying
79 // stream may not be reused and a new SpdyProxyClientSocket must be
80 // created (possibly on top of the same SPDY Session).
81 next_state_ = STATE_DISCONNECTED;
Matt Menkeae2cbb82019-02-21 20:20:1382 return ERR_UNABLE_TO_REUSE_CONNECTION_FOR_PROXY_AUTH;
[email protected]c0fe941d2012-02-25 00:15:3283}
84
Matt Menkeedaf3b82019-03-14 21:39:4485// Ignore priority changes, just use priority of initial request. Since multiple
86// requests are pooled on the SpdyProxyClientSocket, reprioritization doesn't
87// really work.
88//
89// TODO(mmenke): Use a single priority value for all SpdyProxyClientSockets,
90// regardless of what priority they're created with.
91void SpdyProxyClientSocket::SetStreamPriority(RequestPriority priority) {}
Lily Chenf11e1292018-11-29 16:42:0992
bnc42331402016-07-25 13:36:1593// Sends a HEADERS frame to the proxy with a CONNECT request
[email protected]fe2f62a2010-10-01 03:34:0794// for the specified endpoint. Waits for the server to send back
bnc42331402016-07-25 13:36:1595// a HEADERS frame. OK will be returned if the status is 200.
[email protected]fe2f62a2010-10-01 03:34:0796// ERR_TUNNEL_CONNECTION_FAILED will be returned for any other status.
97// In any of these cases, Read() may be called to retrieve the HTTP
98// response body. Any other return values should be considered fatal.
[email protected]fe3b7dc2012-02-03 19:52:0999// TODO(rch): handle 407 proxy auth requested correctly, perhaps
100// by creating a new stream for the subsequent request.
[email protected]fe2f62a2010-10-01 03:34:07101// TODO(rch): create a more appropriate error code to disambiguate
102// the HTTPS Proxy tunnel failure from an HTTP Proxy tunnel failure.
Brad Lassey3a814172018-04-26 03:30:21103int SpdyProxyClientSocket::Connect(CompletionOnceCallback callback) {
[email protected]83039bb2011-12-09 18:43:55104 DCHECK(read_callback_.is_null());
[email protected]d9da5fe2010-10-13 22:37:16105 if (next_state_ == STATE_OPEN)
[email protected]fe2f62a2010-10-01 03:34:07106 return OK;
107
[email protected]d9da5fe2010-10-13 22:37:16108 DCHECK_EQ(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07109 next_state_ = STATE_GENERATE_AUTH_TOKEN;
110
111 int rv = DoLoop(OK);
112 if (rv == ERR_IO_PENDING)
Bence Békya25e3f72018-02-13 21:13:39113 read_callback_ = std::move(callback);
[email protected]fe2f62a2010-10-01 03:34:07114 return rv;
115}
116
117void SpdyProxyClientSocket::Disconnect() {
[email protected]ca690b02013-04-17 10:38:43118 read_buffer_queue_.Clear();
Raul Tambre94493c652019-03-11 17:18:35119 user_buffer_ = nullptr;
[email protected]ca690b02013-04-17 10:38:43120 user_buffer_len_ = 0;
[email protected]dbf036f2011-12-06 23:33:24121 read_callback_.Reset();
[email protected]d9da5fe2010-10-13 22:37:16122
123 write_buffer_len_ = 0;
[email protected]83039bb2011-12-09 18:43:55124 write_callback_.Reset();
[email protected]0a428c22014-06-14 02:26:37125 write_callback_weak_factory_.InvalidateWeakPtrs();
[email protected]d9da5fe2010-10-13 22:37:16126
127 next_state_ = STATE_DISCONNECTED;
128
[email protected]11fbca0b2013-06-02 23:37:21129 if (spdy_stream_.get()) {
[email protected]fe2f62a2010-10-01 03:34:07130 // This will cause OnClose to be invoked, which takes care of
131 // cleaning up all the internal state.
Bence Béky6b9c1352018-05-10 11:51:25132 spdy_stream_->Cancel(ERR_ABORTED);
[email protected]11fbca0b2013-06-02 23:37:21133 DCHECK(!spdy_stream_.get());
[email protected]f6a78292013-03-09 14:36:34134 }
[email protected]fe2f62a2010-10-01 03:34:07135}
136
137bool SpdyProxyClientSocket::IsConnected() const {
[email protected]194c7a92011-12-03 04:54:18138 return next_state_ == STATE_OPEN;
[email protected]fe2f62a2010-10-01 03:34:07139}
140
141bool SpdyProxyClientSocket::IsConnectedAndIdle() const {
[email protected]ca690b02013-04-17 10:38:43142 return IsConnected() && read_buffer_queue_.IsEmpty() &&
[email protected]5ff0ed32014-02-12 17:48:51143 spdy_stream_->IsOpen();
[email protected]fe2f62a2010-10-01 03:34:07144}
145
tfarina42834112016-09-22 13:38:20146const NetLogWithSource& SpdyProxyClientSocket::NetLog() const {
[email protected]e4be2dd2010-12-14 00:44:39147 return net_log_;
148}
149
[email protected]fe2f62a2010-10-01 03:34:07150bool SpdyProxyClientSocket::WasEverUsed() const {
[email protected]11fbca0b2013-06-02 23:37:21151 return was_ever_used_ || (spdy_stream_.get() && spdy_stream_->WasEverUsed());
[email protected]fe2f62a2010-10-01 03:34:07152}
153
tfarina2846404c2016-12-25 14:31:37154bool SpdyProxyClientSocket::WasAlpnNegotiated() const {
David Benjamin854a9922022-01-22 01:25:10155 // Do not delegate to `spdy_stream_`. While `spdy_stream_` negotiated ALPN
156 // with the proxy, this object represents the tunneled TCP connection to the
157 // origin.
[email protected]2d88e7d2012-07-19 17:55:17158 return false;
159}
160
[email protected]33661e482012-04-03 16:16:26161NextProto SpdyProxyClientSocket::GetNegotiatedProtocol() const {
David Benjamin854a9922022-01-22 01:25:10162 // Do not delegate to `spdy_stream_`. While `spdy_stream_` negotiated ALPN
163 // with the proxy, this object represents the tunneled TCP connection to the
164 // origin.
[email protected]33661e482012-04-03 16:16:26165 return kProtoUnknown;
166}
167
[email protected]2d88e7d2012-07-19 17:55:17168bool SpdyProxyClientSocket::GetSSLInfo(SSLInfo* ssl_info) {
David Benjamin854a9922022-01-22 01:25:10169 // Do not delegate to `spdy_stream_`. While `spdy_stream_` connected to the
170 // proxy with TLS, this object represents the tunneled TCP connection to the
171 // origin.
172 return false;
[email protected]2d88e7d2012-07-19 17:55:17173}
174
tbansalf82cc8e2015-10-14 20:05:49175int64_t SpdyProxyClientSocket::GetTotalReceivedBytes() const {
176 NOTIMPLEMENTED();
177 return 0;
178}
179
Paul Jensen0f49dec2017-12-12 23:39:58180void SpdyProxyClientSocket::ApplySocketTag(const SocketTag& tag) {
Matt Menke5b7974fa2019-03-09 01:18:12181 // In the case of a connection to the proxy using HTTP/2 or HTTP/3 where the
182 // underlying socket may multiplex multiple streams, applying this request's
183 // socket tag to the multiplexed session would incorrectly apply the socket
184 // tag to all mutliplexed streams. Fortunately socket tagging is only
185 // supported on Android without the data reduction proxy, so only simple HTTP
186 // proxies are supported, so proxies won't be using HTTP/2 or HTTP/3. Enforce
187 // that a specific (non-default) tag isn't being applied.
188 CHECK(tag == SocketTag());
Paul Jensen0f49dec2017-12-12 23:39:58189}
190
Bence Békya25e3f72018-02-13 21:13:39191int SpdyProxyClientSocket::Read(IOBuffer* buf,
192 int buf_len,
Brad Lassey3a814172018-04-26 03:30:21193 CompletionOnceCallback callback) {
Helen Lid7141882018-08-22 21:41:17194 int rv = ReadIfReady(buf, buf_len, std::move(callback));
195 if (rv == ERR_IO_PENDING) {
196 user_buffer_ = buf;
197 user_buffer_len_ = static_cast<size_t>(buf_len);
198 }
199 return rv;
200}
201
202int SpdyProxyClientSocket::ReadIfReady(IOBuffer* buf,
203 int buf_len,
204 CompletionOnceCallback callback) {
205 DCHECK(!read_callback_);
206 DCHECK(!user_buffer_);
[email protected]3f55aa12011-12-07 02:03:33207
208 if (next_state_ == STATE_DISCONNECTED)
209 return ERR_SOCKET_NOT_CONNECTED;
210
[email protected]ca690b02013-04-17 10:38:43211 if (next_state_ == STATE_CLOSED && read_buffer_queue_.IsEmpty()) {
[email protected]3f55aa12011-12-07 02:03:33212 return 0;
213 }
214
215 DCHECK(next_state_ == STATE_OPEN || next_state_ == STATE_CLOSED);
216 DCHECK(buf);
[email protected]ca690b02013-04-17 10:38:43217 size_t result = PopulateUserReadBuffer(buf->data(), buf_len);
[email protected]3f55aa12011-12-07 02:03:33218 if (result == 0) {
Brad Lassey3a814172018-04-26 03:30:21219 read_callback_ = std::move(callback);
[email protected]3f55aa12011-12-07 02:03:33220 return ERR_IO_PENDING;
221 }
[email protected]3f55aa12011-12-07 02:03:33222 return result;
223}
[email protected]fe2f62a2010-10-01 03:34:07224
Helen Lid7141882018-08-22 21:41:17225int SpdyProxyClientSocket::CancelReadIfReady() {
226 // Only a pending ReadIfReady() can be canceled.
227 DCHECK(!user_buffer_) << "Pending Read() cannot be canceled";
228 read_callback_.Reset();
229 return OK;
230}
231
[email protected]ca690b02013-04-17 10:38:43232size_t SpdyProxyClientSocket::PopulateUserReadBuffer(char* data, size_t len) {
[email protected]09a8d9172013-04-17 19:23:49233 return read_buffer_queue_.Dequeue(data, len);
[email protected]fe2f62a2010-10-01 03:34:07234}
235
[email protected]a2b2cfc2017-12-06 09:06:08236int SpdyProxyClientSocket::Write(
237 IOBuffer* buf,
238 int buf_len,
Brad Lassey3a814172018-04-26 03:30:21239 CompletionOnceCallback callback,
[email protected]a2b2cfc2017-12-06 09:06:08240 const NetworkTrafficAnnotationTag& traffic_annotation) {
[email protected]83039bb2011-12-09 18:43:55241 DCHECK(write_callback_.is_null());
[email protected]194c7a92011-12-03 04:54:18242 if (next_state_ != STATE_OPEN)
[email protected]d9da5fe2010-10-13 22:37:16243 return ERR_SOCKET_NOT_CONNECTED;
Kenichi Ishibashi43fb0f72022-05-06 00:08:32244 if (end_stream_state_ == EndStreamState::kEndStreamSent)
245 return ERR_CONNECTION_CLOSED;
[email protected]d9da5fe2010-10-13 22:37:16246
[email protected]11fbca0b2013-06-02 23:37:21247 DCHECK(spdy_stream_.get());
[email protected]edbfa8c2013-05-29 00:22:33248 spdy_stream_->SendData(buf, buf_len, MORE_DATA_TO_SEND);
mikecirone8b85c432016-09-08 19:11:00249 net_log_.AddByteTransferEvent(NetLogEventType::SOCKET_BYTES_SENT, buf_len,
250 buf->data());
Brad Lassey3a814172018-04-26 03:30:21251 write_callback_ = std::move(callback);
[email protected]aa19cfc2013-05-23 16:41:38252 write_buffer_len_ = buf_len;
253 return ERR_IO_PENDING;
[email protected]fe2f62a2010-10-01 03:34:07254}
255
Avi Drissman13fc8932015-12-20 04:40:46256int SpdyProxyClientSocket::SetReceiveBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10257 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15258 // is not safe for callers to change this underlying socket.
259 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07260}
261
Avi Drissman13fc8932015-12-20 04:40:46262int SpdyProxyClientSocket::SetSendBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10263 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15264 // is not safe for callers to change this underlying socket.
265 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07266}
267
[email protected]a3528692012-06-08 00:11:42268int SpdyProxyClientSocket::GetPeerAddress(IPEndPoint* address) const {
[email protected]fe2f62a2010-10-01 03:34:07269 if (!IsConnected())
[email protected]88e03fa2010-10-05 03:09:04270 return ERR_SOCKET_NOT_CONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07271 return spdy_stream_->GetPeerAddress(address);
272}
273
[email protected]e7f74da2011-04-19 23:49:35274int SpdyProxyClientSocket::GetLocalAddress(IPEndPoint* address) const {
275 if (!IsConnected())
276 return ERR_SOCKET_NOT_CONNECTED;
277 return spdy_stream_->GetLocalAddress(address);
278}
279
Kenichi Ishibashib0207d72022-02-09 22:06:36280void SpdyProxyClientSocket::RunWriteCallback(CompletionOnceCallback callback,
281 int result) const {
Bence Békya25e3f72018-02-13 21:13:39282 std::move(callback).Run(result);
Kenichi Ishibashib0207d72022-02-09 22:06:36283
284 if (end_stream_state_ == EndStreamState::kEndStreamReceived) {
Sean Maher5b9af51f2022-11-21 15:32:47285 base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
Kenichi Ishibashib0207d72022-02-09 22:06:36286 FROM_HERE, base::BindOnce(&SpdyProxyClientSocket::MaybeSendEndStream,
Alan Cutter13a534d2022-10-06 20:46:34287 weak_factory_.GetMutableWeakPtr()));
Kenichi Ishibashib0207d72022-02-09 22:06:36288 }
[email protected]0a428c22014-06-14 02:26:37289}
290
[email protected]fe2f62a2010-10-01 03:34:07291void SpdyProxyClientSocket::OnIOComplete(int result) {
[email protected]d9da5fe2010-10-13 22:37:16292 DCHECK_NE(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07293 int rv = DoLoop(result);
294 if (rv != ERR_IO_PENDING) {
Brad Lassey3a814172018-04-26 03:30:21295 std::move(read_callback_).Run(rv);
[email protected]fe2f62a2010-10-01 03:34:07296 }
297}
298
299int SpdyProxyClientSocket::DoLoop(int last_io_result) {
[email protected]d9da5fe2010-10-13 22:37:16300 DCHECK_NE(next_state_, STATE_DISCONNECTED);
[email protected]fe2f62a2010-10-01 03:34:07301 int rv = last_io_result;
302 do {
303 State state = next_state_;
[email protected]d9da5fe2010-10-13 22:37:16304 next_state_ = STATE_DISCONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07305 switch (state) {
306 case STATE_GENERATE_AUTH_TOKEN:
307 DCHECK_EQ(OK, rv);
308 rv = DoGenerateAuthToken();
309 break;
310 case STATE_GENERATE_AUTH_TOKEN_COMPLETE:
311 rv = DoGenerateAuthTokenComplete(rv);
312 break;
313 case STATE_SEND_REQUEST:
314 DCHECK_EQ(OK, rv);
mikecirone8b85c432016-09-08 19:11:00315 net_log_.BeginEvent(
316 NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST);
[email protected]fe2f62a2010-10-01 03:34:07317 rv = DoSendRequest();
318 break;
319 case STATE_SEND_REQUEST_COMPLETE:
[email protected]d7fd1782011-02-08 19:16:43320 net_log_.EndEventWithNetErrorCode(
mikecirone8b85c432016-09-08 19:11:00321 NetLogEventType::HTTP_TRANSACTION_TUNNEL_SEND_REQUEST, rv);
[email protected]fe2f62a2010-10-01 03:34:07322 rv = DoSendRequestComplete(rv);
[email protected]f6c63db52013-02-02 00:35:22323 if (rv >= 0 || rv == ERR_IO_PENDING) {
324 // Emit extra event so can use the same events as
325 // HttpProxyClientSocket.
326 net_log_.BeginEvent(
mikecirone8b85c432016-09-08 19:11:00327 NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS);
[email protected]f6c63db52013-02-02 00:35:22328 }
[email protected]fe2f62a2010-10-01 03:34:07329 break;
330 case STATE_READ_REPLY_COMPLETE:
331 rv = DoReadReplyComplete(rv);
[email protected]d7fd1782011-02-08 19:16:43332 net_log_.EndEventWithNetErrorCode(
mikecirone8b85c432016-09-08 19:11:00333 NetLogEventType::HTTP_TRANSACTION_TUNNEL_READ_HEADERS, rv);
[email protected]fe2f62a2010-10-01 03:34:07334 break;
335 default:
336 NOTREACHED() << "bad state";
337 rv = ERR_UNEXPECTED;
338 break;
339 }
[email protected]d9da5fe2010-10-13 22:37:16340 } while (rv != ERR_IO_PENDING && next_state_ != STATE_DISCONNECTED &&
341 next_state_ != STATE_OPEN);
[email protected]fe2f62a2010-10-01 03:34:07342 return rv;
343}
344
345int SpdyProxyClientSocket::DoGenerateAuthToken() {
346 next_state_ = STATE_GENERATE_AUTH_TOKEN_COMPLETE;
[email protected]49639fa2011-12-20 23:22:41347 return auth_->MaybeGenerateAuthToken(
348 &request_,
Yannic Bonenberger00e09842019-08-31 20:46:19349 base::BindOnce(&SpdyProxyClientSocket::OnIOComplete,
350 weak_factory_.GetWeakPtr()),
[email protected]49639fa2011-12-20 23:22:41351 net_log_);
[email protected]fe2f62a2010-10-01 03:34:07352}
353
354int SpdyProxyClientSocket::DoGenerateAuthTokenComplete(int result) {
355 DCHECK_NE(ERR_IO_PENDING, result);
356 if (result == OK)
357 next_state_ = STATE_SEND_REQUEST;
358 return result;
359}
360
361int SpdyProxyClientSocket::DoSendRequest() {
362 next_state_ = STATE_SEND_REQUEST_COMPLETE;
363
364 // Add Proxy-Authentication header if necessary.
365 HttpRequestHeaders authorization_headers;
366 if (auth_->HaveAuth()) {
367 auth_->AddAuthorizationHeader(&authorization_headers);
368 }
369
Robert Ogden78d4f9eb2020-03-17 20:56:38370 if (proxy_delegate_) {
371 HttpRequestHeaders proxy_delegate_headers;
372 proxy_delegate_->OnBeforeTunnelRequest(proxy_server_,
373 &proxy_delegate_headers);
374 request_.extra_headers.MergeFrom(proxy_delegate_headers);
375 }
376
Bence Béky4e83f492018-05-13 23:14:25377 std::string request_line;
rchecd3c552015-04-07 20:53:54378 BuildTunnelRequest(endpoint_, authorization_headers, user_agent_,
379 &request_line, &request_.extra_headers);
[email protected]3abacd62012-06-10 20:20:32380
Eric Roman06bd9742019-07-13 15:19:13381 NetLogRequestHeaders(net_log_,
382 NetLogEventType::HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
383 request_line, &request_.extra_headers);
[email protected]fe2f62a2010-10-01 03:34:07384
Bence Béky4c325e52020-10-22 20:48:01385 spdy::Http2HeaderBlock headers;
Bence Béky1af94d6f2018-02-08 00:40:14386 CreateSpdyHeadersFromHttpRequest(request_, request_.extra_headers, &headers);
[email protected]fe2f62a2010-10-01 03:34:07387
dchengc7eeda422015-12-26 03:56:48388 return spdy_stream_->SendRequestHeaders(std::move(headers),
389 MORE_DATA_TO_SEND);
[email protected]fe2f62a2010-10-01 03:34:07390}
391
392int SpdyProxyClientSocket::DoSendRequestComplete(int result) {
393 if (result < 0)
394 return result;
395
bnc42331402016-07-25 13:36:15396 // Wait for HEADERS frame from the server
[email protected]fe2f62a2010-10-01 03:34:07397 next_state_ = STATE_READ_REPLY_COMPLETE;
398 return ERR_IO_PENDING;
399}
400
401int SpdyProxyClientSocket::DoReadReplyComplete(int result) {
402 // We enter this method directly from DoSendRequestComplete, since
bnc42331402016-07-25 13:36:15403 // we are notified by a callback when the HEADERS frame arrives.
[email protected]fe2f62a2010-10-01 03:34:07404
405 if (result < 0)
406 return result;
407
[email protected]fe2f62a2010-10-01 03:34:07408 // Require the "HTTP/1.x" status line for SSL CONNECT.
bncbe0f6af2015-10-15 17:49:56409 if (response_.headers->GetHttpVersion() < HttpVersion(1, 0))
[email protected]fe2f62a2010-10-01 03:34:07410 return ERR_TUNNEL_CONNECTION_FAILED;
411
Eric Roman06bd9742019-07-13 15:19:13412 NetLogResponseHeaders(
413 net_log_, NetLogEventType::HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
414 response_.headers.get());
[email protected]fe2f62a2010-10-01 03:34:07415
Robert Ogden78d4f9eb2020-03-17 20:56:38416 if (proxy_delegate_) {
417 int rv = proxy_delegate_->OnTunnelHeadersReceived(proxy_server_,
418 *response_.headers);
419 if (rv != OK) {
420 DCHECK_NE(ERR_IO_PENDING, rv);
421 return rv;
422 }
423 }
424
[email protected]4eddbc732012-08-09 05:40:17425 switch (response_.headers->response_code()) {
426 case 200: // OK
427 next_state_ = STATE_OPEN;
428 return OK;
429
[email protected]4eddbc732012-08-09 05:40:17430 case 407: // Proxy Authentication Required
431 next_state_ = STATE_OPEN;
Jeremy Roman4803d2332021-04-26 22:37:22432 SanitizeProxyAuth(response_);
[email protected]90499482013-06-01 00:39:50433 return HandleProxyAuthChallenge(auth_.get(), &response_, net_log_);
[email protected]4eddbc732012-08-09 05:40:17434
435 default:
436 // Ignore response to avoid letting the proxy impersonate the target
437 // server. (See http://crbug.com/137891.)
[email protected]4eddbc732012-08-09 05:40:17438 return ERR_TUNNEL_CONNECTION_FAILED;
[email protected]511f6f52010-12-17 03:58:29439 }
[email protected]fe2f62a2010-10-01 03:34:07440}
441
442// SpdyStream::Delegate methods:
443// Called when SYN frame has been sent.
444// Returns true if no more data to be sent after SYN frame.
bnc4c214312016-11-28 16:49:15445void SpdyProxyClientSocket::OnHeadersSent() {
[email protected]fe2f62a2010-10-01 03:34:07446 DCHECK_EQ(next_state_, STATE_SEND_REQUEST_COMPLETE);
447
[email protected]d46715c2013-04-15 00:21:42448 OnIOComplete(OK);
[email protected]fe2f62a2010-10-01 03:34:07449}
450
Kenichi Ishibashi74155532021-03-13 01:38:06451void SpdyProxyClientSocket::OnEarlyHintsReceived(
452 const spdy::Http2HeaderBlock& headers) {}
453
bnc4c214312016-11-28 16:49:15454void SpdyProxyClientSocket::OnHeadersReceived(
Bence Béky4c325e52020-10-22 20:48:01455 const spdy::Http2HeaderBlock& response_headers,
456 const spdy::Http2HeaderBlock* pushed_request_headers) {
[email protected]d08358502010-12-03 22:04:03457 // If we've already received the reply, existing headers are too late.
458 // TODO(mbelshe): figure out a way to make HEADERS frames useful after the
459 // initial response.
460 if (next_state_ != STATE_READ_REPLY_COMPLETE)
bnc4c214312016-11-28 16:49:15461 return;
[email protected]fe2f62a2010-10-01 03:34:07462
[email protected]d08358502010-12-03 22:04:03463 // Save the response
Kenichi Ishibashie3708df2022-02-26 00:55:40464 const int rv = SpdyHeadersToHttpResponse(response_headers, &response_);
465 DCHECK_NE(rv, ERR_INCOMPLETE_HTTP2_HEADERS);
[email protected]fe2f62a2010-10-01 03:34:07466
[email protected]6d116e1a2013-06-24 07:42:15467 OnIOComplete(OK);
[email protected]fe2f62a2010-10-01 03:34:07468}
469
Kenichi Ishibashib0207d72022-02-09 22:06:36470// Called when data is received or on EOF (if `buffer is nullptr).
danakjaee3e1ec2016-04-16 00:23:18471void SpdyProxyClientSocket::OnDataReceived(std::unique_ptr<SpdyBuffer> buffer) {
[email protected]ca690b02013-04-17 10:38:43472 if (buffer) {
mikecirone8b85c432016-09-08 19:11:00473 net_log_.AddByteTransferEvent(NetLogEventType::SOCKET_BYTES_RECEIVED,
[email protected]ca690b02013-04-17 10:38:43474 buffer->GetRemainingSize(),
475 buffer->GetRemainingData());
dchengc7eeda422015-12-26 03:56:48476 read_buffer_queue_.Enqueue(std::move(buffer));
[email protected]ca690b02013-04-17 10:38:43477 } else {
mikecirone8b85c432016-09-08 19:11:00478 net_log_.AddByteTransferEvent(NetLogEventType::SOCKET_BYTES_RECEIVED, 0,
Raul Tambre94493c652019-03-11 17:18:35479 nullptr);
Kenichi Ishibashib0207d72022-02-09 22:06:36480
481 if (end_stream_state_ == EndStreamState::kNone) {
482 // The peer sent END_STREAM. Schedule a DATA frame with END_STREAM.
483 end_stream_state_ = EndStreamState::kEndStreamReceived;
Sean Maher5b9af51f2022-11-21 15:32:47484 base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
Kenichi Ishibashib0207d72022-02-09 22:06:36485 FROM_HERE, base::BindOnce(&SpdyProxyClientSocket::MaybeSendEndStream,
486 weak_factory_.GetWeakPtr()));
487 }
[email protected]fe2f62a2010-10-01 03:34:07488 }
489
Helen Lid7141882018-08-22 21:41:17490 if (read_callback_) {
491 if (user_buffer_) {
492 int rv = PopulateUserReadBuffer(user_buffer_->data(), user_buffer_len_);
493 user_buffer_ = nullptr;
494 user_buffer_len_ = 0;
495 std::move(read_callback_).Run(rv);
496 } else {
497 // If ReadIfReady() is used instead of Read(), tell the caller that data
498 // is available for reading.
499 std::move(read_callback_).Run(OK);
500 }
[email protected]fe2f62a2010-10-01 03:34:07501 }
502}
503
Kenichi Ishibashib0207d72022-02-09 22:06:36504void SpdyProxyClientSocket::OnDataSent() {
505 if (end_stream_state_ == EndStreamState::kEndStreamSent) {
506 CHECK(write_callback_.is_null());
507 return;
508 }
509
[email protected]83039bb2011-12-09 18:43:55510 DCHECK(!write_callback_.is_null());
[email protected]fe2f62a2010-10-01 03:34:07511
[email protected]aa19cfc2013-05-23 16:41:38512 int rv = write_buffer_len_;
513 write_buffer_len_ = 0;
[email protected]89d4d792014-04-06 16:27:15514
515 // Proxy write callbacks result in deep callback chains. Post to allow the
516 // stream's write callback chain to unwind (see crbug.com/355511).
Sean Maher5b9af51f2022-11-21 15:32:47517 base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
Kenichi Ishibashib0207d72022-02-09 22:06:36518 FROM_HERE, base::BindOnce(&SpdyProxyClientSocket::RunWriteCallback,
Bence Békya25e3f72018-02-13 21:13:39519 write_callback_weak_factory_.GetWeakPtr(),
Brad Lassey3a814172018-04-26 03:30:21520 std::move(write_callback_), rv));
[email protected]fe2f62a2010-10-01 03:34:07521}
522
Bence Béky4c325e52020-10-22 20:48:01523void SpdyProxyClientSocket::OnTrailers(const spdy::Http2HeaderBlock& trailers) {
xunjieli294da722015-08-11 19:15:02524 // |spdy_stream_| is of type SPDY_BIDIRECTIONAL_STREAM, so trailers are
525 // combined with response headers and this method will not be calld.
526 NOTREACHED();
527}
528
[email protected]fe2f62a2010-10-01 03:34:07529void SpdyProxyClientSocket::OnClose(int status) {
[email protected]fe2f62a2010-10-01 03:34:07530 was_ever_used_ = spdy_stream_->WasEverUsed();
[email protected]d26ff352013-05-13 08:48:28531 spdy_stream_.reset();
[email protected]d9da5fe2010-10-13 22:37:16532
533 bool connecting = next_state_ != STATE_DISCONNECTED &&
534 next_state_ < STATE_OPEN;
535 if (next_state_ == STATE_OPEN)
536 next_state_ = STATE_CLOSED;
537 else
538 next_state_ = STATE_DISCONNECTED;
539
[email protected]6af4e412011-11-29 23:39:18540 base::WeakPtr<SpdyProxyClientSocket> weak_ptr = weak_factory_.GetWeakPtr();
Bence Békya25e3f72018-02-13 21:13:39541 CompletionOnceCallback write_callback = std::move(write_callback_);
[email protected]d9da5fe2010-10-13 22:37:16542 write_buffer_len_ = 0;
[email protected]d9da5fe2010-10-13 22:37:16543
544 // If we're in the middle of connecting, we need to make sure
545 // we invoke the connect callback.
546 if (connecting) {
[email protected]83039bb2011-12-09 18:43:55547 DCHECK(!read_callback_.is_null());
Brad Lassey3a814172018-04-26 03:30:21548 std::move(read_callback_).Run(status);
[email protected]83039bb2011-12-09 18:43:55549 } else if (!read_callback_.is_null()) {
[email protected]6af4e412011-11-29 23:39:18550 // If we have a read_callback_, the we need to make sure we call it back.
danakjaee3e1ec2016-04-16 00:23:18551 OnDataReceived(std::unique_ptr<SpdyBuffer>());
[email protected]d9da5fe2010-10-13 22:37:16552 }
[email protected]6af4e412011-11-29 23:39:18553 // This may have been deleted by read_callback_, so check first.
[email protected]11fbca0b2013-06-02 23:37:21554 if (weak_ptr.get() && !write_callback.is_null())
Bence Békya25e3f72018-02-13 21:13:39555 std::move(write_callback).Run(ERR_CONNECTION_CLOSED);
[email protected]fe2f62a2010-10-01 03:34:07556}
557
Bence Béky21755dd62019-11-19 13:47:35558bool SpdyProxyClientSocket::CanGreaseFrameType() const {
559 return false;
560}
561
bnc3d95ca92017-03-29 13:20:34562NetLogSource SpdyProxyClientSocket::source_dependency() const {
563 return source_dependency_;
564}
565
Kenichi Ishibashib0207d72022-02-09 22:06:36566void SpdyProxyClientSocket::MaybeSendEndStream() {
567 DCHECK_NE(end_stream_state_, EndStreamState::kNone);
568 if (end_stream_state_ == EndStreamState::kEndStreamSent)
569 return;
570
571 if (!spdy_stream_)
572 return;
573
574 // When there is a pending write, wait until the write completes.
575 if (write_callback_)
576 return;
577
578 auto buffer = base::MakeRefCounted<IOBuffer>(/*buffer_size=*/0);
579 spdy_stream_->SendData(buffer.get(), /*length=*/0, NO_MORE_DATA_TO_SEND);
580 end_stream_state_ = EndStreamState::kEndStreamSent;
581}
582
[email protected]fe2f62a2010-10-01 03:34:07583} // namespace net