OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
websocket_server.hpp
1#pragma once
2#include <ossia/detail/config.hpp>
3
4#include <ossia/detail/json.hpp>
6#include <ossia/network/context.hpp>
7#include <ossia/network/exceptions.hpp>
8#include <ossia/network/sockets/websocket_reply.hpp>
9
10#include <websocketpp/config/asio_no_tls.hpp>
11#include <websocketpp/http/request.hpp>
12#include <websocketpp/server.hpp>
13#if defined(OSSIA_BENCHMARK)
14#include <chrono>
15#endif
16namespace ossia::net
17{
18
21{
22public:
23 using config = websocketpp::config::asio;
24 using transport_type = websocketpp::config::asio::transport_type;
25 using server_t = websocketpp::server<websocketpp::config::asio>;
26 using type = server_t;
27 using connection_handler = websocketpp::connection_hdl;
28
29 typedef typename config::concurrency_type concurrency_type;
30 typedef websocketpp::connection<config> connection_type;
31 typedef typename connection_type::ptr connection_ptr;
32 typedef typename transport_type::transport_con_type transport_con_type;
33 typedef typename transport_con_type::ptr transport_con_ptr;
34 typedef websocketpp::endpoint<connection_type, config> endpoint_type;
35
37 : m_server{std::make_shared<server_t>()}
38 , m_owns_context{true}
39 {
40 m_server->init_asio();
41 m_server->set_reuse_addr(true);
42 m_server->clear_access_channels(websocketpp::log::alevel::all);
43 m_server->set_socket_init_handler(init_handler);
44 }
45
46 websocket_server(ossia::net::network_context_ptr ctx)
47 : m_server{std::make_shared<server_t>()}
48 , m_context{ctx}
49 , m_owns_context{false}
50 {
51 m_server->init_asio(&ctx->context);
52 m_server->set_reuse_addr(true);
53 m_server->clear_access_channels(websocketpp::log::alevel::all);
54 m_server->set_socket_init_handler(init_handler);
55 }
56
57 static void init_handler(websocketpp::connection_hdl, boost::asio::ip::tcp::socket& s)
58 {
59 boost::asio::ip::tcp::no_delay option(true);
60 try
61 {
62 s.set_option(option);
63 }
64 catch(...)
65 {
66 ossia::logger().trace("Could not set TCP nodelay option");
67 }
68 }
69
70 template <typename Handler>
71 void set_open_handler(Handler h)
72 {
73 m_server->set_open_handler(h);
74 }
75
76 template <typename Handler>
77 void set_close_handler(Handler h)
78 {
79 m_server->set_close_handler(h);
80 }
81
82 template <typename Handler>
83 void set_message_handler(Handler h)
84 {
85 m_server->set_message_handler(
86 [this, h](connection_handler hdl, server_t::message_ptr msg) {
87#if defined OSSIA_BENCHMARK
88 auto t1 = std::chrono::high_resolution_clock::now();
89#endif
90 try
91 {
92 auto res = h(hdl, msg->get_opcode(), msg->get_raw_payload());
93 if(res.data.size() > 0)
94 {
95 send_message(hdl, res);
96 }
97 }
98 catch(const ossia::node_not_found_error& e)
99 {
100 auto con = m_server->get_con_from_hdl(hdl);
101 ossia::logger().error(
102 "Node not found: {} ==> {}", con->get_uri()->get_resource(), e.what());
103 }
104 catch(const ossia::bad_request_error& e)
105 {
106 auto con = m_server->get_con_from_hdl(hdl);
107 ossia::logger().error(
108 "Error in request: {} ==> {}", con->get_uri()->get_resource(), e.what());
109 }
110 catch(const std::exception& e)
111 {
112 auto con = m_server->get_con_from_hdl(hdl);
113 ossia::logger().error("Error in request: {}", e.what());
114 }
115 catch(...)
116 {
117 auto con = m_server->get_con_from_hdl(hdl);
118 ossia::logger().error("Error in request");
119 }
120
121#if defined OSSIA_BENCHMARK
122 auto t2 = std::chrono::high_resolution_clock::now();
123 ossia::logger().info(
124 "Time taken: {}",
125 std::chrono::duration_cast<std::chrono::microseconds>(t2 - t1).count());
126#endif
127 });
128
129 m_server->set_http_handler([this, h](connection_handler hdl) {
130 auto con = m_server->get_con_from_hdl(hdl);
131
132 // enable cross origin requests from anywhere
133 con->append_header("Access-Control-Allow-Origin", "*");
134
135 try
136 {
137 ossia::net::server_reply str
138 = h(hdl, websocketpp::frame::opcode::TEXT, con->get_uri()->get_resource());
139
140 switch(str.type)
141 {
142 case server_reply::data_type::json: {
143 con->replace_header("Content-Type", "application/json; charset=utf-8");
144 str.data += "\0";
145 break;
146 }
147 case server_reply::data_type::html: {
148 con->replace_header("Content-Type", "text/html; charset=utf-8");
149 break;
150 }
151 default:
152 break;
153 }
154 con->replace_header("Connection", "close");
155 con->set_body(std::move(str.data));
156 con->set_status(websocketpp::http::status_code::ok);
157 return;
158 }
159 catch(const ossia::node_not_found_error& e)
160 {
161 con->set_status(websocketpp::http::status_code::not_found);
162 }
163 catch(const ossia::bad_request_error& e)
164 {
165 ossia::logger().error(
166 "Error in request: {} ==> {}", con->get_uri()->get_resource(), e.what());
167 con->set_status(websocketpp::http::status_code::bad_request);
168 }
169 catch(const std::exception& e)
170 {
171 ossia::logger().error("Error in request: {}", e.what());
172 }
173 catch(...)
174 {
175 ossia::logger().error("Error in request");
176 }
177 con->set_status(websocketpp::http::status_code::internal_server_error);
178 });
179 }
180
181 void listen(uint16_t port = 9002)
182 {
183 m_server->listen(boost::asio::ip::tcp::v4(), port);
184
185 boost::system::error_code ec;
186 start_accept(ec);
187 }
188
189 void start_accept(boost::system::error_code& ec)
190 {
191 ec = boost::system::error_code();
192 auto con = m_server->get_connection(ec);
193
194 if(!con)
195 {
197 return;
198 }
199
200 namespace lib = websocketpp::lib;
201 m_server->transport_type::async_accept(
202 lib::static_pointer_cast<transport_con_type>(con),
203 [server = m_server, con](lib::error_code ec) {
204 if(server)
205 {
206 if(ec)
207 ec = websocketpp::error::http_connection_ended;
208 server->handle_accept_legacy(con, ec);
209 }
210 }, ec);
211
212 if(ec && con)
213 {
214 // If the connection was constructed but the accept failed,
215 // terminate the connection to prevent memory leaks
216 con->terminate(lib::error_code());
217 }
218 }
219
220 void run() { m_server->run(); }
221
222 void stop()
223 {
224 // this change was undone because of OSSIA/libossia#416 :
225
226 // // (temporarily?) changed to stop_listening()
227 // // "Straight up stop forcibly stops a bunch of things
228 // // in a way that bypasses most, if not all, of the cleanup routines"
229
230 try
231 {
232 boost::system::error_code ec;
233 if(m_server->is_listening())
234 m_server->stop_listening(ec);
235 }
236 catch(...)
237 {
238 }
239
240 if(m_owns_context)
241 {
242 m_server->stop();
243 }
244 else
245 {
246 boost::asio::post(
247 m_context->context, [s = m_server]() mutable { new decltype(s){s}; });
248 }
249 }
250
251 void close(connection_handler hdl)
252 {
253 auto con = m_server->get_con_from_hdl(hdl);
254 con->close(websocketpp::close::status::going_away, "Server shutdown");
255 }
256
257 void send_message(connection_handler hdl, const std::string& message)
258 {
259 auto con = m_server->get_con_from_hdl(hdl);
260 con->send(message);
261 }
262
263 void send_message(connection_handler hdl, const ossia::net::server_reply& message)
264 {
265 auto con = m_server->get_con_from_hdl(hdl);
266 switch(message.type)
267 {
268 case server_reply::data_type::json:
269 case server_reply::data_type::html:
270 con->send(message.data, websocketpp::frame::opcode::TEXT);
271 break;
272 default:
273 con->send(message.data, websocketpp::frame::opcode::BINARY);
274 break;
275 }
276 }
277
278 void send_message(connection_handler hdl, const rapidjson::StringBuffer& message)
279 {
280 auto con = m_server->get_con_from_hdl(hdl);
281 con->send(message.GetString(), message.GetSize(), websocketpp::frame::opcode::text);
282 }
283
284 void send_binary_message(connection_handler hdl, const std::string& message)
285 {
286 auto con = m_server->get_con_from_hdl(hdl);
287 con->send(message.data(), message.size(), websocketpp::frame::opcode::binary);
288 }
289
290 void send_binary_message(connection_handler hdl, std::string_view message)
291 {
292 auto con = m_server->get_con_from_hdl(hdl);
293 con->send(message.data(), message.size(), websocketpp::frame::opcode::binary);
294 }
295
296 server_t& impl() { return *m_server; }
297
298protected:
299 std::shared_ptr<server_t> m_server;
300 ossia::net::network_context_ptr m_context;
301 bool m_owns_context{};
302};
303}
Low-level websocket & http server for oscquery.
Definition websocket_server.hpp:21
void start_accept(boost::system::error_code &ec)
Definition websocket_server.hpp:189
spdlog::logger & logger() noexcept
Where the errors will be logged. Default is stderr.
Definition context.cpp:118
Used when a bad network request is done on a local server.
Definition network/exceptions.hpp:72
Used when a requested node could not be found.
Definition network/exceptions.hpp:60