/* * libjingle * Copyright 2004--2008, 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_P2P_CLIENT_HTTPPORTALLOCATOR_H_ #define TALK_P2P_CLIENT_HTTPPORTALLOCATOR_H_ #include #include #include #include "talk/p2p/client/basicportallocator.h" class HttpPortAllocatorTest_TestSessionRequestUrl_Test; namespace rtc { class AsyncHttpRequest; class SignalThread; } namespace cricket { class HttpPortAllocatorBase : public BasicPortAllocator { public: // The number of HTTP requests we should attempt before giving up. static const int kNumRetries; // Records the URL that we will GET in order to create a session. static const char kCreateSessionURL[]; HttpPortAllocatorBase(rtc::NetworkManager* network_manager, const std::string& user_agent); HttpPortAllocatorBase(rtc::NetworkManager* network_manager, rtc::PacketSocketFactory* socket_factory, const std::string& user_agent); virtual ~HttpPortAllocatorBase(); // CreateSession is defined in BasicPortAllocator but is // redefined here as pure virtual. virtual PortAllocatorSession* CreateSessionInternal( const std::string& content_name, int component, const std::string& ice_ufrag, const std::string& ice_pwd) = 0; void SetStunHosts(const std::vector& hosts) { if (!hosts.empty()) { stun_hosts_ = hosts; } } void SetRelayHosts(const std::vector& hosts) { if (!hosts.empty()) { relay_hosts_ = hosts; } } void SetRelayToken(const std::string& relay) { relay_token_ = relay; } const std::vector& stun_hosts() const { return stun_hosts_; } const std::vector& relay_hosts() const { return relay_hosts_; } const std::string& relay_token() const { return relay_token_; } const std::string& user_agent() const { return agent_; } private: std::vector stun_hosts_; std::vector relay_hosts_; std::string relay_token_; std::string agent_; }; class RequestData; class HttpPortAllocatorSessionBase : public BasicPortAllocatorSession { public: HttpPortAllocatorSessionBase( HttpPortAllocatorBase* allocator, const std::string& content_name, int component, const std::string& ice_ufrag, const std::string& ice_pwd, const std::vector& stun_hosts, const std::vector& relay_hosts, const std::string& relay, const std::string& agent); virtual ~HttpPortAllocatorSessionBase(); const std::string& relay_token() const { return relay_token_; } const std::string& user_agent() const { return agent_; } virtual void SendSessionRequest(const std::string& host, int port) = 0; virtual void ReceiveSessionResponse(const std::string& response); // Made public for testing. Should be protected. std::string GetSessionRequestUrl(); protected: virtual void GetPortConfigurations(); void TryCreateRelaySession(); virtual HttpPortAllocatorBase* allocator() { return static_cast( BasicPortAllocatorSession::allocator()); } private: std::vector relay_hosts_; std::vector stun_hosts_; std::string relay_token_; std::string agent_; int attempts_; }; class HttpPortAllocator : public HttpPortAllocatorBase { public: HttpPortAllocator(rtc::NetworkManager* network_manager, const std::string& user_agent); HttpPortAllocator(rtc::NetworkManager* network_manager, rtc::PacketSocketFactory* socket_factory, const std::string& user_agent); virtual ~HttpPortAllocator(); virtual PortAllocatorSession* CreateSessionInternal( const std::string& content_name, int component, const std::string& ice_ufrag, const std::string& ice_pwd); }; class HttpPortAllocatorSession : public HttpPortAllocatorSessionBase { public: HttpPortAllocatorSession( HttpPortAllocator* allocator, const std::string& content_name, int component, const std::string& ice_ufrag, const std::string& ice_pwd, const std::vector& stun_hosts, const std::vector& relay_hosts, const std::string& relay, const std::string& agent); virtual ~HttpPortAllocatorSession(); virtual void SendSessionRequest(const std::string& host, int port); protected: // Protected for diagnostics. virtual void OnRequestDone(rtc::SignalThread* request); private: std::list requests_; }; } // namespace cricket #endif // TALK_P2P_CLIENT_HTTPPORTALLOCATOR_H_