OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
osc/detail/bundle.hpp
1#pragma once
2#include <ossia/detail/buffer_pool.hpp>
4#include <ossia/network/base/bundle.hpp>
5#include <ossia/network/base/osc_address.hpp>
6#include <ossia/network/osc/detail/osc_1_0_policy.hpp>
7#include <ossia/network/osc/detail/osc_messages.hpp>
8#include <ossia/network/value/value.hpp>
9
10#include <oscpack/osc/OscOutboundPacketStream.h>
11
12#include <optional>
13
14namespace ossia::net
15{
16
17// Default case
18template <typename OscPolicy>
19struct bundle_common_policy
20{
21 template <typename Addr_T>
22 void
23 operator()(oscpack::OutboundPacketStream& str, ossia::value& val, const Addr_T& addr)
24 {
25 if(val = bound_value(addr, val); val.valid())
26 {
27 str << oscpack::BeginMessageN(osc_address(addr));
28 val.apply(typename OscPolicy::dynamic_policy{{str, addr.get_unit()}});
29 str << oscpack::EndMessage();
30 }
31 }
32};
33
34template <typename OscPolicy>
35struct bundle_client_policy
36{
37 template <typename Addr_T>
38 void
39 operator()(oscpack::OutboundPacketStream& str, ossia::value& val, const Addr_T& addr)
40 {
41 if(addr.get_access() == ossia::access_mode::GET)
42 return;
43
44 bundle_common_policy<OscPolicy>{}(str, val, addr);
45 }
46};
47
48template <typename OscPolicy>
49using bundle_server_policy = bundle_common_policy<OscPolicy>;
50
51// Pre-bounded case
52template <typename OscPolicy>
53struct bundle_bounded_common_policy
54{
55 template <typename Addr_T>
56 void
57 operator()(oscpack::OutboundPacketStream& str, ossia::value& val, const Addr_T& addr)
58 {
59 if(val.valid())
60 {
61 str << oscpack::BeginMessageN(osc_address(addr));
62 val.apply(typename OscPolicy::dynamic_policy{{str, addr.get_unit()}});
63 str << oscpack::EndMessage();
64 }
65 }
66};
67
68template <typename OscPolicy>
69struct bundle_bounded_client_policy
70{
71 template <typename Addr_T>
72 void
73 operator()(oscpack::OutboundPacketStream& str, ossia::value& val, const Addr_T& addr)
74 {
75 if(addr.get_access() == ossia::access_mode::GET)
76 return;
77
78 bundle_bounded_common_policy<OscPolicy>{}(str, val, addr);
79 }
80};
81
82template <typename OscPolicy>
83using bundle_bounded_server_policy = bundle_bounded_common_policy<OscPolicy>;
84
85static inline auto& access_parameter(const ossia::net::parameter_base* p)
86{
87 return *p;
88}
89static inline auto& access_parameter(const ossia::net::full_parameter_data& p)
90{
91 return p;
92}
93static inline auto& access_parameter(const ossia::bundle_element& p)
94{
95 return *p.parameter;
96}
97
98struct bundle
99{
100 ossia::buffer_pool::buffer data;
101 bool critical{};
102};
103
104template <typename NetworkPolicy, typename Addresses>
105std::optional<bundle>
106make_bundle(NetworkPolicy add_element_to_bundle, const Addresses& addresses)
107try
108{
109 bundle ret{ossia::buffer_pool::instance().acquire(max_osc_message_size), false};
110 {
111 oscpack::OutboundPacketStream str(ret.data.data(), max_osc_message_size);
112 str << oscpack::BeginBundleImmediate();
113
114 ossia::value val;
115 for(const auto& a : addresses)
116 {
117 auto& param = access_parameter(a);
118 ret.critical |= param.get_critical();
119 val = param.value();
120 add_element_to_bundle(str, val, param);
121 }
122 str << oscpack::EndBundle();
123 ret.data.resize(str.Size());
124
125 // TODO useless condition for now.
126 // But if we know that we are going through ws we can increase the size
127 // beyond 65k. ret.critical |= str.Size() > max_osc_message_size;
128 }
129 return ret;
130}
131catch(const oscpack::OutOfBufferMemoryException&)
132{
133 ossia::logger().error(
134 "make_bundle_client: message too large (limit is {} bytes)", max_osc_message_size);
135 return {};
136}
137catch(const std::runtime_error& e)
138{
139 ossia::logger().error("make_bundle_client: {}", e.what());
140 return {};
141}
142catch(...)
143{
144 ossia::logger().error("make_bundle_client: unknown error");
145 return {};
146}
147
148template <typename NetworkPolicy>
149std::optional<bundle> make_bundle(
150 NetworkPolicy add_element_to_bundle, const ossia::net::full_parameter_data& param)
151try
152{
153 bundle ret{
154 ossia::buffer_pool::instance().acquire(max_osc_message_size), param.critical};
155 {
156 oscpack::OutboundPacketStream str(ret.data.data(), max_osc_message_size);
157 str << oscpack::BeginBundleImmediate();
158 auto val = param.value();
159 add_element_to_bundle(str, val, param);
160 str << oscpack::EndBundle();
161 ret.data.resize(str.Size());
162
163 // TODO useless condition for now.
164 // But if we know that we are going through ws we can increase the size
165 // beyond 65k. ret.critical |= str.Size() > max_osc_message_size;
166 }
167 return ret;
168}
169catch(const oscpack::OutOfBufferMemoryException&)
170{
171 ossia::logger().error(
172 "make_bundle_client: message too large (limit is {} bytes)", max_osc_message_size);
173 return {};
174}
175catch(const std::runtime_error& e)
176{
177 ossia::logger().error("make_bundle_client: {}", e.what());
178 return {};
179}
180catch(...)
181{
182 ossia::logger().error("make_bundle_client: unknown error");
183 return {};
184}
185
186template <typename NetworkPolicy>
187std::optional<bundle> make_bundle(
188 NetworkPolicy add_element_to_bundle, const ossia::net::parameter_base& param,
189 ossia::value& v)
190try
191{
192 bundle ret{
193 ossia::buffer_pool::instance().acquire(max_osc_message_size),
194 param.get_critical()};
195 {
196 oscpack::OutboundPacketStream str(ret.data.data(), max_osc_message_size);
197 str << oscpack::BeginBundleImmediate();
198 add_element_to_bundle(str, v, param);
199 str << oscpack::EndBundle();
200 ret.data.resize(str.Size());
201
202 // TODO useless condition for now.
203 // But if we know that we are going through ws we can increase the size
204 // beyond 65k. ret.critical |= str.Size() > max_osc_message_size;
205 }
206 return ret;
207}
208catch(const oscpack::OutOfBufferMemoryException&)
209{
210 ossia::logger().error(
211 "make_bundle_client: message too large (limit is {} bytes)", max_osc_message_size);
212 return {};
213}
214catch(const std::runtime_error& e)
215{
216 ossia::logger().error("make_bundle_client: {}", e.what());
217 return {};
218}
219catch(...)
220{
221 ossia::logger().error("make_bundle_client: unknown error");
222 return {};
223}
224
225template <typename NetworkPolicy>
226std::optional<bundle> make_bundle(
227 NetworkPolicy add_element_to_bundle,
228 const tcb::span<ossia::bundle_element>& addresses)
229try
230{
231 bundle ret{ossia::buffer_pool::instance().acquire(max_osc_message_size), false};
232 {
233 oscpack::OutboundPacketStream str(ret.data.data(), max_osc_message_size);
234 str << oscpack::BeginBundleImmediate();
235
236 for(auto& [p, v] : addresses)
237 {
238 auto& param = *p;
239 ret.critical |= param.get_critical();
240 add_element_to_bundle(str, v, param);
241 }
242 str << oscpack::EndBundle();
243 ret.data.resize(str.Size());
244
245 // TODO useless condition for now.
246 // But if we know that we are going through ws we can increase the size
247 // beyond 65k. ret.critical |= str.Size() > max_osc_message_size;
248 }
249 return ret;
250}
251catch(const oscpack::OutOfBufferMemoryException&)
252{
253 ossia::logger().error(
254 "make_bundle_client: message too large (limit is {} bytes)", max_osc_message_size);
255 return {};
256}
257catch(const std::runtime_error& e)
258{
259 ossia::logger().error("make_bundle_client: {}", e.what());
260 return {};
261}
262catch(...)
263{
264 ossia::logger().error("make_bundle_client: unknown error");
265 return {};
266}
267
268template <typename NetworkPolicy>
269bool make_bundle_bounded(
270 NetworkPolicy add_element_to_bundle,
271 const tcb::span<ossia::bundle_element>& addresses, auto callback)
272{
273 bundle ret{ossia::buffer_pool::instance().acquire(max_osc_message_size), false};
274 try
275 {
276 // FIXME iterate until the size is enough
277 oscpack::OutboundPacketStream str(ret.data.data(), max_osc_message_size);
278 str << oscpack::BeginBundleImmediate();
279
280 for(auto& [p, v] : addresses)
281 {
282 auto& param = *p;
283 ret.critical |= param.get_critical();
284 add_element_to_bundle(str, v, param);
285 }
286 str << oscpack::EndBundle();
287 ret.data.resize(str.Size());
288
289 callback(ret);
290
291 // TODO useless condition for now.
292 // But if we know that we are going through ws we can increase the size
293 // beyond 65k. ret.critical |= str.Size() > max_osc_message_size;
294 }
295 catch(const oscpack::OutOfBufferMemoryException&)
296 {
297 ossia::logger().error(
298 "make_bundle_client: message too large (limit is {} bytes)",
299 max_osc_message_size);
300 }
301 catch(const std::runtime_error& e)
302 {
303 ossia::logger().error("make_bundle_client: {}", e.what());
304 }
305 catch(...)
306 {
307 ossia::logger().error("make_bundle_client: unknown error");
308 }
309 ossia::buffer_pool::instance().release(std::move(ret.data));
310 return true;
311}
312}
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
The value class.
Definition value.hpp:173
spdlog::logger & logger() noexcept
Where the errors will be logged. Default is stderr.
Definition context.cpp:118
@ GET
The value can be retrieved and changed.
bool critical
Means that the node is very important, e.g. a "play" message.
Definition node_attributes.hpp:92
Full information about a parameter.
Definition parameter_data.hpp:61