/* * libjingle * Copyright 2004--2005, Google Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ // Copyright 2005 Google Inc. All Rights Reserved. // #ifndef TALK_BASE_HTTPBASE_H__ #define TALK_BASE_HTTPBASE_H__ #include "talk/base/httpcommon.h" namespace talk_base { class StreamInterface; /////////////////////////////////////////////////////////////////////////////// // HttpParser - Parses an HTTP stream provided via Process and end_of_input, and // generates events for: // Structural Elements: Leader, Headers, Document Data // Events: End of Headers, End of Document, Errors /////////////////////////////////////////////////////////////////////////////// class HttpParser { public: enum ProcessResult { PR_CONTINUE, PR_BLOCK, PR_COMPLETE }; HttpParser(); virtual ~HttpParser(); void reset(); ProcessResult Process(const char* buffer, size_t len, size_t* processed, HttpError* error); bool is_valid_end_of_input() const; void complete(HttpError err); size_t GetDataRemaining() const { return data_size_; } protected: ProcessResult ProcessLine(const char* line, size_t len, HttpError* error); // HttpParser Interface virtual ProcessResult ProcessLeader(const char* line, size_t len, HttpError* error) = 0; virtual ProcessResult ProcessHeader(const char* name, size_t nlen, const char* value, size_t vlen, HttpError* error) = 0; virtual ProcessResult ProcessHeaderComplete(bool chunked, size_t& data_size, HttpError* error) = 0; virtual ProcessResult ProcessData(const char* data, size_t len, size_t& read, HttpError* error) = 0; virtual void OnComplete(HttpError err) = 0; private: enum State { ST_LEADER, ST_HEADERS, ST_CHUNKSIZE, ST_CHUNKTERM, ST_TRAILERS, ST_DATA, ST_COMPLETE } state_; bool chunked_; size_t data_size_; }; /////////////////////////////////////////////////////////////////////////////// // IHttpNotify /////////////////////////////////////////////////////////////////////////////// enum HttpMode { HM_NONE, HM_CONNECT, HM_RECV, HM_SEND }; class IHttpNotify { public: virtual ~IHttpNotify() {} virtual HttpError onHttpHeaderComplete(bool chunked, size_t& data_size) = 0; virtual void onHttpComplete(HttpMode mode, HttpError err) = 0; virtual void onHttpClosed(HttpError err) = 0; }; /////////////////////////////////////////////////////////////////////////////// // HttpBase - Provides a state machine for implementing HTTP-based components. // Attach HttpBase to a StreamInterface which represents a bidirectional HTTP // stream, and then call send() or recv() to initiate sending or receiving one // side of an HTTP transaction. By default, HttpBase operates as an I/O pump, // moving data from the HTTP stream to the HttpData object and vice versa. // However, it can also operate in stream mode, in which case the user of the // stream interface drives I/O via calls to Read(). /////////////////////////////////////////////////////////////////////////////// class HttpBase : private HttpParser, public sigslot::has_slots<> { public: HttpBase(); virtual ~HttpBase(); void notify(IHttpNotify* notify) { notify_ = notify; } bool attach(StreamInterface* stream); StreamInterface* stream() { return http_stream_; } StreamInterface* detach(); bool isConnected() const; void send(HttpData* data); void recv(HttpData* data); void abort(HttpError err); HttpMode mode() const { return mode_; } void set_ignore_data(bool ignore) { ignore_data_ = ignore; } bool ignore_data() const { return ignore_data_; } // Obtaining this stream puts HttpBase into stream mode until the stream // is closed. HttpBase can only expose one open stream interface at a time. // Further calls will return NULL. StreamInterface* GetDocumentStream(); protected: // Do cleanup when the http stream closes (error may be 0 for a clean // shutdown), and return the error code to signal. HttpError HandleStreamClose(int error); // DoReceiveLoop acts as a data pump, pulling data from the http stream, // pushing it through the HttpParser, and then populating the HttpData object // based on the callbacks from the parser. One of the most interesting // callbacks is ProcessData, which provides the actual http document body. // This data is then written to the HttpData::document. As a result, data // flows from the network to the document, with some incidental protocol // parsing in between. // Ideally, we would pass in the document* to DoReceiveLoop, to more easily // support GetDocumentStream(). However, since the HttpParser is callback // driven, we are forced to store the pointer somewhere until the callback // is triggered. // Returns true if the received document has finished, and // HttpParser::complete should be called. bool DoReceiveLoop(HttpError* err); void read_and_process_data(); void flush_data(); bool queue_headers(); void do_complete(HttpError err = HE_NONE); void OnHttpStreamEvent(StreamInterface* stream, int events, int error); void OnDocumentEvent(StreamInterface* stream, int events, int error); // HttpParser Interface virtual ProcessResult ProcessLeader(const char* line, size_t len, HttpError* error); virtual ProcessResult ProcessHeader(const char* name, size_t nlen, const char* value, size_t vlen, HttpError* error); virtual ProcessResult ProcessHeaderComplete(bool chunked, size_t& data_size, HttpError* error); virtual ProcessResult ProcessData(const char* data, size_t len, size_t& read, HttpError* error); virtual void OnComplete(HttpError err); private: class DocumentStream; friend class DocumentStream; enum { kBufferSize = 32 * 1024 }; HttpMode mode_; HttpData* data_; IHttpNotify* notify_; StreamInterface* http_stream_; DocumentStream* doc_stream_; char buffer_[kBufferSize]; size_t len_; bool ignore_data_, chunk_data_; HttpData::const_iterator header_; }; /////////////////////////////////////////////////////////////////////////////// } // namespace talk_base #endif // TALK_BASE_HTTPBASE_H__