OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
unix_socket.hpp
1#pragma once
3#include <ossia/network/context.hpp>
4#include <ossia/network/sockets/configuration.hpp>
5
6#include <boost/asio/io_context.hpp>
7#include <boost/asio/ip/udp.hpp>
8#include <boost/asio/local/datagram_protocol.hpp>
9#include <boost/asio/local/stream_protocol.hpp>
10#include <boost/asio/placeholders.hpp>
11#include <boost/asio/strand.hpp>
12#include <boost/asio/write.hpp>
13
14#include <nano_signal_slot.hpp>
15
16namespace ossia::net
17{
18#if defined(BOOST_ASIO_HAS_LOCAL_SOCKETS)
19class unix_datagram_socket
20{
21 using proto = boost::asio::local::datagram_protocol;
22
23public:
24 unix_datagram_socket(const fd_configuration& conf, boost::asio::io_context& ctx)
25 : m_context{ctx}
26 , m_endpoint{conf.fd}
27 , m_socket{boost::asio::make_strand(ctx)}
28 {
29 }
30
31 void open()
32 {
33 ::unlink(m_endpoint.path().data());
34 m_socket.open();
35 m_socket.bind(m_endpoint);
36 }
37
38 void connect()
39 {
40 m_socket.open();
41 // m_socket.connect(m_endpoint);
42 }
43
44 void close()
45 {
46 if(m_socket.is_open())
47 {
48 boost::asio::post(m_context, [this] {
49 m_socket.close();
50 on_close();
51 });
52 }
53 }
54
55 template <typename F>
56 void receive(F f)
57 {
58 m_socket.async_receive_from(
59 boost::asio::mutable_buffer(&m_data[0], std::size(m_data)), m_endpoint,
60 [this, f](boost::system::error_code ec, std::size_t sz) {
61 if(ec == boost::asio::error::operation_aborted)
62 return;
63
64 if(!ec && sz > 0)
65 try
66 {
67 f(m_data, sz);
68 }
69 catch(...)
70 {
71 }
72
73 this->receive(f);
74 });
75 }
76
77 void write(const char* data, std::size_t sz)
78 {
79 m_socket.send_to(boost::asio::buffer(data, sz), m_endpoint);
80 }
81
82 Nano::Signal<void()> on_close;
83
84 boost::asio::io_context& m_context;
85 proto::endpoint m_endpoint;
86 proto::socket m_socket;
87 alignas(16) char m_data[65535];
88};
89
90class unix_stream_listener
91{
92public:
93 using proto = boost::asio::local::stream_protocol;
94 unix_stream_listener() = delete;
95 unix_stream_listener(const unix_stream_listener&) = delete;
96 unix_stream_listener& operator=(const unix_stream_listener&) = delete;
97 unix_stream_listener(unix_stream_listener&&) = default;
98 unix_stream_listener& operator=(unix_stream_listener&&) = default;
99 explicit unix_stream_listener(proto::socket sock)
100 : m_socket{std::move(sock)}
101 {
102 }
103
104 void close() { m_socket.close(); }
105
106 void write(const boost::asio::const_buffer& buf) { boost::asio::write(m_socket, buf); }
107
108 void on_close() { }
109
110 void on_fail() { }
111
112 proto::socket m_socket;
113};
114
115class unix_stream_server
116{
117public:
118 using proto = boost::asio::local::stream_protocol;
119 using listener = unix_stream_listener;
120 [[no_unique_address]] struct ensure_reuse
121 {
122 explicit ensure_reuse(const proto::endpoint& endpoint)
123 {
124 ::unlink(endpoint.path().data());
125 }
126 } m_ensure_reuse;
127
128 unix_stream_server(const fd_configuration& conf, boost::asio::io_context& ctx)
129 : m_ensure_reuse{conf.fd}
130 , m_context{ctx}
131 , m_acceptor{boost::asio::make_strand(ctx), conf.fd}
132 {
133 }
134
135 unix_stream_server(const fd_configuration& conf, ossia::net::network_context_ptr ctx)
136 : unix_stream_server{conf, ctx->context}
137 {
138 }
139
140 boost::asio::io_context& m_context;
141 proto::acceptor m_acceptor;
142};
143
144class unix_stream_client
145{
146public:
147 using proto = boost::asio::local::stream_protocol;
148 using socket = typename proto::socket;
149
150 unix_stream_client(const fd_configuration& conf, boost::asio::io_context& ctx)
151 : m_context{ctx}
152 , m_endpoint{conf.fd}
153 , m_socket{boost::asio::make_strand(ctx)}
154 {
155 }
156
157 void connect()
158 {
159 m_socket.connect(m_endpoint);
160 on_open();
161 }
162
163 bool connected() const { return m_connected; }
164
165 void close()
166 {
167 boost::asio::post(m_context, [this] {
168 m_socket.close();
169 on_close();
170 });
171 }
172
173 void write(const char* data, std::size_t sz)
174 {
175 boost::asio::write(m_socket, boost::asio::buffer(data, sz));
176 }
177
178 Nano::Signal<void()> on_open;
179 Nano::Signal<void()> on_close;
180 Nano::Signal<void()> on_fail;
181
182 boost::asio::io_context& m_context;
183 proto::endpoint m_endpoint;
184 proto::socket m_socket;
185 bool m_connected{false};
186};
187#endif
188}