| /* |
| * 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. |
| */ |
| |
| #ifndef TALK_BASE_SOCKETADAPTERS_H_ |
| #define TALK_BASE_SOCKETADAPTERS_H_ |
| |
| #include <map> |
| #include <string> |
| |
| #include "talk/base/asyncsocket.h" |
| #include "talk/base/cryptstring.h" |
| #include "talk/base/logging.h" |
| |
| namespace talk_base { |
| |
| struct HttpAuthContext; |
| class ByteBuffer; |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| // Implements a socket adapter that can buffer and process data internally, |
| // as in the case of connecting to a proxy, where you must speak the proxy |
| // protocol before commencing normal socket behavior. |
| class BufferedReadAdapter : public AsyncSocketAdapter { |
| public: |
| BufferedReadAdapter(AsyncSocket* socket, size_t buffer_size); |
| virtual ~BufferedReadAdapter(); |
| |
| virtual int Send(const void* pv, size_t cb); |
| virtual int Recv(void* pv, size_t cb); |
| |
| protected: |
| int DirectSend(const void* pv, size_t cb) { |
| return AsyncSocketAdapter::Send(pv, cb); |
| } |
| |
| void BufferInput(bool on = true); |
| virtual void ProcessInput(char* data, size_t* len) = 0; |
| |
| virtual void OnReadEvent(AsyncSocket * socket); |
| |
| private: |
| char * buffer_; |
| size_t buffer_size_, data_len_; |
| bool buffering_; |
| DISALLOW_EVIL_CONSTRUCTORS(BufferedReadAdapter); |
| }; |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| // Interface for implementing proxy server sockets. |
| class AsyncProxyServerSocket : public BufferedReadAdapter { |
| public: |
| AsyncProxyServerSocket(AsyncSocket* socket, size_t buffer_size) |
| : BufferedReadAdapter(socket, buffer_size) {} |
| sigslot::signal2<AsyncProxyServerSocket*, |
| const SocketAddress&> SignalConnectRequest; |
| virtual void SendConnectResult(int err, const SocketAddress& addr) = 0; |
| }; |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| // Implements a socket adapter that performs the client side of a |
| // fake SSL handshake. Used for "ssltcp" P2P functionality. |
| class AsyncSSLSocket : public BufferedReadAdapter { |
| public: |
| explicit AsyncSSLSocket(AsyncSocket* socket); |
| |
| virtual int Connect(const SocketAddress& addr); |
| |
| protected: |
| virtual void OnConnectEvent(AsyncSocket* socket); |
| virtual void ProcessInput(char* data, size_t* len); |
| DISALLOW_EVIL_CONSTRUCTORS(AsyncSSLSocket); |
| }; |
| |
| // Implements a socket adapter that performs the server side of a |
| // fake SSL handshake. Used when implementing a relay server that does "ssltcp". |
| class AsyncSSLServerSocket : public BufferedReadAdapter { |
| public: |
| explicit AsyncSSLServerSocket(AsyncSocket* socket); |
| |
| protected: |
| virtual void ProcessInput(char* data, size_t* len); |
| DISALLOW_EVIL_CONSTRUCTORS(AsyncSSLServerSocket); |
| }; |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| // Implements a socket adapter that speaks the HTTP/S proxy protocol. |
| class AsyncHttpsProxySocket : public BufferedReadAdapter { |
| public: |
| AsyncHttpsProxySocket(AsyncSocket* socket, const std::string& user_agent, |
| const SocketAddress& proxy, |
| const std::string& username, const CryptString& password); |
| virtual ~AsyncHttpsProxySocket(); |
| |
| // If connect is forced, the adapter will always issue an HTTP CONNECT to the |
| // target address. Otherwise, it will connect only if the destination port |
| // is not port 80. |
| void SetForceConnect(bool force) { force_connect_ = force; } |
| |
| virtual int Connect(const SocketAddress& addr); |
| virtual SocketAddress GetRemoteAddress() const; |
| virtual int Close(); |
| virtual ConnState GetState() const; |
| |
| protected: |
| virtual void OnConnectEvent(AsyncSocket* socket); |
| virtual void OnCloseEvent(AsyncSocket* socket, int err); |
| virtual void ProcessInput(char* data, size_t* len); |
| |
| bool ShouldIssueConnect() const; |
| void SendRequest(); |
| void ProcessLine(char* data, size_t len); |
| void EndResponse(); |
| void Error(int error); |
| |
| private: |
| SocketAddress proxy_, dest_; |
| std::string agent_, user_, headers_; |
| CryptString pass_; |
| bool force_connect_; |
| size_t content_length_; |
| int defer_error_; |
| bool expect_close_; |
| enum ProxyState { |
| PS_INIT, PS_LEADER, PS_AUTHENTICATE, PS_SKIP_HEADERS, PS_ERROR_HEADERS, |
| PS_TUNNEL_HEADERS, PS_SKIP_BODY, PS_TUNNEL, PS_WAIT_CLOSE, PS_ERROR |
| } state_; |
| HttpAuthContext * context_; |
| std::string unknown_mechanisms_; |
| DISALLOW_EVIL_CONSTRUCTORS(AsyncHttpsProxySocket); |
| }; |
| |
| /* TODO: Implement this. |
| class AsyncHttpsProxyServerSocket : public AsyncProxyServerSocket { |
| public: |
| explicit AsyncHttpsProxyServerSocket(AsyncSocket* socket); |
| |
| private: |
| virtual void ProcessInput(char * data, size_t& len); |
| void Error(int error); |
| DISALLOW_EVIL_CONSTRUCTORS(AsyncHttpsProxyServerSocket); |
| }; |
| */ |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| // Implements a socket adapter that speaks the SOCKS proxy protocol. |
| class AsyncSocksProxySocket : public BufferedReadAdapter { |
| public: |
| AsyncSocksProxySocket(AsyncSocket* socket, const SocketAddress& proxy, |
| const std::string& username, const CryptString& password); |
| |
| virtual int Connect(const SocketAddress& addr); |
| virtual SocketAddress GetRemoteAddress() const; |
| virtual int Close(); |
| virtual ConnState GetState() const; |
| |
| protected: |
| virtual void OnConnectEvent(AsyncSocket* socket); |
| virtual void ProcessInput(char* data, size_t* len); |
| |
| void SendHello(); |
| void SendConnect(); |
| void SendAuth(); |
| void Error(int error); |
| |
| private: |
| enum State { |
| SS_INIT, SS_HELLO, SS_AUTH, SS_CONNECT, SS_TUNNEL, SS_ERROR |
| }; |
| State state_; |
| SocketAddress proxy_, dest_; |
| std::string user_; |
| CryptString pass_; |
| DISALLOW_EVIL_CONSTRUCTORS(AsyncSocksProxySocket); |
| }; |
| |
| // Implements a proxy server socket for the SOCKS protocol. |
| class AsyncSocksProxyServerSocket : public AsyncProxyServerSocket { |
| public: |
| explicit AsyncSocksProxyServerSocket(AsyncSocket* socket); |
| |
| private: |
| virtual void ProcessInput(char* data, size_t* len); |
| void DirectSend(const ByteBuffer& buf); |
| |
| void HandleHello(ByteBuffer* request); |
| void SendHelloReply(int method); |
| void HandleAuth(ByteBuffer* request); |
| void SendAuthReply(int result); |
| void HandleConnect(ByteBuffer* request); |
| virtual void SendConnectResult(int result, const SocketAddress& addr); |
| |
| void Error(int error); |
| |
| static const int kBufferSize = 1024; |
| enum State { |
| SS_HELLO, SS_AUTH, SS_CONNECT, SS_CONNECT_PENDING, SS_TUNNEL, SS_ERROR |
| }; |
| State state_; |
| DISALLOW_EVIL_CONSTRUCTORS(AsyncSocksProxyServerSocket); |
| }; |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| // Implements a socket adapter that logs everything that it sends and receives. |
| class LoggingSocketAdapter : public AsyncSocketAdapter { |
| public: |
| LoggingSocketAdapter(AsyncSocket* socket, LoggingSeverity level, |
| const char * label, bool hex_mode = false); |
| |
| virtual int Send(const void *pv, size_t cb); |
| virtual int SendTo(const void *pv, size_t cb, const SocketAddress& addr); |
| virtual int Recv(void *pv, size_t cb); |
| virtual int RecvFrom(void *pv, size_t cb, SocketAddress *paddr); |
| virtual int Close(); |
| |
| protected: |
| virtual void OnConnectEvent(AsyncSocket * socket); |
| virtual void OnCloseEvent(AsyncSocket * socket, int err); |
| |
| private: |
| LoggingSeverity level_; |
| std::string label_; |
| bool hex_mode_; |
| LogMultilineState lms_; |
| DISALLOW_EVIL_CONSTRUCTORS(LoggingSocketAdapter); |
| }; |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| } // namespace talk_base |
| |
| #endif // TALK_BASE_SOCKETADAPTERS_H_ |