blob: df2530b19dce583d4fbadaba2df97544a4c82955 [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"
skyostil4891b25b2015-06-11 11:43:4517#include "base/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"
21#include "net/base/net_util.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"
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"
[email protected]57d2dfa2013-06-24 06:04:1226#include "net/http/proxy_connect_redirect_http_stream.h"
[email protected]fe2f62a2010-10-01 03:34:0727#include "net/spdy/spdy_http_utils.h"
[email protected]f89276a72013-07-12 06:41:5428#include "url/gurl.h"
[email protected]fe2f62a2010-10-01 03:34:0729
30namespace net {
31
32SpdyProxyClientSocket::SpdyProxyClientSocket(
[email protected]d26ff352013-05-13 08:48:2833 const base::WeakPtr<SpdyStream>& spdy_stream,
[email protected]fe2f62a2010-10-01 03:34:0734 const std::string& user_agent,
35 const HostPortPair& endpoint,
[email protected]fe2f62a2010-10-01 03:34:0736 const HostPortPair& proxy_server,
[email protected]f6c63db52013-02-02 00:35:2237 const BoundNetLog& source_net_log,
mmenke2a1781d2015-10-07 19:25:3338 HttpAuthController* auth_controller)
[email protected]49639fa2011-12-20 23:22:4139 : next_state_(STATE_DISCONNECTED),
[email protected]fe2f62a2010-10-01 03:34:0740 spdy_stream_(spdy_stream),
[email protected]fe2f62a2010-10-01 03:34:0741 endpoint_(endpoint),
mmenke2a1781d2015-10-07 19:25:3342 auth_(auth_controller),
rchecd3c552015-04-07 20:53:5443 user_agent_(user_agent),
[email protected]ca690b02013-04-17 10:38:4344 user_buffer_len_(0),
[email protected]fe2f62a2010-10-01 03:34:0745 write_buffer_len_(0),
[email protected]57d2dfa2013-06-24 06:04:1246 was_ever_used_(false),
47 redirect_has_load_timing_info_(false),
[email protected]f6c63db52013-02-02 00:35:2248 net_log_(BoundNetLog::Make(spdy_stream->net_log().net_log(),
[email protected]0a30cf512014-05-27 20:55:1849 NetLog::SOURCE_PROXY_CLIENT_SOCKET)),
[email protected]0a428c22014-06-14 02:26:3750 weak_factory_(this),
51 write_callback_weak_factory_(this) {
[email protected]fe2f62a2010-10-01 03:34:0752 request_.method = "CONNECT";
rchecd3c552015-04-07 20:53:5453 request_.url = GURL("https://" + endpoint.ToString());
[email protected]f6c63db52013-02-02 00:35:2254 net_log_.BeginEvent(NetLog::TYPE_SOCKET_ALIVE,
55 source_net_log.source().ToEventParametersCallback());
56 net_log_.AddEvent(
bncbe32a032015-02-25 17:22:5557 NetLog::TYPE_HTTP2_PROXY_CLIENT_SESSION,
[email protected]f6c63db52013-02-02 00:35:2258 spdy_stream->net_log().source().ToEventParametersCallback());
59
[email protected]fe2f62a2010-10-01 03:34:0760 spdy_stream_->SetDelegate(this);
61 was_ever_used_ = spdy_stream_->WasEverUsed();
62}
63
64SpdyProxyClientSocket::~SpdyProxyClientSocket() {
65 Disconnect();
[email protected]f6c63db52013-02-02 00:35:2266 net_log_.EndEvent(NetLog::TYPE_SOCKET_ALIVE);
[email protected]fe2f62a2010-10-01 03:34:0767}
68
[email protected]be1a48b2011-01-20 00:12:1369const HttpResponseInfo* SpdyProxyClientSocket::GetConnectResponseInfo() const {
[email protected]90499482013-06-01 00:39:5070 return response_.headers.get() ? &response_ : NULL;
[email protected]be1a48b2011-01-20 00:12:1371}
72
[email protected]c0fe941d2012-02-25 00:15:3273const scoped_refptr<HttpAuthController>&
74SpdyProxyClientSocket::GetAuthController() const {
75 return auth_;
76}
77
78int SpdyProxyClientSocket::RestartWithAuth(const CompletionCallback& callback) {
79 // A SPDY Stream can only handle a single request, so the underlying
80 // stream may not be reused and a new SpdyProxyClientSocket must be
81 // created (possibly on top of the same SPDY Session).
82 next_state_ = STATE_DISCONNECTED;
[email protected]0c5fb722012-02-28 11:50:3583 return OK;
[email protected]c0fe941d2012-02-25 00:15:3284}
85
86bool SpdyProxyClientSocket::IsUsingSpdy() const {
87 return true;
88}
89
[email protected]8e3c78cb2012-03-31 03:58:4690NextProto SpdyProxyClientSocket::GetProtocolNegotiated() const {
[email protected]c0fe941d2012-02-25 00:15:3291 // Save the negotiated protocol
92 SSLInfo ssl_info;
93 bool was_npn_negotiated;
[email protected]8e3c78cb2012-03-31 03:58:4694 NextProto protocol_negotiated;
[email protected]c0fe941d2012-02-25 00:15:3295 spdy_stream_->GetSSLInfo(&ssl_info, &was_npn_negotiated,
96 &protocol_negotiated);
97 return protocol_negotiated;
98}
99
[email protected]511f6f52010-12-17 03:58:29100HttpStream* SpdyProxyClientSocket::CreateConnectResponseStream() {
[email protected]57d2dfa2013-06-24 06:04:12101 return new ProxyConnectRedirectHttpStream(
102 redirect_has_load_timing_info_ ? &redirect_load_timing_info_ : NULL);
[email protected]511f6f52010-12-17 03:58:29103}
104
[email protected]fe2f62a2010-10-01 03:34:07105// Sends a SYN_STREAM frame to the proxy with a CONNECT request
106// for the specified endpoint. Waits for the server to send back
107// a SYN_REPLY frame. OK will be returned if the status is 200.
108// ERR_TUNNEL_CONNECTION_FAILED will be returned for any other status.
109// In any of these cases, Read() may be called to retrieve the HTTP
110// response body. Any other return values should be considered fatal.
[email protected]fe3b7dc2012-02-03 19:52:09111// TODO(rch): handle 407 proxy auth requested correctly, perhaps
112// by creating a new stream for the subsequent request.
[email protected]fe2f62a2010-10-01 03:34:07113// TODO(rch): create a more appropriate error code to disambiguate
114// the HTTPS Proxy tunnel failure from an HTTP Proxy tunnel failure.
[email protected]dbf036f2011-12-06 23:33:24115int SpdyProxyClientSocket::Connect(const CompletionCallback& callback) {
[email protected]83039bb2011-12-09 18:43:55116 DCHECK(read_callback_.is_null());
[email protected]d9da5fe2010-10-13 22:37:16117 if (next_state_ == STATE_OPEN)
[email protected]fe2f62a2010-10-01 03:34:07118 return OK;
119
[email protected]d9da5fe2010-10-13 22:37:16120 DCHECK_EQ(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07121 next_state_ = STATE_GENERATE_AUTH_TOKEN;
122
123 int rv = DoLoop(OK);
124 if (rv == ERR_IO_PENDING)
125 read_callback_ = callback;
126 return rv;
127}
128
129void SpdyProxyClientSocket::Disconnect() {
[email protected]ca690b02013-04-17 10:38:43130 read_buffer_queue_.Clear();
[email protected]d9da5fe2010-10-13 22:37:16131 user_buffer_ = NULL;
[email protected]ca690b02013-04-17 10:38:43132 user_buffer_len_ = 0;
[email protected]dbf036f2011-12-06 23:33:24133 read_callback_.Reset();
[email protected]d9da5fe2010-10-13 22:37:16134
135 write_buffer_len_ = 0;
[email protected]83039bb2011-12-09 18:43:55136 write_callback_.Reset();
[email protected]0a428c22014-06-14 02:26:37137 write_callback_weak_factory_.InvalidateWeakPtrs();
[email protected]d9da5fe2010-10-13 22:37:16138
139 next_state_ = STATE_DISCONNECTED;
140
[email protected]11fbca0b2013-06-02 23:37:21141 if (spdy_stream_.get()) {
[email protected]fe2f62a2010-10-01 03:34:07142 // This will cause OnClose to be invoked, which takes care of
143 // cleaning up all the internal state.
144 spdy_stream_->Cancel();
[email protected]11fbca0b2013-06-02 23:37:21145 DCHECK(!spdy_stream_.get());
[email protected]f6a78292013-03-09 14:36:34146 }
[email protected]fe2f62a2010-10-01 03:34:07147}
148
149bool SpdyProxyClientSocket::IsConnected() const {
[email protected]194c7a92011-12-03 04:54:18150 return next_state_ == STATE_OPEN;
[email protected]fe2f62a2010-10-01 03:34:07151}
152
153bool SpdyProxyClientSocket::IsConnectedAndIdle() const {
[email protected]ca690b02013-04-17 10:38:43154 return IsConnected() && read_buffer_queue_.IsEmpty() &&
[email protected]5ff0ed32014-02-12 17:48:51155 spdy_stream_->IsOpen();
[email protected]fe2f62a2010-10-01 03:34:07156}
157
[email protected]e4be2dd2010-12-14 00:44:39158const BoundNetLog& SpdyProxyClientSocket::NetLog() const {
159 return net_log_;
160}
161
[email protected]fe2f62a2010-10-01 03:34:07162void SpdyProxyClientSocket::SetSubresourceSpeculation() {
163 // TODO(rch): what should this implementation be?
164}
165
166void SpdyProxyClientSocket::SetOmniboxSpeculation() {
167 // TODO(rch): what should this implementation be?
168}
169
170bool SpdyProxyClientSocket::WasEverUsed() const {
[email protected]11fbca0b2013-06-02 23:37:21171 return was_ever_used_ || (spdy_stream_.get() && spdy_stream_->WasEverUsed());
[email protected]fe2f62a2010-10-01 03:34:07172}
173
[email protected]7f7e92392010-10-26 18:29:29174bool SpdyProxyClientSocket::UsingTCPFastOpen() const {
175 return false;
176}
177
[email protected]2d88e7d2012-07-19 17:55:17178bool SpdyProxyClientSocket::WasNpnNegotiated() const {
179 return false;
180}
181
[email protected]33661e482012-04-03 16:16:26182NextProto SpdyProxyClientSocket::GetNegotiatedProtocol() const {
183 return kProtoUnknown;
184}
185
[email protected]2d88e7d2012-07-19 17:55:17186bool SpdyProxyClientSocket::GetSSLInfo(SSLInfo* ssl_info) {
187 bool was_npn_negotiated;
188 NextProto protocol_negotiated;
189 return spdy_stream_->GetSSLInfo(ssl_info, &was_npn_negotiated,
190 &protocol_negotiated);
191}
192
ttuttle23fdb7b2015-05-15 01:28:03193void SpdyProxyClientSocket::GetConnectionAttempts(
194 ConnectionAttempts* out) const {
195 out->clear();
196}
197
tbansalf82cc8e2015-10-14 20:05:49198int64_t SpdyProxyClientSocket::GetTotalReceivedBytes() const {
199 NOTIMPLEMENTED();
200 return 0;
201}
202
[email protected]fe2f62a2010-10-01 03:34:07203int SpdyProxyClientSocket::Read(IOBuffer* buf, int buf_len,
[email protected]3f55aa12011-12-07 02:03:33204 const CompletionCallback& callback) {
[email protected]83039bb2011-12-09 18:43:55205 DCHECK(read_callback_.is_null());
[email protected]90499482013-06-01 00:39:50206 DCHECK(!user_buffer_.get());
[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) {
[email protected]ca690b02013-04-17 10:38:43219 user_buffer_ = buf;
220 user_buffer_len_ = static_cast<size_t>(buf_len);
[email protected]3f55aa12011-12-07 02:03:33221 DCHECK(!callback.is_null());
222 read_callback_ = callback;
223 return ERR_IO_PENDING;
224 }
225 user_buffer_ = NULL;
226 return result;
227}
[email protected]fe2f62a2010-10-01 03:34:07228
[email protected]ca690b02013-04-17 10:38:43229size_t SpdyProxyClientSocket::PopulateUserReadBuffer(char* data, size_t len) {
[email protected]09a8d9172013-04-17 19:23:49230 return read_buffer_queue_.Dequeue(data, len);
[email protected]fe2f62a2010-10-01 03:34:07231}
232
233int SpdyProxyClientSocket::Write(IOBuffer* buf, int buf_len,
[email protected]83039bb2011-12-09 18:43:55234 const CompletionCallback& callback) {
235 DCHECK(write_callback_.is_null());
[email protected]194c7a92011-12-03 04:54:18236 if (next_state_ != STATE_OPEN)
[email protected]d9da5fe2010-10-13 22:37:16237 return ERR_SOCKET_NOT_CONNECTED;
238
[email protected]11fbca0b2013-06-02 23:37:21239 DCHECK(spdy_stream_.get());
[email protected]edbfa8c2013-05-29 00:22:33240 spdy_stream_->SendData(buf, buf_len, MORE_DATA_TO_SEND);
[email protected]aa19cfc2013-05-23 16:41:38241 net_log_.AddByteTransferEvent(NetLog::TYPE_SOCKET_BYTES_SENT,
242 buf_len, buf->data());
243 write_callback_ = callback;
244 write_buffer_len_ = buf_len;
245 return ERR_IO_PENDING;
[email protected]fe2f62a2010-10-01 03:34:07246}
247
Avi Drissman13fc8932015-12-20 04:40:46248int SpdyProxyClientSocket::SetReceiveBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10249 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15250 // is not safe for callers to change this underlying socket.
251 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07252}
253
Avi Drissman13fc8932015-12-20 04:40:46254int SpdyProxyClientSocket::SetSendBufferSize(int32_t size) {
[email protected]3268023f2011-05-05 00:08:10255 // Since this StreamSocket sits on top of a shared SpdySession, it
[email protected]28b96d1c2014-04-09 12:21:15256 // is not safe for callers to change this underlying socket.
257 return ERR_NOT_IMPLEMENTED;
[email protected]fe2f62a2010-10-01 03:34:07258}
259
[email protected]a3528692012-06-08 00:11:42260int SpdyProxyClientSocket::GetPeerAddress(IPEndPoint* address) const {
[email protected]fe2f62a2010-10-01 03:34:07261 if (!IsConnected())
[email protected]88e03fa2010-10-05 03:09:04262 return ERR_SOCKET_NOT_CONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07263 return spdy_stream_->GetPeerAddress(address);
264}
265
[email protected]e7f74da2011-04-19 23:49:35266int SpdyProxyClientSocket::GetLocalAddress(IPEndPoint* address) const {
267 if (!IsConnected())
268 return ERR_SOCKET_NOT_CONNECTED;
269 return spdy_stream_->GetLocalAddress(address);
270}
271
[email protected]4eddbc732012-08-09 05:40:17272void SpdyProxyClientSocket::LogBlockedTunnelResponse() const {
273 ProxyClientSocket::LogBlockedTunnelResponse(
274 response_.headers->response_code(),
[email protected]4eddbc732012-08-09 05:40:17275 /* is_https_proxy = */ true);
276}
277
[email protected]0a428c22014-06-14 02:26:37278void SpdyProxyClientSocket::RunCallback(const CompletionCallback& callback,
279 int result) const {
280 callback.Run(result);
281}
282
[email protected]fe2f62a2010-10-01 03:34:07283void SpdyProxyClientSocket::OnIOComplete(int result) {
[email protected]d9da5fe2010-10-13 22:37:16284 DCHECK_NE(STATE_DISCONNECTED, next_state_);
[email protected]fe2f62a2010-10-01 03:34:07285 int rv = DoLoop(result);
286 if (rv != ERR_IO_PENDING) {
[email protected]83039bb2011-12-09 18:43:55287 CompletionCallback c = read_callback_;
288 read_callback_.Reset();
289 c.Run(rv);
[email protected]fe2f62a2010-10-01 03:34:07290 }
291}
292
293int SpdyProxyClientSocket::DoLoop(int last_io_result) {
[email protected]d9da5fe2010-10-13 22:37:16294 DCHECK_NE(next_state_, STATE_DISCONNECTED);
[email protected]fe2f62a2010-10-01 03:34:07295 int rv = last_io_result;
296 do {
297 State state = next_state_;
[email protected]d9da5fe2010-10-13 22:37:16298 next_state_ = STATE_DISCONNECTED;
[email protected]fe2f62a2010-10-01 03:34:07299 switch (state) {
300 case STATE_GENERATE_AUTH_TOKEN:
301 DCHECK_EQ(OK, rv);
302 rv = DoGenerateAuthToken();
303 break;
304 case STATE_GENERATE_AUTH_TOKEN_COMPLETE:
305 rv = DoGenerateAuthTokenComplete(rv);
306 break;
307 case STATE_SEND_REQUEST:
308 DCHECK_EQ(OK, rv);
[email protected]f6c63db52013-02-02 00:35:22309 net_log_.BeginEvent(NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_SEND_REQUEST);
[email protected]fe2f62a2010-10-01 03:34:07310 rv = DoSendRequest();
311 break;
312 case STATE_SEND_REQUEST_COMPLETE:
[email protected]d7fd1782011-02-08 19:16:43313 net_log_.EndEventWithNetErrorCode(
314 NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_SEND_REQUEST, rv);
[email protected]fe2f62a2010-10-01 03:34:07315 rv = DoSendRequestComplete(rv);
[email protected]f6c63db52013-02-02 00:35:22316 if (rv >= 0 || rv == ERR_IO_PENDING) {
317 // Emit extra event so can use the same events as
318 // HttpProxyClientSocket.
319 net_log_.BeginEvent(
320 NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_READ_HEADERS);
321 }
[email protected]fe2f62a2010-10-01 03:34:07322 break;
323 case STATE_READ_REPLY_COMPLETE:
324 rv = DoReadReplyComplete(rv);
[email protected]d7fd1782011-02-08 19:16:43325 net_log_.EndEventWithNetErrorCode(
326 NetLog::TYPE_HTTP_TRANSACTION_TUNNEL_READ_HEADERS, rv);
[email protected]fe2f62a2010-10-01 03:34:07327 break;
328 default:
329 NOTREACHED() << "bad state";
330 rv = ERR_UNEXPECTED;
331 break;
332 }
[email protected]d9da5fe2010-10-13 22:37:16333 } while (rv != ERR_IO_PENDING && next_state_ != STATE_DISCONNECTED &&
334 next_state_ != STATE_OPEN);
[email protected]fe2f62a2010-10-01 03:34:07335 return rv;
336}
337
338int SpdyProxyClientSocket::DoGenerateAuthToken() {
339 next_state_ = STATE_GENERATE_AUTH_TOKEN_COMPLETE;
[email protected]49639fa2011-12-20 23:22:41340 return auth_->MaybeGenerateAuthToken(
341 &request_,
[email protected]f6a78292013-03-09 14:36:34342 base::Bind(&SpdyProxyClientSocket::OnIOComplete,
343 weak_factory_.GetWeakPtr()),
[email protected]49639fa2011-12-20 23:22:41344 net_log_);
[email protected]fe2f62a2010-10-01 03:34:07345}
346
347int SpdyProxyClientSocket::DoGenerateAuthTokenComplete(int result) {
348 DCHECK_NE(ERR_IO_PENDING, result);
349 if (result == OK)
350 next_state_ = STATE_SEND_REQUEST;
351 return result;
352}
353
354int SpdyProxyClientSocket::DoSendRequest() {
355 next_state_ = STATE_SEND_REQUEST_COMPLETE;
356
357 // Add Proxy-Authentication header if necessary.
358 HttpRequestHeaders authorization_headers;
359 if (auth_->HaveAuth()) {
360 auth_->AddAuthorizationHeader(&authorization_headers);
361 }
362
363 std::string request_line;
rchecd3c552015-04-07 20:53:54364 BuildTunnelRequest(endpoint_, authorization_headers, user_agent_,
365 &request_line, &request_.extra_headers);
[email protected]3abacd62012-06-10 20:20:32366
367 net_log_.AddEvent(
368 NetLog::TYPE_HTTP_TRANSACTION_SEND_TUNNEL_HEADERS,
369 base::Bind(&HttpRequestHeaders::NetLogCallback,
rchecd3c552015-04-07 20:53:54370 base::Unretained(&request_.extra_headers), &request_line));
[email protected]fe2f62a2010-10-01 03:34:07371
[email protected]0a01d890fc2012-07-18 17:24:05372 scoped_ptr<SpdyHeaderBlock> headers(new SpdyHeaderBlock());
rchecd3c552015-04-07 20:53:54373 CreateSpdyHeadersFromHttpRequest(request_, request_.extra_headers,
[email protected]585df962014-07-01 22:21:23374 spdy_stream_->GetProtocolVersion(), true,
375 headers.get());
[email protected]fe2f62a2010-10-01 03:34:07376
dchengc7eeda422015-12-26 03:56:48377 return spdy_stream_->SendRequestHeaders(std::move(headers),
378 MORE_DATA_TO_SEND);
[email protected]fe2f62a2010-10-01 03:34:07379}
380
381int SpdyProxyClientSocket::DoSendRequestComplete(int result) {
382 if (result < 0)
383 return result;
384
385 // Wait for SYN_REPLY frame from the server
386 next_state_ = STATE_READ_REPLY_COMPLETE;
387 return ERR_IO_PENDING;
388}
389
390int SpdyProxyClientSocket::DoReadReplyComplete(int result) {
391 // We enter this method directly from DoSendRequestComplete, since
392 // we are notified by a callback when the SYN_REPLY frame arrives
393
394 if (result < 0)
395 return result;
396
[email protected]fe2f62a2010-10-01 03:34:07397 // Require the "HTTP/1.x" status line for SSL CONNECT.
bncbe0f6af2015-10-15 17:49:56398 if (response_.headers->GetHttpVersion() < HttpVersion(1, 0))
[email protected]fe2f62a2010-10-01 03:34:07399 return ERR_TUNNEL_CONNECTION_FAILED;
400
[email protected]3abacd62012-06-10 20:20:32401 net_log_.AddEvent(
402 NetLog::TYPE_HTTP_TRANSACTION_READ_TUNNEL_RESPONSE_HEADERS,
403 base::Bind(&HttpResponseHeaders::NetLogCallback, response_.headers));
[email protected]fe2f62a2010-10-01 03:34:07404
[email protected]4eddbc732012-08-09 05:40:17405 switch (response_.headers->response_code()) {
406 case 200: // OK
407 next_state_ = STATE_OPEN;
408 return OK;
409
410 case 302: // Found / Moved Temporarily
411 // Try to return a sanitized response so we can follow auth redirects.
412 // If we can't, fail the tunnel connection.
ttuttle7933c112015-01-06 00:55:24413 if (!SanitizeProxyRedirect(&response_)) {
[email protected]4eddbc732012-08-09 05:40:17414 LogBlockedTunnelResponse();
415 return ERR_TUNNEL_CONNECTION_FAILED;
416 }
417
ttuttle7933c112015-01-06 00:55:24418 redirect_has_load_timing_info_ =
419 spdy_stream_->GetLoadTimingInfo(&redirect_load_timing_info_);
420 // Note that this triggers a RST_STREAM_CANCEL.
421 spdy_stream_->DetachDelegate();
422 next_state_ = STATE_DISCONNECTED;
423 return ERR_HTTPS_PROXY_TUNNEL_RESPONSE;
424
[email protected]4eddbc732012-08-09 05:40:17425 case 407: // Proxy Authentication Required
426 next_state_ = STATE_OPEN;
ttuttle7933c112015-01-06 00:55:24427 if (!SanitizeProxyAuth(&response_)) {
428 LogBlockedTunnelResponse();
429 return ERR_TUNNEL_CONNECTION_FAILED;
430 }
[email protected]90499482013-06-01 00:39:50431 return HandleProxyAuthChallenge(auth_.get(), &response_, net_log_);
[email protected]4eddbc732012-08-09 05:40:17432
433 default:
434 // Ignore response to avoid letting the proxy impersonate the target
435 // server. (See http://crbug.com/137891.)
436 LogBlockedTunnelResponse();
437 return ERR_TUNNEL_CONNECTION_FAILED;
[email protected]511f6f52010-12-17 03:58:29438 }
[email protected]fe2f62a2010-10-01 03:34:07439}
440
441// SpdyStream::Delegate methods:
442// Called when SYN frame has been sent.
443// Returns true if no more data to be sent after SYN frame.
[email protected]edbfa8c2013-05-29 00:22:33444void SpdyProxyClientSocket::OnRequestHeadersSent() {
[email protected]fe2f62a2010-10-01 03:34:07445 DCHECK_EQ(next_state_, STATE_SEND_REQUEST_COMPLETE);
446
[email protected]d46715c2013-04-15 00:21:42447 OnIOComplete(OK);
[email protected]fe2f62a2010-10-01 03:34:07448}
449
[email protected]6d116e1a2013-06-24 07:42:15450SpdyResponseHeadersStatus SpdyProxyClientSocket::OnResponseHeadersUpdated(
451 const SpdyHeaderBlock& response_headers) {
[email protected]d08358502010-12-03 22:04:03452 // If we've already received the reply, existing headers are too late.
453 // TODO(mbelshe): figure out a way to make HEADERS frames useful after the
454 // initial response.
455 if (next_state_ != STATE_READ_REPLY_COMPLETE)
[email protected]6d116e1a2013-06-24 07:42:15456 return RESPONSE_HEADERS_ARE_COMPLETE;
[email protected]fe2f62a2010-10-01 03:34:07457
[email protected]d08358502010-12-03 22:04:03458 // Save the response
[email protected]d42dedd02012-04-03 19:42:06459 if (!SpdyHeadersToHttpResponse(
[email protected]6d116e1a2013-06-24 07:42:15460 response_headers, spdy_stream_->GetProtocolVersion(), &response_))
461 return RESPONSE_HEADERS_ARE_INCOMPLETE;
[email protected]fe2f62a2010-10-01 03:34:07462
[email protected]6d116e1a2013-06-24 07:42:15463 OnIOComplete(OK);
464 return RESPONSE_HEADERS_ARE_COMPLETE;
[email protected]fe2f62a2010-10-01 03:34:07465}
466
[email protected]ca690b02013-04-17 10:38:43467// Called when data is received or on EOF (if |buffer| is NULL).
[email protected]6d116e1a2013-06-24 07:42:15468void SpdyProxyClientSocket::OnDataReceived(scoped_ptr<SpdyBuffer> buffer) {
[email protected]ca690b02013-04-17 10:38:43469 if (buffer) {
470 net_log_.AddByteTransferEvent(NetLog::TYPE_SOCKET_BYTES_RECEIVED,
471 buffer->GetRemainingSize(),
472 buffer->GetRemainingData());
dchengc7eeda422015-12-26 03:56:48473 read_buffer_queue_.Enqueue(std::move(buffer));
[email protected]ca690b02013-04-17 10:38:43474 } else {
475 net_log_.AddByteTransferEvent(NetLog::TYPE_SOCKET_BYTES_RECEIVED, 0, NULL);
[email protected]fe2f62a2010-10-01 03:34:07476 }
477
[email protected]83039bb2011-12-09 18:43:55478 if (!read_callback_.is_null()) {
[email protected]ca690b02013-04-17 10:38:43479 int rv = PopulateUserReadBuffer(user_buffer_->data(), user_buffer_len_);
[email protected]dbf036f2011-12-06 23:33:24480 CompletionCallback c = read_callback_;
481 read_callback_.Reset();
482 user_buffer_ = NULL;
[email protected]ca690b02013-04-17 10:38:43483 user_buffer_len_ = 0;
[email protected]dbf036f2011-12-06 23:33:24484 c.Run(rv);
[email protected]fe2f62a2010-10-01 03:34:07485 }
486}
487
[email protected]aa19cfc2013-05-23 16:41:38488void SpdyProxyClientSocket::OnDataSent() {
[email protected]83039bb2011-12-09 18:43:55489 DCHECK(!write_callback_.is_null());
[email protected]fe2f62a2010-10-01 03:34:07490
[email protected]aa19cfc2013-05-23 16:41:38491 int rv = write_buffer_len_;
492 write_buffer_len_ = 0;
[email protected]89d4d792014-04-06 16:27:15493
494 // Proxy write callbacks result in deep callback chains. Post to allow the
495 // stream's write callback chain to unwind (see crbug.com/355511).
skyostil4891b25b2015-06-11 11:43:45496 base::ThreadTaskRunnerHandle::Get()->PostTask(
497 FROM_HERE, base::Bind(&SpdyProxyClientSocket::RunCallback,
498 write_callback_weak_factory_.GetWeakPtr(),
499 ResetAndReturn(&write_callback_), rv));
[email protected]fe2f62a2010-10-01 03:34:07500}
501
xunjieli294da722015-08-11 19:15:02502void SpdyProxyClientSocket::OnTrailers(const SpdyHeaderBlock& trailers) {
503 // |spdy_stream_| is of type SPDY_BIDIRECTIONAL_STREAM, so trailers are
504 // combined with response headers and this method will not be calld.
505 NOTREACHED();
506}
507
[email protected]fe2f62a2010-10-01 03:34:07508void SpdyProxyClientSocket::OnClose(int status) {
[email protected]fe2f62a2010-10-01 03:34:07509 was_ever_used_ = spdy_stream_->WasEverUsed();
[email protected]d26ff352013-05-13 08:48:28510 spdy_stream_.reset();
[email protected]d9da5fe2010-10-13 22:37:16511
512 bool connecting = next_state_ != STATE_DISCONNECTED &&
513 next_state_ < STATE_OPEN;
514 if (next_state_ == STATE_OPEN)
515 next_state_ = STATE_CLOSED;
516 else
517 next_state_ = STATE_DISCONNECTED;
518
[email protected]6af4e412011-11-29 23:39:18519 base::WeakPtr<SpdyProxyClientSocket> weak_ptr = weak_factory_.GetWeakPtr();
[email protected]83039bb2011-12-09 18:43:55520 CompletionCallback write_callback = write_callback_;
521 write_callback_.Reset();
[email protected]d9da5fe2010-10-13 22:37:16522 write_buffer_len_ = 0;
[email protected]d9da5fe2010-10-13 22:37:16523
524 // If we're in the middle of connecting, we need to make sure
525 // we invoke the connect callback.
526 if (connecting) {
[email protected]83039bb2011-12-09 18:43:55527 DCHECK(!read_callback_.is_null());
528 CompletionCallback read_callback = read_callback_;
529 read_callback_.Reset();
530 read_callback.Run(status);
531 } else if (!read_callback_.is_null()) {
[email protected]6af4e412011-11-29 23:39:18532 // If we have a read_callback_, the we need to make sure we call it back.
[email protected]ca690b02013-04-17 10:38:43533 OnDataReceived(scoped_ptr<SpdyBuffer>());
[email protected]d9da5fe2010-10-13 22:37:16534 }
[email protected]6af4e412011-11-29 23:39:18535 // This may have been deleted by read_callback_, so check first.
[email protected]11fbca0b2013-06-02 23:37:21536 if (weak_ptr.get() && !write_callback.is_null())
[email protected]83039bb2011-12-09 18:43:55537 write_callback.Run(ERR_CONNECTION_CLOSED);
[email protected]fe2f62a2010-10-01 03:34:07538}
539
[email protected]fe2f62a2010-10-01 03:34:07540} // namespace net