| // Copyright (c) 2009 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef NET_FLIP_FLIP_SESSION_H_ |
| #define NET_FLIP_FLIP_SESSION_H_ |
| |
| #include <deque> |
| #include <list> |
| #include <map> |
| #include <queue> |
| #include <string> |
| |
| #include "base/ref_counted.h" |
| #include "net/base/io_buffer.h" |
| #include "net/base/load_states.h" |
| #include "net/base/net_errors.h" |
| #include "net/base/request_priority.h" |
| #include "net/base/ssl_config_service.h" |
| #include "net/base/upload_data_stream.h" |
| #include "net/flip/flip_framer.h" |
| #include "net/flip/flip_io_buffer.h" |
| #include "net/flip/flip_protocol.h" |
| #include "net/flip/flip_session_pool.h" |
| #include "net/socket/client_socket.h" |
| #include "net/socket/client_socket_handle.h" |
| #include "testing/platform_test.h" |
| |
| namespace net { |
| |
| class FlipStream; |
| class HttpNetworkSession; |
| class HttpRequestInfo; |
| class HttpResponseInfo; |
| class LoadLog; |
| class SSLInfo; |
| |
| class FlipSession : public base::RefCounted<FlipSession>, |
| public flip::FlipFramerVisitorInterface { |
| public: |
| // Get the domain for this FlipSession. |
| const std::string& domain() const { return domain_; } |
| |
| // Connect the FLIP Socket. |
| // Returns net::Error::OK on success. |
| // Note that this call does not wait for the connect to complete. Callers can |
| // immediately start using the FlipSession while it connects. |
| net::Error Connect(const std::string& group_name, |
| const HostResolver::RequestInfo& host, |
| RequestPriority priority, |
| LoadLog* load_log); |
| |
| // Get a stream for a given |request|. In the typical case, this will involve |
| // the creation of a new stream (and will send the SYN frame). If the server |
| // initiates a stream, it might already exist for a given path. The server |
| // might also not have initiated the stream yet, but indicated it will via |
| // X-Associated-Content. |
| // Returns the new or existing stream. Never returns NULL. |
| scoped_refptr<FlipStream> GetOrCreateStream(const HttpRequestInfo& request, |
| const UploadDataStream* upload_data, LoadLog* log); |
| |
| // Write a data frame to the stream. |
| // Used to create and queue a data frame for the given stream. |
| int WriteStreamData(flip::FlipStreamId stream_id, net::IOBuffer* data, |
| int len); |
| |
| // Cancel a stream. |
| bool CancelStream(flip::FlipStreamId stream_id); |
| |
| // Check if a stream is active. |
| bool IsStreamActive(flip::FlipStreamId stream_id) const; |
| |
| // The LoadState is used for informing the user of the current network |
| // status, such as "resolving host", "connecting", etc. |
| LoadState GetLoadState() const; |
| |
| // Enable or disable SSL. |
| static void SetSSLMode(bool enable) { use_ssl_ = enable; } |
| static bool SSLMode() { return use_ssl_; } |
| |
| protected: |
| friend class FlipSessionPool; |
| |
| enum State { |
| IDLE, |
| CONNECTING, |
| CONNECTED, |
| CLOSED |
| }; |
| |
| // Provide access to the framer for testing. |
| flip::FlipFramer* GetFramer() { return &flip_framer_; } |
| |
| // Create a new FlipSession. |
| // |host| is the hostname that this session connects to. |
| FlipSession(const std::string& host, HttpNetworkSession* session); |
| |
| // Closes all open streams. Used as part of shutdown. |
| void CloseAllStreams(net::Error code); |
| |
| private: |
| friend class base::RefCounted<FlipSession>; |
| |
| typedef std::map<int, scoped_refptr<FlipStream> > ActiveStreamMap; |
| typedef std::list<scoped_refptr<FlipStream> > ActiveStreamList; |
| typedef std::map<std::string, scoped_refptr<FlipStream> > PendingStreamMap; |
| typedef std::priority_queue<FlipIOBuffer> OutputQueue; |
| |
| virtual ~FlipSession(); |
| |
| // Used by FlipSessionPool to initialize with a pre-existing socket. |
| void InitializeWithSocket(ClientSocketHandle* connection); |
| |
| // FlipFramerVisitorInterface |
| virtual void OnError(flip::FlipFramer*); |
| virtual void OnStreamFrameData(flip::FlipStreamId stream_id, |
| const char* data, |
| size_t len); |
| virtual void OnControl(const flip::FlipControlFrame* frame); |
| |
| // Control frame handlers. |
| void OnSyn(const flip::FlipSynStreamControlFrame* frame, |
| const flip::FlipHeaderBlock* headers); |
| void OnSynReply(const flip::FlipSynReplyControlFrame* frame, |
| const flip::FlipHeaderBlock* headers); |
| void OnFin(const flip::FlipFinStreamControlFrame* frame); |
| |
| // IO Callbacks |
| void OnTCPConnect(int result); |
| void OnSSLConnect(int result); |
| void OnReadComplete(int result); |
| void OnWriteComplete(int result); |
| |
| // Start reading from the socket. |
| void ReadSocket(); |
| |
| // Write current data to the socket. |
| void WriteSocketLater(); |
| void WriteSocket(); |
| |
| // Get a new stream id. |
| int GetNewStreamId(); |
| |
| // Closes this session. This will close all active streams and mark |
| // the session as permanently closed. |
| // |err| should not be OK; this function is intended to be called on |
| // error. |
| void CloseSessionOnError(net::Error err); |
| |
| // Track active streams in the active stream list. |
| void ActivateStream(FlipStream* stream); |
| void DeactivateStream(flip::FlipStreamId id); |
| |
| // Check if we have a pending pushed-stream for this url |
| // Returns the stream if found (and returns it from the pending |
| // list), returns NULL otherwise. |
| scoped_refptr<FlipStream> GetPushStream(const std::string& url); |
| |
| void GetSSLInfo(SSLInfo* ssl_info); |
| |
| // Callbacks for the Flip session. |
| CompletionCallbackImpl<FlipSession> connect_callback_; |
| CompletionCallbackImpl<FlipSession> ssl_connect_callback_; |
| CompletionCallbackImpl<FlipSession> read_callback_; |
| CompletionCallbackImpl<FlipSession> write_callback_; |
| |
| // The domain this session is connected to. |
| std::string domain_; |
| |
| SSLConfig ssl_config_; |
| |
| scoped_refptr<HttpNetworkSession> session_; |
| |
| // The socket handle for this session. |
| scoped_ptr<ClientSocketHandle> connection_; |
| |
| // The read buffer used to read data from the socket. |
| scoped_refptr<IOBuffer> read_buffer_; |
| bool read_pending_; |
| |
| int stream_hi_water_mark_; // The next stream id to use. |
| |
| // TODO(mbelshe): We need to track these stream lists better. |
| // I suspect it is possible to remove a stream from |
| // one list, but not the other. |
| |
| // Map from stream id to all active streams. Streams are active in the sense |
| // that they have a consumer (typically FlipNetworkTransaction and regardless |
| // of whether or not there is currently any ongoing IO [might be waiting for |
| // the server to start pushing the stream]) or there are still network events |
| // incoming even though the consumer has already gone away (cancellation). |
| // TODO(willchan): Perhaps we should separate out cancelled streams and move |
| // them into a separate ActiveStreamMap, and not deliver network events to |
| // them? |
| ActiveStreamMap active_streams_; |
| // List of all the streams that have already started to be pushed by the |
| // server, but do not have consumers yet. |
| ActiveStreamList pushed_streams_; |
| // List of streams declared in X-Associated-Content headers, but do not have |
| // consumers yet. |
| // The key is a string representing the path of the URI being pushed. |
| PendingStreamMap pending_streams_; |
| |
| // As we gather data to be sent, we put it into the output queue. |
| OutputQueue queue_; |
| |
| // The packet we are currently sending. |
| bool write_pending_; // Will be true when a write is in progress. |
| FlipIOBuffer in_flight_write_; // This is the write buffer in progress. |
| |
| // Flag if we have a pending message scheduled for WriteSocket. |
| bool delayed_write_pending_; |
| |
| // Flag if we're using an SSL connection for this FlipSession. |
| bool is_secure_; |
| |
| // Flip Frame state. |
| flip::FlipFramer flip_framer_; |
| |
| // If an error has occurred on the session, the session is effectively |
| // dead. Record this error here. When no error has occurred, |error_| will |
| // be OK. |
| net::Error error_; |
| State state_; |
| |
| // Some statistics counters for the session. |
| int streams_initiated_count_; |
| int streams_pushed_count_; |
| int streams_pushed_and_claimed_count_; |
| int streams_abandoned_count_; |
| |
| static bool use_ssl_; |
| }; |
| |
| } // namespace net |
| |
| #endif // NET_FLIP_FLIP_SESSION_H_ |