Using pre-existing constants instead of hard-coding
[chromium-blink-merge.git] / net / spdy / spdy_stream.h
blobe107143f9c83f0cfd9fa2f1f0b7e47aea84ad281
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #ifndef NET_SPDY_SPDY_STREAM_H_
6 #define NET_SPDY_SPDY_STREAM_H_
8 #include <deque>
9 #include <string>
10 #include <vector>
12 #include "base/basictypes.h"
13 #include "base/memory/ref_counted.h"
14 #include "base/memory/scoped_ptr.h"
15 #include "base/memory/scoped_vector.h"
16 #include "base/memory/weak_ptr.h"
17 #include "net/base/bandwidth_metrics.h"
18 #include "net/base/io_buffer.h"
19 #include "net/base/net_export.h"
20 #include "net/base/net_log.h"
21 #include "net/base/request_priority.h"
22 #include "net/socket/ssl_client_socket.h"
23 #include "net/spdy/spdy_buffer.h"
24 #include "net/spdy/spdy_framer.h"
25 #include "net/spdy/spdy_header_block.h"
26 #include "net/spdy/spdy_protocol.h"
27 #include "net/ssl/server_bound_cert_service.h"
28 #include "net/ssl/ssl_client_cert_type.h"
29 #include "url/gurl.h"
31 namespace net {
33 class AddressList;
34 class IPEndPoint;
35 struct LoadTimingInfo;
36 class SSLCertRequestInfo;
37 class SSLInfo;
38 class SpdySession;
40 enum SpdyStreamType {
41 // The most general type of stream; there are no restrictions on
42 // when data can be sent and received.
43 SPDY_BIDIRECTIONAL_STREAM,
44 // A stream where the client sends a request with possibly a body,
45 // and the server then sends a response with a body.
46 SPDY_REQUEST_RESPONSE_STREAM,
47 // A server-initiated stream where the server just sends a response
48 // with a body and the client does not send anything.
49 SPDY_PUSH_STREAM
52 // Passed to some SpdyStream functions to indicate whether there's
53 // more data to send.
54 enum SpdySendStatus {
55 MORE_DATA_TO_SEND,
56 NO_MORE_DATA_TO_SEND
59 // Returned by SpdyStream::OnResponseHeadersUpdated() to indicate
60 // whether the current response headers are complete or not.
61 enum SpdyResponseHeadersStatus {
62 RESPONSE_HEADERS_ARE_INCOMPLETE,
63 RESPONSE_HEADERS_ARE_COMPLETE
66 // The SpdyStream is used by the SpdySession to represent each stream known
67 // on the SpdySession. This class provides interfaces for SpdySession to use.
68 // Streams can be created either by the client or by the server. When they
69 // are initiated by the client, both the SpdySession and client object (such as
70 // a SpdyNetworkTransaction) will maintain a reference to the stream. When
71 // initiated by the server, only the SpdySession will maintain any reference,
72 // until such a time as a client object requests a stream for the path.
73 class NET_EXPORT_PRIVATE SpdyStream {
74 public:
75 // Delegate handles protocol specific behavior of spdy stream.
76 class NET_EXPORT_PRIVATE Delegate {
77 public:
78 Delegate() {}
80 // Called when the request headers have been sent. Never called
81 // for push streams. Must not cause the stream to be closed.
82 virtual void OnRequestHeadersSent() = 0;
84 // WARNING: This function is complicated! Be sure to read the
85 // whole comment below if you're working with code that implements
86 // or calls this function.
88 // Called when the response headers are updated from the
89 // server. |response_headers| contains the set of all headers
90 // received up to this point; delegates can assume that any
91 // headers previously received remain unchanged.
93 // This is called at least once before any data is received. If
94 // RESPONSE_HEADERS_ARE_INCOMPLETE is returned, this will be
95 // called again when more headers are received until
96 // RESPONSE_HEADERS_ARE_COMPLETE is returned, and any data
97 // received before then will be treated as a protocol error.
99 // If RESPONSE_HEADERS_ARE_INCOMPLETE is returned, the delegate
100 // must not have closed the stream. Otherwise, if
101 // RESPONSE_HEADERS_ARE_COMPLETE is returned, the delegate has
102 // processed the headers successfully. However, it still may have
103 // closed the stream, e.g. if the headers indicated an error
104 // condition.
106 // Some type-specific behavior:
108 // - For bidirectional streams, this may be called even after
109 // data is received, but it is expected that
110 // RESPONSE_HEADERS_ARE_COMPLETE is always returned. If
111 // RESPONSE_HEADERS_ARE_INCOMPLETE is returned, this is
112 // treated as a protocol error.
114 // - For request/response streams, this function is called
115 // exactly once before data is received, and it is expected
116 // that RESPONSE_HEADERS_ARE_COMPLETE is returned. If
117 // RESPONSE_HEADERS_ARE_INCOMPLETE is returned, this is
118 // treated as a protocol error.
120 // - For push streams, it is expected that this function will be
121 // called until RESPONSE_HEADERS_ARE_COMPLETE is returned
122 // before any data is received; any deviation from this is
123 // treated as a protocol error.
125 // TODO(akalin): Treat headers received after data has been
126 // received as a protocol error for non-bidirectional streams.
127 // TODO(jgraettinger): This should be at the semantic (HTTP) rather
128 // than stream layer. Streams shouldn't have a notion of header
129 // completeness. Move to SpdyHttpStream/SpdyWebsocketStream.
130 virtual SpdyResponseHeadersStatus OnResponseHeadersUpdated(
131 const SpdyHeaderBlock& response_headers) = 0;
133 // Called when data is received after all required response
134 // headers have been received. |buffer| may be NULL, which signals
135 // EOF. Must return OK if the data was received successfully, or
136 // a network error code otherwise.
138 // May cause the stream to be closed.
139 virtual void OnDataReceived(scoped_ptr<SpdyBuffer> buffer) = 0;
141 // Called when data is sent. Must not cause the stream to be
142 // closed.
143 virtual void OnDataSent() = 0;
145 // Called when SpdyStream is closed. No other delegate functions
146 // will be called after this is called, and the delegate must not
147 // access the stream after this is called. Must not cause the
148 // stream to be be (re-)closed.
150 // TODO(akalin): Allow this function to re-close the stream and
151 // handle it gracefully.
152 virtual void OnClose(int status) = 0;
154 protected:
155 virtual ~Delegate() {}
157 private:
158 DISALLOW_COPY_AND_ASSIGN(Delegate);
161 // SpdyStream constructor
162 SpdyStream(SpdyStreamType type,
163 const base::WeakPtr<SpdySession>& session,
164 const GURL& url,
165 RequestPriority priority,
166 int32 initial_send_window_size,
167 int32 initial_recv_window_size,
168 const BoundNetLog& net_log);
170 ~SpdyStream();
172 // Set the delegate, which must not be NULL. Must not be called more
173 // than once. For push streams, calling this may cause buffered data
174 // to be sent to the delegate (from a posted task).
175 void SetDelegate(Delegate* delegate);
177 // Detach the delegate from the stream, which must not yet be
178 // closed, and cancel it.
179 void DetachDelegate();
181 // The time at which the first bytes of the response were received
182 // from the server, or null if the response hasn't been received
183 // yet.
184 base::Time response_time() const { return response_time_; }
186 SpdyStreamType type() const { return type_; }
188 SpdyStreamId stream_id() const { return stream_id_; }
189 void set_stream_id(SpdyStreamId stream_id) { stream_id_ = stream_id; }
191 const GURL& url() const { return url_; }
193 RequestPriority priority() const { return priority_; }
195 int32 send_window_size() const { return send_window_size_; }
197 int32 recv_window_size() const { return recv_window_size_; }
199 bool send_stalled_by_flow_control() const {
200 return send_stalled_by_flow_control_;
203 void set_send_stalled_by_flow_control(bool stalled) {
204 send_stalled_by_flow_control_ = stalled;
207 // Called by the session to adjust this stream's send window size by
208 // |delta_window_size|, which is the difference between the
209 // SETTINGS_INITIAL_WINDOW_SIZE in the most recent SETTINGS frame
210 // and the previous initial send window size, possibly unstalling
211 // this stream. Although |delta_window_size| may cause this stream's
212 // send window size to go negative, it must not cause it to wrap
213 // around in either direction. Does nothing if the stream is already
214 // closed.
216 // If stream flow control is turned off, this must not be called.
217 void AdjustSendWindowSize(int32 delta_window_size);
219 // Called when bytes are consumed from a SpdyBuffer for a DATA frame
220 // that is to be written or is being written. Increases the send
221 // window size accordingly if some or all of the SpdyBuffer is being
222 // discarded.
224 // If stream flow control is turned off, this must not be called.
225 void OnWriteBufferConsumed(size_t frame_payload_size,
226 size_t consume_size,
227 SpdyBuffer::ConsumeSource consume_source);
229 // Called by the session to increase this stream's send window size
230 // by |delta_window_size| (which must be at least 1) from a received
231 // WINDOW_UPDATE frame or from a dropped DATA frame that was
232 // intended to be sent, possibly unstalling this stream. If
233 // |delta_window_size| would cause this stream's send window size to
234 // overflow, calls into the session to reset this stream. Does
235 // nothing if the stream is already closed.
237 // If stream flow control is turned off, this must not be called.
238 void IncreaseSendWindowSize(int32 delta_window_size);
240 // If stream flow control is turned on, called by the session to
241 // decrease this stream's send window size by |delta_window_size|,
242 // which must be at least 0 and at most kMaxSpdyFrameChunkSize.
243 // |delta_window_size| must not cause this stream's send window size
244 // to go negative. Does nothing if the stream is already closed.
246 // If stream flow control is turned off, this must not be called.
247 void DecreaseSendWindowSize(int32 delta_window_size);
249 // Called when bytes are consumed by the delegate from a SpdyBuffer
250 // containing received data. Increases the receive window size
251 // accordingly.
253 // If stream flow control is turned off, this must not be called.
254 void OnReadBufferConsumed(size_t consume_size,
255 SpdyBuffer::ConsumeSource consume_source);
257 // Called by OnReadBufferConsume to increase this stream's receive
258 // window size by |delta_window_size|, which must be at least 1 and
259 // must not cause this stream's receive window size to overflow,
260 // possibly also sending a WINDOW_UPDATE frame. Does nothing if the
261 // stream is not active.
263 // If stream flow control is turned off, this must not be called.
264 void IncreaseRecvWindowSize(int32 delta_window_size);
266 // Called by OnDataReceived (which is in turn called by the session)
267 // to decrease this stream's receive window size by
268 // |delta_window_size|, which must be at least 1 and must not cause
269 // this stream's receive window size to go negative.
271 // If stream flow control is turned off or the stream is not active,
272 // this must not be called.
273 void DecreaseRecvWindowSize(int32 delta_window_size);
275 int GetPeerAddress(IPEndPoint* address) const;
276 int GetLocalAddress(IPEndPoint* address) const;
278 // Returns true if the underlying transport socket ever had any reads or
279 // writes.
280 bool WasEverUsed() const;
282 const BoundNetLog& net_log() const { return net_log_; }
284 base::Time GetRequestTime() const;
285 void SetRequestTime(base::Time t);
287 // Called at most once by the SpdySession when the initial response
288 // headers have been received for this stream, i.e., a SYN_REPLY (or
289 // SYN_STREAM for push streams) frame has been received. This is the
290 // entry point for a push stream. Returns a status code; if it is
291 // an error, the stream was closed by this function.
292 int OnInitialResponseHeadersReceived(const SpdyHeaderBlock& response_headers,
293 base::Time response_time,
294 base::TimeTicks recv_first_byte_time);
296 // Called by the SpdySession (only after
297 // OnInitialResponseHeadersReceived() has been called) when
298 // late-bound headers are received for a stream. Returns a status
299 // code; if it is an error, the stream was closed by this function.
300 int OnAdditionalResponseHeadersReceived(
301 const SpdyHeaderBlock& additional_response_headers);
303 // Called by the SpdySession when response data has been received
304 // for this stream. This callback may be called multiple times as
305 // data arrives from the network, and will never be called prior to
306 // OnResponseHeadersReceived.
308 // |buffer| contains the data received, or NULL if the stream is
309 // being closed. The stream must copy any data from this
310 // buffer before returning from this callback.
312 // |length| is the number of bytes received (at most 2^24 - 1) or 0 if
313 // the stream is being closed.
314 void OnDataReceived(scoped_ptr<SpdyBuffer> buffer);
316 // Called by the SpdySession when a frame has been successfully and
317 // completely written. |frame_size| is the total size of the frame
318 // in bytes, including framing overhead.
319 void OnFrameWriteComplete(SpdyFrameType frame_type, size_t frame_size);
321 // SYN_STREAM-specific write handler invoked by OnFrameWriteComplete().
322 int OnRequestHeadersSent();
324 // DATA-specific write handler invoked by OnFrameWriteComplete().
325 // If more data is already available to be written, the next write is
326 // queued and ERR_IO_PENDING is returned. Returns OK otherwise.
327 int OnDataSent(size_t frame_size);
329 // Called by the SpdySession when the request is finished. This callback
330 // will always be called at the end of the request and signals to the
331 // stream that the stream has no more network events. No further callbacks
332 // to the stream will be made after this call.
333 // |status| is an error code or OK.
334 void OnClose(int status);
336 // Called by the SpdySession to log stream related errors.
337 void LogStreamError(int status, const std::string& description);
339 // If this stream is active, reset it, and close it otherwise. In
340 // either case the stream is deleted.
341 void Cancel();
343 // Close this stream without sending a RST_STREAM and delete
344 // it.
345 void Close();
347 // Must be used only by |session_|.
348 base::WeakPtr<SpdyStream> GetWeakPtr();
350 // Interface for the delegate to use.
352 // Only one send can be in flight at a time, except for push
353 // streams, which must not send anything.
355 // Sends the request headers. The delegate is called back via
356 // OnRequestHeadersSent() when the request headers have completed
357 // sending. |send_status| must be MORE_DATA_TO_SEND for
358 // bidirectional streams; for request/response streams, it must be
359 // MORE_DATA_TO_SEND if the request has data to upload, or
360 // NO_MORE_DATA_TO_SEND if not.
361 int SendRequestHeaders(scoped_ptr<SpdyHeaderBlock> request_headers,
362 SpdySendStatus send_status);
364 // Sends a DATA frame. The delegate will be notified via
365 // OnDataSent() when the send is complete. |send_status| must be
366 // MORE_DATA_TO_SEND for bidirectional streams; for request/response
367 // streams, it must be MORE_DATA_TO_SEND if there is more data to
368 // upload, or NO_MORE_DATA_TO_SEND if not.
369 void SendData(IOBuffer* data, int length, SpdySendStatus send_status);
371 // Fills SSL info in |ssl_info| and returns true when SSL is in use.
372 bool GetSSLInfo(SSLInfo* ssl_info,
373 bool* was_npn_negotiated,
374 NextProto* protocol_negotiated);
376 // Fills SSL Certificate Request info |cert_request_info| and returns
377 // true when SSL is in use.
378 bool GetSSLCertRequestInfo(SSLCertRequestInfo* cert_request_info);
380 // If the stream is stalled on sending data, but the session is not
381 // stalled on sending data and |send_window_size_| is positive, then
382 // set |send_stalled_by_flow_control_| to false and unstall the data
383 // sending. Called by the session or by the stream itself. Must be
384 // called only when the stream is still open.
385 void PossiblyResumeIfSendStalled();
387 // Returns whether or not this stream is closed. Note that the only
388 // time a stream is closed and not deleted is in its delegate's
389 // OnClose() method.
390 bool IsClosed() const;
392 // Returns whether the streams local endpoint is closed.
393 // The remote endpoint may still be active.
394 bool IsLocallyClosed() const;
396 // Returns whether this stream is IDLE: request and response headers
397 // have neither been sent nor receieved.
398 bool IsIdle() const;
400 // Returns whether or not this stream is fully open: that request and
401 // response headers are complete, and it is not in a half-closed state.
402 bool IsOpen() const;
404 // Returns the protocol used by this stream. Always between
405 // kProtoSPDYMinimumVersion and kProtoSPDYMaximumVersion.
406 NextProto GetProtocol() const;
408 int response_status() const { return response_status_; }
410 void IncrementRawReceivedBytes(size_t received_bytes) {
411 raw_received_bytes_ += received_bytes;
414 int64 raw_received_bytes() const { return raw_received_bytes_; }
416 bool GetLoadTimingInfo(LoadTimingInfo* load_timing_info) const;
418 // Get the URL from the appropriate stream headers, or the empty
419 // GURL() if it is unknown.
421 // TODO(akalin): Figure out if we really need this function,
422 // i.e. can we just use the URL this stream was created with and/or
423 // one we receive headers validate that the URL from them is the
424 // same.
425 GURL GetUrlFromHeaders() const;
427 // Returns whether the URL for this stream is known.
429 // TODO(akalin): Remove this, as it's only used in tests.
430 bool HasUrlFromHeaders() const;
432 SpdyMajorVersion GetProtocolVersion() const;
434 private:
435 class SynStreamBufferProducer;
436 class HeaderBufferProducer;
438 // SpdyStream states and transitions are modeled
439 // on the HTTP/2 stream state machine. All states and transitions
440 // are modeled, with the exceptions of RESERVED_LOCAL (the client
441 // cannot initate push streams), and the transition to OPEN due to
442 // a remote SYN_STREAM (the client can only initate streams).
443 // TODO(jgraettinger): RESERVED_REMOTE must be added to the state
444 // machine when PUSH_PROMISE is implemented.
445 enum State {
446 STATE_IDLE,
447 STATE_OPEN,
448 STATE_HALF_CLOSED_LOCAL_UNCLAIMED,
449 STATE_HALF_CLOSED_LOCAL,
450 STATE_HALF_CLOSED_REMOTE,
451 STATE_CLOSED,
454 // Update the histograms. Can safely be called repeatedly, but should only
455 // be called after the stream has completed.
456 void UpdateHistograms();
458 // When a server-push stream is claimed by SetDelegate(), this function is
459 // posted on the current MessageLoop to replay everything the server has sent.
460 // From the perspective of SpdyStream's state machine, headers, data, and
461 // FIN states received prior to the delegate being attached have not yet been
462 // read. While buffered by |pending_recv_data_| it's not until
463 // PushedStreamReplay() is invoked that reads are considered
464 // to have occurred, driving the state machine forward.
465 void PushedStreamReplay();
467 // Produces the SYN_STREAM frame for the stream. The stream must
468 // already be activated.
469 scoped_ptr<SpdyFrame> ProduceSynStreamFrame();
471 // Produce the initial HEADER frame for the stream with the given
472 // block. The stream must already be activated.
473 scoped_ptr<SpdyFrame> ProduceHeaderFrame(
474 scoped_ptr<SpdyHeaderBlock> header_block);
476 // Queues the send for next frame of the remaining data in
477 // |pending_send_data_|. Must be called only when
478 // |pending_send_data_| is set.
479 void QueueNextDataFrame();
481 // Merge the given headers into |response_headers_| and calls
482 // OnResponseHeadersUpdated() on the delegate (if attached).
483 // Returns a status code; if it is an error, the stream was closed
484 // by this function.
485 int MergeWithResponseHeaders(const SpdyHeaderBlock& new_response_headers);
487 static std::string DescribeState(State state);
489 const SpdyStreamType type_;
491 SpdyStreamId stream_id_;
492 const GURL url_;
493 const RequestPriority priority_;
495 // Flow control variables.
496 bool send_stalled_by_flow_control_;
497 int32 send_window_size_;
498 int32 recv_window_size_;
499 int32 unacked_recv_window_bytes_;
501 ScopedBandwidthMetrics metrics_;
503 const base::WeakPtr<SpdySession> session_;
505 // The transaction should own the delegate.
506 SpdyStream::Delegate* delegate_;
508 // The headers for the request to send.
510 // TODO(akalin): Hang onto this only until we send it. This
511 // necessitates stashing the URL separately.
512 scoped_ptr<SpdyHeaderBlock> request_headers_;
514 // Data waiting to be sent, and the close state of the local endpoint
515 // after the data is fully written.
516 scoped_refptr<DrainableIOBuffer> pending_send_data_;
517 SpdySendStatus pending_send_status_;
519 // Data waiting to be received, and the close state of the remote endpoint
520 // after the data is fully read. Specifically, data received before the
521 // delegate is attached must be buffered and later replayed. A remote FIN
522 // is represented by a final, zero-length buffer.
523 ScopedVector<SpdyBuffer> pending_recv_data_;
525 // The time at which the request was made that resulted in this response.
526 // For cached responses, this time could be "far" in the past.
527 base::Time request_time_;
529 SpdyHeaderBlock response_headers_;
530 SpdyResponseHeadersStatus response_headers_status_;
531 base::Time response_time_;
533 State io_state_;
535 // Since we buffer the response, we also buffer the response status.
536 // Not valid until the stream is closed.
537 int response_status_;
539 BoundNetLog net_log_;
541 base::TimeTicks send_time_;
542 base::TimeTicks recv_first_byte_time_;
543 base::TimeTicks recv_last_byte_time_;
545 // Number of bytes that have been received on this stream, including frame
546 // overhead and headers.
547 int64 raw_received_bytes_;
549 // Number of data bytes that have been sent/received on this stream, not
550 // including frame overhead. Note that this does not count headers.
551 int send_bytes_;
552 int recv_bytes_;
554 std::string domain_bound_private_key_;
555 std::string domain_bound_cert_;
556 ServerBoundCertService::RequestHandle domain_bound_cert_request_handle_;
558 // Guards calls of delegate write handlers ensuring |this| is not destroyed.
559 // TODO(jgraettinger): Consider removing after crbug.com/35511 is tracked
560 // down.
561 bool write_handler_guard_;
563 base::WeakPtrFactory<SpdyStream> weak_ptr_factory_;
565 DISALLOW_COPY_AND_ASSIGN(SpdyStream);
568 } // namespace net
570 #endif // NET_SPDY_SPDY_STREAM_H_