blob: f52262cd7911e7664983bbbe4a5621c650c10d54 [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
5#include "net/spdy/spdy_proxy_client_socket.h"
6
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"
11#include "base/bind_helpers.h"
[email protected]aa19cfc2013-05-23 16:41:3812#include "base/callback_helpers.h"
skyostil4891b25b2015-06-11 11:43:4513#include "base/location.h"
[email protected]fe2f62a2010-10-01 03:34:0714#include "base/logging.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"
[email protected]fe3b7dc2012-02-03 19:52:0921#include "net/http/http_auth_cache.h"
22#include "net/http/http_auth_handler_factory.h"
rchc5c07de2015-04-08 07:28:1823#include "net/http/http_request_info.h"
[email protected]b104b502010-10-18 20:21:3124#include "net/http/http_response_headers.h"
[email protected]57d2dfa2013-06-24 06:04:1225#include "net/http/proxy_connect_redirect_http_stream.h"
[email protected]fe2f62a2010-10-01 03:34:0726#include "net/spdy/spdy_http_utils.h"
[email protected]f89276a72013-07-12 06:41:5427#include "url/gurl.h"
[email protected]fe2f62a2010-10-01 03:34:0728
29namespace net {
30
31SpdyProxyClientSocket::SpdyProxyClientSocket(
[email protected]d26ff352013-05-13 08:48:2832 const base::WeakPtr<SpdyStream>& spdy_stream,
[email protected]fe2f62a2010-10-01 03:34:0733 const std::string& user_agent,
34 const HostPortPair& endpoint,
[email protected]fe2f62a2010-10-01 03:34:0735 const HostPortPair& proxy_server,
[email protected]f6c63db52013-02-02 00:35:2236 const BoundNetLog& source_net_log,
mmenke2a1781d2015-10-07 19:25:3337 HttpAuthController* auth_controller)
[email protected]49639fa2011-12-20 23:22:4138 : next_state_(STATE_DISCONNECTED),
[email protected]fe2f62a2010-10-01 03:34:0739 spdy_stream_(spdy_stream),
[email protected]fe2f62a2010-10-01 03:34:0740 endpoint_(endpoint),
mmenke2a1781d2015-10-07 19:25:3341 auth_(auth_controller),
rchecd3c552015-04-07 20:53:5442 user_agent_(user_agent),
[email protected]ca690b02013-04-17 10:38:4343 user_buffer_len_(0),
[email protected]fe2f62a2010-10-01 03:34:0744 write_buffer_len_(0),
[email protected]57d2dfa2013-06-24 06:04:1245 was_ever_used_(false),
46 redirect_has_load_timing_info_(false),
[email protected]f6c63db52013-02-02 00:35:2247 net_log_(BoundNetLog::Make(spdy_stream->net_log().net_log(),
[email protected]0a30cf512014-05-27 20:55:1848 NetLog::SOURCE_PROXY_CLIENT_SOCKET)),
[email protected]0a428c22014-06-14 02:26:3749 weak_factory_(this),
50 write_callback_weak_factory_(this) {
[email protected]fe2f62a2010-10-01 03:34:0751 request_.method = "CONNECT";
rchecd3c552015-04-07 20:53:5452 request_.url = GURL("https://" + endpoint.ToString());
[email protected]f6c63db52013-02-02 00:35:2253 net_log_.BeginEvent(NetLog::TYPE_SOCKET_ALIVE,
54 source_net_log.source().ToEventParametersCallback());
55 net_log_.AddEvent(
bncbe32a032015-02-25 17:22:5556 NetLog::TYPE_HTTP2_PROXY_CLIENT_SESSION,
[email protected]f6c63db52013-02-02 00:35:2257 spdy_stream->net_log().source().ToEventParametersCallback());
58
[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();
[email protected]f6c63db52013-02-02 00:35:2265 net_log_.EndEvent(NetLog::TYPE_SOCKET_ALIVE);
[email protected]fe2f62a2010-10-01 03:34:0766}
67
[email protected]be1a48b2011-01-20 00:12:1368const HttpResponseInfo* SpdyProxyClientSocket::GetConnectResponseInfo() const {
[email protected]90499482013-06-01 00:39:5069 return response_.headers.get() ? &response_ : NULL;
[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
77int SpdyProxyClientSocket::RestartWithAuth(const CompletionCallback& callback) {
78 // 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;
[email protected]0c5fb722012-02-28 11:50:3582 return OK;
[email protected]c0fe941d2012-02-25 00:15:3283}
84
85bool SpdyProxyClientSocket::IsUsingSpdy() const {
86 return true;
87}
88
[email protected]8e3c78cb2012-03-31 03:58:4689NextProto SpdyProxyClientSocket::GetProtocolNegotiated() const {
bnc4cb660c2016-08-11 19:43:1290 return spdy_stream_->GetNegotiatedProtocol();
[email protected]c0fe941d2012-02-25 00:15:3291}
92
[email protected]511f6f52010-12-17 03:58:2993HttpStream* SpdyProxyClientSocket::CreateConnectResponseStream() {
[email protected]57d2dfa2013-06-24 06:04:1294 return new ProxyConnectRedirectHttpStream(
95 redirect_has_load_timing_info_ ? &redirect_load_timing_info_ : NULL);
[email protected]511f6f52010-12-17 03:58:2996}
97
bnc42331402016-07-25 13:36:1598// Sends a HEADERS frame to the proxy with a CONNECT request
[email protected]fe2f62a2010-10-01 03:34:0799// for the specified endpoint. Waits for the server to send back
bnc42331402016-07-25 13:36:15100// a HEADERS frame. OK will be returned if the status is 200.
[email protected]fe2f62a2010-10-01 03:34:07101// ERR_TUNNEL_CONNECTION_FAILED will be returned for any other status.
102// In any of these cases, Read() may be called to retrieve the HTTP
103// response body. Any other return values should be considered fatal.
[email protected]fe3b7dc2012-02-03 19:52:09104// TODO(rch): handle 407 proxy auth requested correctly, perhaps
105// by creating a new stream for the subsequent request.
[email protected]fe2f62a2010-10-01 03:34:07106// TODO(rch): create a more appropriate error code to disambiguate
107// the HTTPS Proxy tunnel failure from an HTTP Proxy tunnel failure.
[email protected]dbf036f2011-12-06 23:33:24108int SpdyProxyClientSocket::Connect(const CompletionCallback& callback) {
[email protected]83039bb2011-12-09 18:43:55109 DCHECK(read_callback_.is_null());
[email protected]d9da5fe2010-10-13 22:37:16110 if (next_state_ == STATE_OPEN)
[email protected]fe2f62a2010-10-01 03:34:07111 return OK;
112
[email protected]d9da5fe2010-10-13 22:37:16113 DCHECK_EQ(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07114 next_state_ = STATE_GENERATE_AUTH_TOKEN;
115
116 int rv = DoLoop(OK);
117 if (rv == ERR_IO_PENDING)
118 read_callback_ = callback;
119 return rv;
120}
121
122void SpdyProxyClientSocket::Disconnect() {
[email protected]ca690b02013-04-17 10:38:43123 read_buffer_queue_.Clear();
[email protected]d9da5fe2010-10-13 22:37:16124 user_buffer_ = NULL;
[email protected]ca690b02013-04-17 10:38:43125 user_buffer_len_ = 0;
[email protected]dbf036f2011-12-06 23:33:24126 read_callback_.Reset();
[email protected]d9da5fe2010-10-13 22:37:16127
128 write_buffer_len_ = 0;
[email protected]83039bb2011-12-09 18:43:55129 write_callback_.Reset();
[email protected]0a428c22014-06-14 02:26:37130 write_callback_weak_factory_.InvalidateWeakPtrs();
[email protected]d9da5fe2010-10-13 22:37:16131
132 next_state_ = STATE_DISCONNECTED;
133
[email protected]11fbca0b2013-06-02 23:37:21134 if (spdy_stream_.get()) {
[email protected]fe2f62a2010-10-01 03:34:07135 // This will cause OnClose to be invoked, which takes care of
136 // cleaning up all the internal state.
137 spdy_stream_->Cancel();
[email protected]11fbca0b2013-06-02 23:37:21138 DCHECK(!spdy_stream_.get());
[email protected]f6a78292013-03-09 14:36:34139 }
[email protected]fe2f62a2010-10-01 03:34:07140}
141
142bool SpdyProxyClientSocket::IsConnected() const {
[email protected]194c7a92011-12-03 04:54:18143 return next_state_ == STATE_OPEN;
[email protected]fe2f62a2010-10-01 03:34:07144}
145
146bool SpdyProxyClientSocket::IsConnectedAndIdle() const {
[email protected]ca690b02013-04-17 10:38:43147 return IsConnected() && read_buffer_queue_.IsEmpty() &&
[email protected]5ff0ed32014-02-12 17:48:51148 spdy_stream_->IsOpen();
[email protected]fe2f62a2010-10-01 03:34:07149}
150
[email protected]e4be2dd2010-12-14 00:44:39151const BoundNetLog& SpdyProxyClientSocket::NetLog() const {
152 return net_log_;
153}
154
[email protected]fe2f62a2010-10-01 03:34:07155void SpdyProxyClientSocket::SetSubresourceSpeculation() {
156 // TODO(rch): what should this implementation be?
157}
158
159void SpdyProxyClientSocket::SetOmniboxSpeculation() {
160 // TODO(rch): what should this implementation be?
161}
162
163bool 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
[email protected]2d88e7d2012-07-19 17:55:17167bool SpdyProxyClientSocket::WasNpnNegotiated() const {
168 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
[email protected]fe2f62a2010-10-01 03:34:07189int SpdyProxyClientSocket::Read(IOBuffer* buf, int buf_len,
[email protected]3f55aa12011-12-07 02:03:33190 const CompletionCallback& callback) {
[email protected]83039bb2011-12-09 18:43:55191 DCHECK(read_callback_.is_null());
[email protected]90499482013-06-01 00:39:50192 DCHECK(!user_buffer_.get());
[email protected]3f55aa12011-12-07 02:03:33193
194 if (next_state_ == STATE_DISCONNECTED)
195 return ERR_SOCKET_NOT_CONNECTED;
196
[email protected]ca690b02013-04-17 10:38:43197 if (next_state_ == STATE_CLOSED && read_buffer_queue_.IsEmpty()) {
[email protected]3f55aa12011-12-07 02:03:33198 return 0;
199 }
200
201 DCHECK(next_state_ == STATE_OPEN || next_state_ == STATE_CLOSED);
202 DCHECK(buf);
[email protected]ca690b02013-04-17 10:38:43203 size_t result = PopulateUserReadBuffer(buf->data(), buf_len);
[email protected]3f55aa12011-12-07 02:03:33204 if (result == 0) {
[email protected]ca690b02013-04-17 10:38:43205 user_buffer_ = buf;
206 user_buffer_len_ = static_cast<size_t>(buf_len);
[email protected]3f55aa12011-12-07 02:03:33207 DCHECK(!callback.is_null());
208 read_callback_ = callback;
209 return ERR_IO_PENDING;
210 }
211 user_buffer_ = NULL;
212 return result;
213}
[email protected]fe2f62a2010-10-01 03:34:07214
[email protected]ca690b02013-04-17 10:38:43215size_t SpdyProxyClientSocket::PopulateUserReadBuffer(char* data, size_t len) {
[email protected]09a8d9172013-04-17 19:23:49216 return read_buffer_queue_.Dequeue(data, len);
[email protected]fe2f62a2010-10-01 03:34:07217}
218
219int SpdyProxyClientSocket::Write(IOBuffer* buf, int buf_len,
[email protected]83039bb2011-12-09 18:43:55220 const CompletionCallback& callback) {
221 DCHECK(write_callback_.is_null());
[email protected]194c7a92011-12-03 04:54:18222 if (next_state_ != STATE_OPEN)
[email protected]d9da5fe2010-10-13 22:37:16223 return ERR_SOCKET_NOT_CONNECTED;
224
[email protected]11fbca0b2013-06-02 23:37:21225 DCHECK(spdy_stream_.get());
[email protected]edbfa8c2013-05-29 00:22:33226 spdy_stream_->SendData(buf, buf_len, MORE_DATA_TO_SEND);
[email protected]aa19cfc2013-05-23 16:41:38227 net_log_.AddByteTransferEvent(NetLog::TYPE_SOCKET_BYTES_SENT,
228 buf_len, buf->data());
229 write_callback_ = callback;
230 write_buffer_len_ = buf_len;
231 return ERR_IO_PENDING;
[email protected]fe2f62a2010-10-01 03:34:07232}
233
Avi Drissman13fc8932015-12-20 04:40:46234int SpdyProxyClientSocket::SetReceiveBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10235 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15236 // is not safe for callers to change this underlying socket.
237 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07238}
239
Avi Drissman13fc8932015-12-20 04:40:46240int SpdyProxyClientSocket::SetSendBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10241 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15242 // is not safe for callers to change this underlying socket.
243 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07244}
245
[email protected]a3528692012-06-08 00:11:42246int SpdyProxyClientSocket::GetPeerAddress(IPEndPoint* address) const {
[email protected]fe2f62a2010-10-01 03:34:07247 if (!IsConnected())
[email protected]88e03fa2010-10-05 03:09:04248 return ERR_SOCKET_NOT_CONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07249 return spdy_stream_->GetPeerAddress(address);
250}
251
[email protected]e7f74da2011-04-19 23:49:35252int SpdyProxyClientSocket::GetLocalAddress(IPEndPoint* address) const {
253 if (!IsConnected())
254 return ERR_SOCKET_NOT_CONNECTED;
255 return spdy_stream_->GetLocalAddress(address);
256}
257
[email protected]4eddbc732012-08-09 05:40:17258void SpdyProxyClientSocket::LogBlockedTunnelResponse() const {
259 ProxyClientSocket::LogBlockedTunnelResponse(
260 response_.headers->response_code(),
[email protected]4eddbc732012-08-09 05:40:17261 /* is_https_proxy = */ true);
262}
263
[email protected]0a428c22014-06-14 02:26:37264void SpdyProxyClientSocket::RunCallback(const CompletionCallback& callback,
265 int result) const {
266 callback.Run(result);
267}
268
[email protected]fe2f62a2010-10-01 03:34:07269void SpdyProxyClientSocket::OnIOComplete(int result) {
[email protected]d9da5fe2010-10-13 22:37:16270 DCHECK_NE(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07271 int rv = DoLoop(result);
272 if (rv != ERR_IO_PENDING) {
[email protected]83039bb2011-12-09 18:43:55273 CompletionCallback c = read_callback_;
274 read_callback_.Reset();
275 c.Run(rv);
[email protected]fe2f62a2010-10-01 03:34:07276 }
277}
278
279int SpdyProxyClientSocket::DoLoop(int last_io_result) {
[email protected]d9da5fe2010-10-13 22:37:16280 DCHECK_NE(next_state_, STATE_DISCONNECTED);
[email protected]fe2f62a2010-10-01 03:34:07281 int rv = last_io_result;
282 do {
283 State state = next_state_;
[email protected]d9da5fe2010-10-13 22:37:16284 next_state_ = STATE_DISCONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07285 switch (state) {
286 case STATE_GENERATE_AUTH_TOKEN:
287 DCHECK_EQ(OK, rv);
288 rv = DoGenerateAuthToken();
289 break;
290 case STATE_GENERATE_AUTH_TOKEN_COMPLETE:
291 rv = DoGenerateAuthTokenComplete(rv);
292 break;
293 case STATE_SEND_REQUEST:
294 DCHECK_EQ(OK, rv);
[email protected]f6c63db52013-02-02 00:35:22295 net_log_.BeginEvent(NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_SEND_REQUEST);
[email protected]fe2f62a2010-10-01 03:34:07296 rv = DoSendRequest();
297 break;
298 case STATE_SEND_REQUEST_COMPLETE:
[email protected]d7fd1782011-02-08 19:16:43299 net_log_.EndEventWithNetErrorCode(
300 NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_SEND_REQUEST, rv);
[email protected]fe2f62a2010-10-01 03:34:07301 rv = DoSendRequestComplete(rv);
[email protected]f6c63db52013-02-02 00:35:22302 if (rv >= 0 || rv == ERR_IO_PENDING) {
303 // Emit extra event so can use the same events as
304 // HttpProxyClientSocket.
305 net_log_.BeginEvent(
306 NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_READ_HEADERS);
307 }
[email protected]fe2f62a2010-10-01 03:34:07308 break;
309 case STATE_READ_REPLY_COMPLETE:
310 rv = DoReadReplyComplete(rv);
[email protected]d7fd1782011-02-08 19:16:43311 net_log_.EndEventWithNetErrorCode(
312 NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_READ_HEADERS, rv);
[email protected]fe2f62a2010-10-01 03:34:07313 break;
314 default:
315 NOTREACHED() << "bad state";
316 rv = ERR_UNEXPECTED;
317 break;
318 }
[email protected]d9da5fe2010-10-13 22:37:16319 } while (rv != ERR_IO_PENDING && next_state_ != STATE_DISCONNECTED &&
320 next_state_ != STATE_OPEN);
[email protected]fe2f62a2010-10-01 03:34:07321 return rv;
322}
323
324int SpdyProxyClientSocket::DoGenerateAuthToken() {
325 next_state_ = STATE_GENERATE_AUTH_TOKEN_COMPLETE;
[email protected]49639fa2011-12-20 23:22:41326 return auth_->MaybeGenerateAuthToken(
327 &request_,
[email protected]f6a78292013-03-09 14:36:34328 base::Bind(&SpdyProxyClientSocket::OnIOComplete,
329 weak_factory_.GetWeakPtr()),
[email protected]49639fa2011-12-20 23:22:41330 net_log_);
[email protected]fe2f62a2010-10-01 03:34:07331}
332
333int SpdyProxyClientSocket::DoGenerateAuthTokenComplete(int result) {
334 DCHECK_NE(ERR_IO_PENDING, result);
335 if (result == OK)
336 next_state_ = STATE_SEND_REQUEST;
337 return result;
338}
339
340int SpdyProxyClientSocket::DoSendRequest() {
341 next_state_ = STATE_SEND_REQUEST_COMPLETE;
342
343 // Add Proxy-Authentication header if necessary.
344 HttpRequestHeaders authorization_headers;
345 if (auth_->HaveAuth()) {
346 auth_->AddAuthorizationHeader(&authorization_headers);
347 }
348
349 std::string request_line;
rchecd3c552015-04-07 20:53:54350 BuildTunnelRequest(endpoint_, authorization_headers, user_agent_,
351 &request_line, &request_.extra_headers);
[email protected]3abacd62012-06-10 20:20:32352
353 net_log_.AddEvent(
354 NetLog::TYPE_HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
355 base::Bind(&HttpRequestHeaders::NetLogCallback,
rchecd3c552015-04-07 20:53:54356 base::Unretained(&request_.extra_headers), &request_line));
[email protected]fe2f62a2010-10-01 03:34:07357
bnc1d1d86462016-07-20 16:51:55358 SpdyHeaderBlock headers;
bncbca843ba2016-07-14 13:05:48359 CreateSpdyHeadersFromHttpRequest(request_, request_.extra_headers, true,
bnc1d1d86462016-07-20 16:51:55360 &headers);
[email protected]fe2f62a2010-10-01 03:34:07361
dchengc7eeda422015-12-26 03:56:48362 return spdy_stream_->SendRequestHeaders(std::move(headers),
363 MORE_DATA_TO_SEND);
[email protected]fe2f62a2010-10-01 03:34:07364}
365
366int SpdyProxyClientSocket::DoSendRequestComplete(int result) {
367 if (result < 0)
368 return result;
369
bnc42331402016-07-25 13:36:15370 // Wait for HEADERS frame from the server
[email protected]fe2f62a2010-10-01 03:34:07371 next_state_ = STATE_READ_REPLY_COMPLETE;
372 return ERR_IO_PENDING;
373}
374
375int SpdyProxyClientSocket::DoReadReplyComplete(int result) {
376 // We enter this method directly from DoSendRequestComplete, since
bnc42331402016-07-25 13:36:15377 // we are notified by a callback when the HEADERS frame arrives.
[email protected]fe2f62a2010-10-01 03:34:07378
379 if (result < 0)
380 return result;
381
[email protected]fe2f62a2010-10-01 03:34:07382 // Require the "HTTP/1.x" status line for SSL CONNECT.
bncbe0f6af2015-10-15 17:49:56383 if (response_.headers->GetHttpVersion() < HttpVersion(1, 0))
[email protected]fe2f62a2010-10-01 03:34:07384 return ERR_TUNNEL_CONNECTION_FAILED;
385
[email protected]3abacd62012-06-10 20:20:32386 net_log_.AddEvent(
387 NetLog::TYPE_HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
388 base::Bind(&HttpResponseHeaders::NetLogCallback, response_.headers));
[email protected]fe2f62a2010-10-01 03:34:07389
[email protected]4eddbc732012-08-09 05:40:17390 switch (response_.headers->response_code()) {
391 case 200: // OK
392 next_state_ = STATE_OPEN;
393 return OK;
394
395 case 302: // Found / Moved Temporarily
396 // Try to return a sanitized response so we can follow auth redirects.
397 // If we can't, fail the tunnel connection.
ttuttle7933c112015-01-06 00:55:24398 if (!SanitizeProxyRedirect(&response_)) {
[email protected]4eddbc732012-08-09 05:40:17399 LogBlockedTunnelResponse();
400 return ERR_TUNNEL_CONNECTION_FAILED;
401 }
402
ttuttle7933c112015-01-06 00:55:24403 redirect_has_load_timing_info_ =
404 spdy_stream_->GetLoadTimingInfo(&redirect_load_timing_info_);
405 // Note that this triggers a RST_STREAM_CANCEL.
406 spdy_stream_->DetachDelegate();
407 next_state_ = STATE_DISCONNECTED;
408 return ERR_HTTPS_PROXY_TUNNEL_RESPONSE;
409
[email protected]4eddbc732012-08-09 05:40:17410 case 407: // Proxy Authentication Required
411 next_state_ = STATE_OPEN;
ttuttle7933c112015-01-06 00:55:24412 if (!SanitizeProxyAuth(&response_)) {
413 LogBlockedTunnelResponse();
414 return ERR_TUNNEL_CONNECTION_FAILED;
415 }
[email protected]90499482013-06-01 00:39:50416 return HandleProxyAuthChallenge(auth_.get(), &response_, net_log_);
[email protected]4eddbc732012-08-09 05:40:17417
418 default:
419 // Ignore response to avoid letting the proxy impersonate the target
420 // server. (See http://crbug.com/137891.)
421 LogBlockedTunnelResponse();
422 return ERR_TUNNEL_CONNECTION_FAILED;
[email protected]511f6f52010-12-17 03:58:29423 }
[email protected]fe2f62a2010-10-01 03:34:07424}
425
426// SpdyStream::Delegate methods:
427// Called when SYN frame has been sent.
428// Returns true if no more data to be sent after SYN frame.
[email protected]edbfa8c2013-05-29 00:22:33429void SpdyProxyClientSocket::OnRequestHeadersSent() {
[email protected]fe2f62a2010-10-01 03:34:07430 DCHECK_EQ(next_state_, STATE_SEND_REQUEST_COMPLETE);
431
[email protected]d46715c2013-04-15 00:21:42432 OnIOComplete(OK);
[email protected]fe2f62a2010-10-01 03:34:07433}
434
[email protected]6d116e1a2013-06-24 07:42:15435SpdyResponseHeadersStatus SpdyProxyClientSocket::OnResponseHeadersUpdated(
436 const SpdyHeaderBlock& response_headers) {
[email protected]d08358502010-12-03 22:04:03437 // If we've already received the reply, existing headers are too late.
438 // TODO(mbelshe): figure out a way to make HEADERS frames useful after the
439 // initial response.
440 if (next_state_ != STATE_READ_REPLY_COMPLETE)
[email protected]6d116e1a2013-06-24 07:42:15441 return RESPONSE_HEADERS_ARE_COMPLETE;
[email protected]fe2f62a2010-10-01 03:34:07442
[email protected]d08358502010-12-03 22:04:03443 // Save the response
bncbca843ba2016-07-14 13:05:48444 if (!SpdyHeadersToHttpResponse(response_headers, &response_))
[email protected]6d116e1a2013-06-24 07:42:15445 return RESPONSE_HEADERS_ARE_INCOMPLETE;
[email protected]fe2f62a2010-10-01 03:34:07446
[email protected]6d116e1a2013-06-24 07:42:15447 OnIOComplete(OK);
448 return RESPONSE_HEADERS_ARE_COMPLETE;
[email protected]fe2f62a2010-10-01 03:34:07449}
450
[email protected]ca690b02013-04-17 10:38:43451// Called when data is received or on EOF (if |buffer| is NULL).
danakjaee3e1ec2016-04-16 00:23:18452void SpdyProxyClientSocket::OnDataReceived(std::unique_ptr<SpdyBuffer> buffer) {
[email protected]ca690b02013-04-17 10:38:43453 if (buffer) {
454 net_log_.AddByteTransferEvent(NetLog::TYPE_SOCKET_BYTES_RECEIVED,
455 buffer->GetRemainingSize(),
456 buffer->GetRemainingData());
dchengc7eeda422015-12-26 03:56:48457 read_buffer_queue_.Enqueue(std::move(buffer));
[email protected]ca690b02013-04-17 10:38:43458 } else {
459 net_log_.AddByteTransferEvent(NetLog::TYPE_SOCKET_BYTES_RECEIVED, 0, NULL);
[email protected]fe2f62a2010-10-01 03:34:07460 }
461
[email protected]83039bb2011-12-09 18:43:55462 if (!read_callback_.is_null()) {
[email protected]ca690b02013-04-17 10:38:43463 int rv = PopulateUserReadBuffer(user_buffer_->data(), user_buffer_len_);
[email protected]dbf036f2011-12-06 23:33:24464 CompletionCallback c = read_callback_;
465 read_callback_.Reset();
466 user_buffer_ = NULL;
[email protected]ca690b02013-04-17 10:38:43467 user_buffer_len_ = 0;
[email protected]dbf036f2011-12-06 23:33:24468 c.Run(rv);
[email protected]fe2f62a2010-10-01 03:34:07469 }
470}
471
[email protected]aa19cfc2013-05-23 16:41:38472void SpdyProxyClientSocket::OnDataSent() {
[email protected]83039bb2011-12-09 18:43:55473 DCHECK(!write_callback_.is_null());
[email protected]fe2f62a2010-10-01 03:34:07474
[email protected]aa19cfc2013-05-23 16:41:38475 int rv = write_buffer_len_;
476 write_buffer_len_ = 0;
[email protected]89d4d792014-04-06 16:27:15477
478 // Proxy write callbacks result in deep callback chains. Post to allow the
479 // stream's write callback chain to unwind (see crbug.com/355511).
skyostil4891b25b2015-06-11 11:43:45480 base::ThreadTaskRunnerHandle::Get()->PostTask(
481 FROM_HERE, base::Bind(&SpdyProxyClientSocket::RunCallback,
482 write_callback_weak_factory_.GetWeakPtr(),
tzik2ac7d802016-02-04 01:57:17483 base::ResetAndReturn(&write_callback_), rv));
[email protected]fe2f62a2010-10-01 03:34:07484}
485
xunjieli294da722015-08-11 19:15:02486void SpdyProxyClientSocket::OnTrailers(const SpdyHeaderBlock& trailers) {
487 // |spdy_stream_| is of type SPDY_BIDIRECTIONAL_STREAM, so trailers are
488 // combined with response headers and this method will not be calld.
489 NOTREACHED();
490}
491
[email protected]fe2f62a2010-10-01 03:34:07492void SpdyProxyClientSocket::OnClose(int status) {
[email protected]fe2f62a2010-10-01 03:34:07493 was_ever_used_ = spdy_stream_->WasEverUsed();
[email protected]d26ff352013-05-13 08:48:28494 spdy_stream_.reset();
[email protected]d9da5fe2010-10-13 22:37:16495
496 bool connecting = next_state_ != STATE_DISCONNECTED &&
497 next_state_ < STATE_OPEN;
498 if (next_state_ == STATE_OPEN)
499 next_state_ = STATE_CLOSED;
500 else
501 next_state_ = STATE_DISCONNECTED;
502
[email protected]6af4e412011-11-29 23:39:18503 base::WeakPtr<SpdyProxyClientSocket> weak_ptr = weak_factory_.GetWeakPtr();
[email protected]83039bb2011-12-09 18:43:55504 CompletionCallback write_callback = write_callback_;
505 write_callback_.Reset();
[email protected]d9da5fe2010-10-13 22:37:16506 write_buffer_len_ = 0;
[email protected]d9da5fe2010-10-13 22:37:16507
508 // If we're in the middle of connecting, we need to make sure
509 // we invoke the connect callback.
510 if (connecting) {
[email protected]83039bb2011-12-09 18:43:55511 DCHECK(!read_callback_.is_null());
512 CompletionCallback read_callback = read_callback_;
513 read_callback_.Reset();
514 read_callback.Run(status);
515 } else if (!read_callback_.is_null()) {
[email protected]6af4e412011-11-29 23:39:18516 // If we have a read_callback_, the we need to make sure we call it back.
danakjaee3e1ec2016-04-16 00:23:18517 OnDataReceived(std::unique_ptr<SpdyBuffer>());
[email protected]d9da5fe2010-10-13 22:37:16518 }
[email protected]6af4e412011-11-29 23:39:18519 // This may have been deleted by read_callback_, so check first.
[email protected]11fbca0b2013-06-02 23:37:21520 if (weak_ptr.get() && !write_callback.is_null())
[email protected]83039bb2011-12-09 18:43:55521 write_callback.Run(ERR_CONNECTION_CLOSED);
[email protected]fe2f62a2010-10-01 03:34:07522}
523
[email protected]fe2f62a2010-10-01 03:34:07524} // namespace net