mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-20 11:05:54 +00:00
159 lines
5.6 KiB
C++
159 lines
5.6 KiB
C++
/*
|
|
* 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 <cstring>
|
|
#include <set>
|
|
|
|
#include <boost/functional.hpp>
|
|
#include <boost/bind.hpp>
|
|
|
|
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<bool(const std::string&)> 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<type> ptr;
|
|
|
|
telemetry_server_handler() : m_done(false),m_value(0) {
|
|
boost::function<bool(const std::string&)> 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<boost::mutex> 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<boost::mutex> guard(m_mutex);
|
|
|
|
std::set<connection_ptr>::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<boost::mutex> guard(m_mutex);
|
|
m_connections.insert(con);
|
|
}
|
|
|
|
// remove an exiting client
|
|
void on_close(connection_ptr con) {
|
|
boost::lock_guard<boost::mutex> guard(m_mutex);
|
|
m_connections.erase(con);
|
|
}
|
|
private:
|
|
bool m_done;
|
|
size_t m_value;
|
|
std::set<connection_ptr> m_connections;
|
|
|
|
boost::mutex m_mutex; // guards m_connections
|
|
boost::shared_ptr<boost::thread> 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;
|
|
}
|