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. 4 5 #ifndef NET_WEBSOCKETS_WEBSOCKET_STREAM_H_ 6 #define NET_WEBSOCKETS_WEBSOCKET_STREAM_H_ 7 8 #include <string> 9 #include <vector> 10 11 #include "base/basictypes.h" 12 #include "base/callback_forward.h" 13 #include "base/memory/scoped_ptr.h" 14 #include "base/memory/scoped_vector.h" 15 #include "net/base/completion_callback.h" 16 #include "net/base/net_export.h" 17 18 class GURL; 19 20 namespace net { 21 22 class BoundNetLog; 23 class URLRequestContext; 24 struct WebSocketFrame; 25 26 // WebSocketStreamRequest is the caller's handle to the process of creation of a 27 // WebSocketStream. Deleting the object before the OnSuccess or OnFailure 28 // callbacks are called will cancel the request (and neither callback will be 29 // called). After OnSuccess or OnFailure have been called, this object may be 30 // safely deleted without side-effects. 31 class NET_EXPORT_PRIVATE WebSocketStreamRequest { 32 public: 33 virtual ~WebSocketStreamRequest(); 34 }; 35 36 // WebSocketStream is a transport-agnostic interface for reading and writing 37 // WebSocket frames. This class provides an abstraction for WebSocket streams 38 // based on various transport layers, such as normal WebSocket connections 39 // (WebSocket protocol upgraded from HTTP handshake), SPDY transports, or 40 // WebSocket connections with multiplexing extension. Subtypes of 41 // WebSocketStream are responsible for managing the underlying transport 42 // appropriately. 43 // 44 // All functions except Close() can be asynchronous. If an operation cannot 45 // be finished synchronously, the function returns ERR_IO_PENDING, and 46 // |callback| will be called when the operation is finished. Non-null |callback| 47 // must be provided to these functions. 48 49 class NET_EXPORT_PRIVATE WebSocketStream { 50 public: 51 // A concrete object derived from ConnectDelegate is supplied by the caller to 52 // CreateAndConnectStream() to receive the result of the connection. 53 class NET_EXPORT_PRIVATE ConnectDelegate { 54 public: 55 virtual ~ConnectDelegate(); 56 // Called on successful connection. The parameter is an object derived from 57 // WebSocketStream. 58 virtual void OnSuccess(scoped_ptr<WebSocketStream> stream) = 0; 59 60 // Called on failure to connect. The parameter is either one of the values 61 // defined in net::WebSocketError, or an error defined by some WebSocket 62 // extension protocol that we implement. 63 virtual void OnFailure(unsigned short websocket_error) = 0; 64 }; 65 66 // Create and connect a WebSocketStream of an appropriate type. The actual 67 // concrete type returned depends on whether multiplexing or SPDY are being 68 // used to communicate with the remote server. If the handshake completed 69 // successfully, then connect_delegate->OnSuccess() is called with a 70 // WebSocketStream instance. If it failed, then connect_delegate->OnFailure() 71 // is called with a WebSocket result code corresponding to the error. Deleting 72 // the returned WebSocketStreamRequest object will cancel the connection, in 73 // which case the |connect_delegate| object that the caller passed will be 74 // deleted without any of its methods being called. Unless cancellation is 75 // required, the caller should keep the WebSocketStreamRequest object alive 76 // until connect_delegate->OnSuccess() or OnFailure() have been called, then 77 // it is safe to delete. 78 static scoped_ptr<WebSocketStreamRequest> CreateAndConnectStream( 79 const GURL& socket_url, 80 const std::vector<std::string>& requested_subprotocols, 81 const GURL& origin, 82 URLRequestContext* url_request_context, 83 const BoundNetLog& net_log, 84 scoped_ptr<ConnectDelegate> connect_delegate); 85 86 // Derived classes must make sure Close() is called when the stream is not 87 // closed on destruction. 88 virtual ~WebSocketStream(); 89 90 // Reads WebSocket frame data. This operation finishes when new frame data 91 // becomes available. 92 // 93 // |frames| remains owned by the caller and must be valid until the 94 // operation completes or Close() is called. |frames| must be empty on 95 // calling. 96 // 97 // This function should not be called while the previous call of ReadFrames() 98 // is still pending. 99 // 100 // Returns net::OK or one of the net::ERR_* codes. 101 // 102 // frames->size() >= 1 if the result is OK. 103 // 104 // Only frames with complete header information are inserted into |frames|. If 105 // the currently available bytes of a new frame do not form a complete frame 106 // header, then the implementation will buffer them until all the fields in 107 // the WebSocketFrameHeader object can be filled. If ReadFrames() is freshly 108 // called in this situation, it will return ERR_IO_PENDING exactly as if no 109 // data was available. 110 // 111 // Original frame boundaries are not preserved. In particular, if only part of 112 // a frame is available, then the frame will be split, and the available data 113 // will be returned immediately. 114 // 115 // When the socket is closed on the remote side, this method will return 116 // ERR_CONNECTION_CLOSED. It will not return OK with an empty vector. 117 // 118 // If the connection is closed in the middle of receiving an incomplete frame, 119 // ReadFrames may discard the incomplete frame. Since the renderer will 120 // discard any incomplete messages when the connection is closed, this makes 121 // no difference to the overall semantics. 122 // 123 // Implementations of ReadFrames() must be able to handle being deleted during 124 // the execution of callback.Run(). In practice this means that the method 125 // calling callback.Run() (and any calling methods in the same object) must 126 // return immediately without any further method calls or access to member 127 // variables. Implementors should write test(s) for this case. 128 virtual int ReadFrames(ScopedVector<WebSocketFrame>* frames, 129 const CompletionCallback& callback) = 0; 130 131 // Writes WebSocket frame data. 132 // 133 // |frames| must be valid until the operation completes or Close() is called. 134 // 135 // This function must not be called while a previous call of WriteFrames() is 136 // still pending. 137 // 138 // This method will only return OK if all frames were written completely. 139 // Otherwise it will return an appropriate net error code. 140 // 141 // The callback implementation is permitted to delete this 142 // object. Implementations of WriteFrames() should be robust against 143 // this. This generally means returning to the event loop immediately after 144 // calling the callback. 145 virtual int WriteFrames(ScopedVector<WebSocketFrame>* frames, 146 const CompletionCallback& callback) = 0; 147 148 // Closes the stream. All pending I/O operations (if any) are cancelled 149 // at this point, so |frames| can be freed. 150 virtual void Close() = 0; 151 152 // The subprotocol that was negotiated for the stream. If no protocol was 153 // negotiated, then the empty string is returned. 154 virtual std::string GetSubProtocol() const = 0; 155 156 // The extensions that were negotiated for the stream. Since WebSocketStreams 157 // can be layered, this may be different from what this particular 158 // WebSocketStream implements. The primary purpose of this accessor is to make 159 // the data available to Javascript. The format of the string is identical to 160 // the contents of the Sec-WebSocket-Extensions header supplied by the server, 161 // with some canonicalisations applied (leading and trailing whitespace 162 // removed, multiple headers concatenated into one comma-separated list). See 163 // RFC6455 section 9.1 for the exact format specification. If no 164 // extensions were negotiated, the empty string is returned. 165 virtual std::string GetExtensions() const = 0; 166 167 protected: 168 WebSocketStream(); 169 170 private: 171 DISALLOW_COPY_AND_ASSIGN(WebSocketStream); 172 }; 173 174 } // namespace net 175 176 #endif // NET_WEBSOCKETS_WEBSOCKET_STREAM_H_ 177