/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. */ // Original author: ekr@rtfm.com #ifndef stunserver_h__ #define stunserver_h__ #include <map> #include <string> #include "prio.h" #include "nsError.h" #include "mozilla/UniquePtr.h" typedef struct nr_stun_server_ctx_ nr_stun_server_ctx; typedef struct nr_socket_ nr_socket; typedef struct nr_local_addr_ nr_local_addr; namespace mozilla { class TestStunServer { public: // Generally, you should only call API in this class from the same thread that // the initial |GetInstance| call was made from. static TestStunServer *GetInstance(int address_family = AF_INET); static void ShutdownInstance(); // |ConfigurePort| will only have an effect if called before the first call // to |GetInstance| (possibly following a |ShutdownInstance| call) static void ConfigurePort(uint16_t port); // AF_INET, AF_INET6 static UniquePtr<TestStunServer> Create(int address_family); virtual ~TestStunServer(); void SetActive(bool active); void SetDelay(uint32_t delay_ms); void SetDropInitialPackets(uint32_t count); const std::string& addr() const { return listen_addr_; } uint16_t port() const { return listen_port_; } // These should only be called from the same thread as the initial // |GetInstance| call. nsresult SetResponseAddr(nr_transport_addr *addr); nsresult SetResponseAddr(const std::string& addr, uint16_t port); void Reset(); static const size_t max_stun_message_size = 4096; virtual nr_socket* GetReceivingSocket(NR_SOCKET s); virtual nr_socket* GetSendingSocket(nr_socket *sock); protected: TestStunServer() : listen_port_(0), listen_sock_(nullptr), send_sock_(nullptr), stun_server_(nullptr), active_(true), delay_ms_(0), initial_ct_(0), response_addr_(nullptr), timer_handle_(nullptr) {} int SetInternalPort(nr_local_addr *addr, uint16_t port); int Initialize(int address_family); static void readable_cb(NR_SOCKET sock, int how, void *cb_arg); private: void Process(const uint8_t *msg, size_t len, nr_transport_addr *addr_in, nr_socket *sock); virtual int TryOpenListenSocket(nr_local_addr *addr, uint16_t port); static void process_cb(NR_SOCKET sock, int how, void *cb_arg); protected: std::string listen_addr_; uint16_t listen_port_; nr_socket *listen_sock_; nr_socket *send_sock_; nr_stun_server_ctx *stun_server_; private: bool active_; uint32_t delay_ms_; uint32_t initial_ct_; nr_transport_addr *response_addr_; void *timer_handle_; std::map<std::string, uint32_t> received_ct_; static TestStunServer *instance; static TestStunServer *instance6; static uint16_t instance_port; }; class TestStunTcpServer: public TestStunServer { public: static TestStunTcpServer *GetInstance(int address_family); static void ShutdownInstance(); static void ConfigurePort(uint16_t port); virtual ~TestStunTcpServer(); virtual nr_socket* GetReceivingSocket(NR_SOCKET s); virtual nr_socket* GetSendingSocket(nr_socket *sock); protected: TestStunTcpServer() {} static void accept_cb(NR_SOCKET sock, int how, void *cb_arg); private: virtual int TryOpenListenSocket(nr_local_addr *addr, uint16_t port); static UniquePtr<TestStunTcpServer> Create(int address_family); static TestStunTcpServer *instance; static TestStunTcpServer *instance6; static uint16_t instance_port; std::map<NR_SOCKET, nr_socket*> connections_; }; } // End of namespace mozilla #endif