/* * Copyright (c) 2012, Peter Thorson. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 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. * * Neither the name of the WebSocket++ Project nor the * names of its contributors may be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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 PETER THORSON 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. * */ #include "../../src/websocketpp.hpp" #include #include #include #include using websocketpp::server; /// Thread body. Counts up indefinitely, one increment per second. After each /// it calls the handler back asking it to broadcast the new value. The handler /// callback returns whether or not the handler would like the telemetry thread /// to stop. If callback returns true the telemetry loop ands end the thread /// exits. void generate_telemetry(boost::function callback) { size_t value = 0; for (;;) { // do some work value++; // broadcast state std::stringstream m; m << value; if (callback(m.str())) { break; } // wait boost::this_thread::sleep(boost::posix_time::milliseconds(1000)); } } class telemetry_server_handler : public server::handler { public: typedef telemetry_server_handler type; typedef boost::shared_ptr ptr; telemetry_server_handler() : m_done(false),m_value(0) { boost::function callback = boost::bind(&type::on_tick,this,_1); // start a thread that will generate telemetry independently and call // this handler back when it has new data to send. m_telemetry_thread.reset(new boost::thread( boost::bind( &generate_telemetry, callback ) )); } // If the handler is going away set done to true and wait for the thread // to exit. ~telemetry_server_handler() { { boost::lock_guard guard(m_mutex); m_done = true; } m_telemetry_thread->join(); } /// Function that we pass to the telemetry thread to broadcast the new /// state. It returns the global "are we done" value so we can control when /// the thread stops running. bool on_tick(const std::string& msg) { boost::lock_guard guard(m_mutex); std::set::iterator it; for (it = m_connections.begin(); it != m_connections.end(); it++) { (*it)->send(msg); } return m_done; } // register a new client void on_open(connection_ptr con) { boost::lock_guard guard(m_mutex); m_connections.insert(con); } // remove an exiting client void on_close(connection_ptr con) { boost::lock_guard guard(m_mutex); m_connections.erase(con); } private: bool m_done; size_t m_value; std::set m_connections; boost::mutex m_mutex; // guards m_connections boost::shared_ptr m_telemetry_thread; }; int main(int argc, char* argv[]) { unsigned short port = 9007; if (argc == 2) { port = atoi(argv[1]); if (port == 0) { std::cout << "Unable to parse port input " << argv[1] << std::endl; return 1; } } try { server::handler::ptr handler(new telemetry_server_handler()); server endpoint(handler); endpoint.alog().unset_level(websocketpp::log::alevel::ALL); endpoint.elog().unset_level(websocketpp::log::elevel::ALL); endpoint.alog().set_level(websocketpp::log::alevel::CONNECT); endpoint.alog().set_level(websocketpp::log::alevel::DISCONNECT); endpoint.elog().set_level(websocketpp::log::elevel::RERROR); endpoint.elog().set_level(websocketpp::log::elevel::FATAL); std::cout << "Starting WebSocket telemetry server on port " << port << std::endl; endpoint.listen(port); } catch (std::exception& e) { std::cerr << "Exception: " << e.what() << std::endl; } return 0; }