aron_conversions.h
Go to the documentation of this file.
1 #pragma once
2 
3 #include <algorithm>
4 #include <map>
5 #include <memory>
6 #include <optional>
7 #include <type_traits>
8 #include <vector>
9 
10 namespace armarx::aron
11 {
12 
13  namespace detail
14  {
15 
16  // Helper concept to avoid ambiguities
17  template <typename DtoT, typename BoT>
19 
20  template <class...>
21  struct is_optional : public std::false_type
22  {
23  };
24 
25  template <class... Ts>
26  struct is_optional<std::optional<Ts...>> : public std::true_type
27  {
28  };
29 
30  } // namespace detail
31 
32  /**
33  * Framework for converting ARON DTOs (Data Transfer Objects) to C++ BOs
34  * (Business Objects) and back.
35  *
36  * To allow conversion between custom ARON and C++ types, declare two
37  * functions in the namespace of the BO:
38  *
39  * @code
40  * // aron_conversions.h
41  *
42  * namespace bo_namespace
43  * {
44  * void toAron(arondto::MyObject& dto, const MyObject& bo);
45  * void fromAron(const arondto::MyObject& dto, MyObject& bo);
46  * }
47  * @endcode
48  *
49  * Note that the DTO always comes first, and the target object is
50  * non-const.
51  *
52  * In the implementation,
53  *
54  *
55  * @code
56  * // aron_conversions.cpp
57  *
58  * #include "aron_conversions.h"
59  * #include <Path/to/MyValue/aron_conversions.h>
60  *
61  * void bo_namespace::toAron(arondto::MyObject& dto, const MyObject& bo)
62  * {
63  * dto.name = bo.name;
64  * toAron(dto.myValue, bo.myValue);
65  * }
66  *
67  * void bo_namespace::fromAron(const arondto::MyObject& dto, MyObject& bo)
68  * {
69  * bo.name = dto.name;
70  * fromAron(dto.myValue, bo.myValue);
71  * }
72  * @endcode
73  */
74  // Same type
75  template <class T>
76  void
77  toAron(T& dto, const T& bo)
78  {
79  dto = bo;
80  }
81 
82  template <class T>
83  void
84  fromAron(const T& dto, T& bo)
85  {
86  bo = dto;
87  }
88 
89  // Generic return version
90  template <class DtoT, class BoT>
91  DtoT
92  toAron(const BoT& bo)
93  {
94  DtoT dto;
95  toAron(dto, bo);
96  return dto;
97  }
98 
99  template <class BoT, class DtoT>
100  BoT
101  fromAron(const DtoT& dto)
102  {
103  BoT bo;
104  fromAron(dto, bo);
105  return bo;
106  }
107 
108  // std::unique_ptr
109  template <class DtoT, class BoT>
110  void
111  toAron(DtoT& dto, const std::unique_ptr<BoT>& bo)
112  {
113  if (bo)
114  {
115  toAron(dto, *bo);
116  }
117  }
118 
119  template <class DtoT, class BoT>
120  void
121  fromAron(const DtoT& dto, std::unique_ptr<BoT>& bo)
122  {
123  bo = std::make_unique<BoT>();
124  fromAron(dto, *bo);
125  }
126 
127  // std::optional
128  template <class DtoT, class BoT>
129  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
130  void
131  toAron(std::optional<DtoT>& dto, const std::optional<BoT>& bo)
132  {
133  if (bo.has_value())
134  {
135  dto = DtoT{};
136  toAron(*dto, *bo);
137  }
138  else
139  {
140  dto = std::nullopt;
141  }
142  }
143 
144  template <class DtoT, class BoT>
145  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
146  void
147  fromAron(const std::optional<DtoT>& dto, std::optional<BoT>& bo)
148  {
149  if (dto.has_value())
150  {
151  bo = BoT{};
152  fromAron(*dto, *bo);
153  }
154  else
155  {
156  bo = std::nullopt;
157  }
158  }
159 
160  // One-sided optional
161  template <class DtoT, class BoT>
163  void
164  toAron(std::optional<DtoT>& dto, const BoT& bo)
165  {
166  dto = DtoT{};
167  toAron(*dto, bo);
168  }
169 
170  template <class DtoT, class BoT>
172  void
173  fromAron(DtoT& dto, const std::optional<BoT>& bo)
174  {
175  bo = BoT{};
176  fromAron(dto, *bo);
177  }
178 
179  // Flag-controlled optional
180  template <class DtoT, class BoT>
181  void
182  toAron(DtoT& dto, bool& dtoValid, const BoT& bo, bool boValid)
183  {
184  dtoValid = boValid;
185  if (boValid)
186  {
187  toAron(dto, bo);
188  }
189  else
190  {
191  dto = {};
192  }
193  }
194 
195  template <class DtoT, class BoT>
196  void
197  fromAron(const DtoT& dto, bool dtoValid, BoT& bo, bool& boValid)
198  {
199  boValid = dtoValid;
200  if (dtoValid)
201  {
202  fromAron(dto, bo);
203  }
204  else
205  {
206  bo = {};
207  }
208  }
209 
210  template <class DtoT, class BoT>
211  void
212  toAron(DtoT& dto, bool& dtoValid, const std::optional<BoT>& bo)
213  {
214  dtoValid = bo.has_value();
215  if (dtoValid)
216  {
217  toAron(dto, *bo);
218  }
219  else
220  {
221  dto = {};
222  }
223  }
224 
225  template <class DtoT, class BoT>
226  void
227  fromAron(const DtoT& dto, bool dtoValid, std::optional<BoT>& bo)
228  {
229  if (dtoValid)
230  {
231  bo = BoT{};
232  fromAron(dto, *bo);
233  }
234  else
235  {
236  bo = std::nullopt;
237  }
238  }
239 
240  // std::vector
241  template <class DtoT, class BoT>
242  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
243  void
244  toAron(std::vector<DtoT>& dtos, const std::vector<BoT>& bos)
245  {
246  dtos.clear();
247  dtos.reserve(bos.size());
248  for (const auto& bo : bos)
249  {
250  toAron(dtos.emplace_back(), bo);
251  }
252  }
253 
254  template <class DtoT, class BoT>
255  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
256  void
257  fromAron(const std::vector<DtoT>& dtos, std::vector<BoT>& bos)
258  {
259  bos.clear();
260  bos.reserve(dtos.size());
261  for (const auto& dto : dtos)
262  {
263  fromAron(dto, bos.emplace_back());
264  }
265  }
266 
267  // std::map
268  template <class DtoKeyT, class DtoValueT, class BoKeyT, class BoValueT>
269  requires(!(detail::DtoAndBoAreSame<DtoKeyT, BoKeyT> and
270  detail::DtoAndBoAreSame<DtoValueT, BoValueT>))
271  void
272  toAron(std::map<DtoKeyT, DtoValueT>& dtoMap, const std::map<BoKeyT, BoValueT>& boMap)
273  {
274  dtoMap.clear();
275  for (const auto& [boKey, boValue] : boMap)
276  {
277  DtoKeyT dtoKey;
278  toAron(dtoKey, boKey);
279  auto [it, _] = dtoMap.emplace(std::move(dtoKey), DtoValueT{});
280  toAron(it->second, boValue);
281  }
282  }
283 
284  template <class DtoKeyT, class DtoValueT, class BoKeyT, class BoValueT>
285  requires(!(detail::DtoAndBoAreSame<DtoKeyT, BoKeyT> and
286  detail::DtoAndBoAreSame<DtoValueT, BoValueT>))
287  void
288  fromAron(const std::map<DtoKeyT, DtoValueT>& dtoMap, std::map<BoKeyT, BoValueT>& boMap)
289  {
290  boMap.clear();
291  for (const auto& [dtoKey, dtoValue] : dtoMap)
292  {
293  BoKeyT boKey;
294  fromAron(dtoKey, boKey);
295  auto [it, _] = boMap.emplace(boKey, BoValueT{});
296  fromAron(dtoValue, it->second);
297  }
298  }
299 
300  template <class DtoKeyT, class DtoValueT, class BoKeyT, class BoValueT>
301  requires(!(detail::DtoAndBoAreSame<DtoKeyT, BoKeyT> and
302  detail::DtoAndBoAreSame<DtoValueT, BoValueT>))
303  std::map<DtoKeyT, DtoValueT>
304  toAron(const std::map<BoKeyT, BoValueT>& boMap)
305  {
306  std::map<DtoKeyT, DtoValueT> dtoMap;
307  toAron(dtoMap, boMap);
308  return dtoMap;
309  }
310 } // namespace armarx::aron
311 
312 // And do the same for the armarx namespace to ensure consistency with all the other aron_conversions declaraions
313 // (which are usually in the armarx namespace)
314 namespace armarx
315 {
316  // Same type
317  template <class T>
318  void
319  toAron(T& dto, const T& bo)
320  {
321  armarx::aron::toAron(dto, bo);
322  }
323 
324  template <class T>
325  void
326  fromAron(const T& dto, T& bo)
327  {
329  }
330 
331  // Generic return version
332  template <class DtoT, class BoT>
333  DtoT
334  toAron(const BoT& bo)
335  {
336  return armarx::aron::toAron<DtoT, BoT>(bo);
337  }
338 
339  template <class BoT, class DtoT>
340  BoT
341  fromAron(const DtoT& dto)
342  {
343  return armarx::aron::fromAron<BoT, DtoT>(dto);
344  }
345 
346  // std::unique_ptr
347  template <class DtoT, class BoT>
348  void
349  toAron(DtoT& dto, const std::unique_ptr<BoT>& bo)
350  {
351  armarx::aron::toAron(dto, bo);
352  }
353 
354  template <class DtoT, class BoT>
355  void
356  fromAron(const DtoT& dto, std::unique_ptr<BoT>& bo)
357  {
359  }
360 
361  // std::optional
362  template <class DtoT, class BoT>
363  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
364  void
365  toAron(std::optional<DtoT>& dto, const std::optional<BoT>& bo)
366  {
367  armarx::aron::toAron(dto, bo);
368  }
369 
370  template <class DtoT, class BoT>
371  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
372  void
373  fromAron(const std::optional<DtoT>& dto, std::optional<BoT>& bo)
374  {
376  }
377 
378  // Flag-controlled optional
379  template <class DtoT, class BoT>
380  void
381  toAron(DtoT& dto, bool& dtoValid, const BoT& bo, bool boValid)
382  {
383  armarx::aron::toAron(dto, dtoValid, bo, boValid);
384  }
385 
386  template <class DtoT, class BoT>
387  void
388  fromAron(const DtoT& dto, bool dtoValid, BoT& bo, bool& boValid)
389  {
390  armarx::aron::fromAron(dto, dtoValid, bo, boValid);
391  }
392 
393  template <class DtoT, class BoT>
394  void
395  toAron(DtoT& dto, bool& dtoValid, const std::optional<BoT>& bo)
396  {
397  armarx::aron::toAron(dto, dtoValid, bo);
398  }
399 
400  template <class DtoT, class BoT>
401  void
402  fromAron(const DtoT& dto, bool dtoValid, std::optional<BoT>& bo)
403  {
404  armarx::aron::fromAron(dto, dtoValid, bo);
405  }
406 
407  // std::vector
408  template <class DtoT, class BoT>
409  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
410  void
411  toAron(std::vector<DtoT>& dtos, const std::vector<BoT>& bos)
412  {
413  armarx::aron::toAron(dtos, bos);
414  }
415 
416  template <class DtoT, class BoT>
417  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
418  void
419  fromAron(const std::vector<DtoT>& dtos, std::vector<BoT>& bos)
420  {
421  armarx::aron::fromAron(dtos, bos);
422  }
423 
424  template <class DtoT, class BoT>
425  requires(!aron::detail::DtoAndBoAreSame<DtoT, BoT>)
426  std::vector<DtoT>
427  toAron(const std::vector<BoT>& bos)
428  {
429  return armarx::aron::toAron<DtoT, BoT>(bos);
430  }
431 
432  // std::map
433  template <class DtoKeyT, class DtoValueT, class BoKeyT, class BoValueT>
434  requires(!(aron::detail::DtoAndBoAreSame<DtoKeyT, BoKeyT> and
435  aron::detail::DtoAndBoAreSame<DtoValueT, BoValueT>))
436  void
437  toAron(std::map<DtoKeyT, DtoValueT>& dtoMap, const std::map<BoKeyT, BoValueT>& boMap)
438  {
439  armarx::aron::toAron(dtoMap, boMap);
440  }
441 
442  template <class DtoKeyT, class DtoValueT, class BoKeyT, class BoValueT>
443  requires(!(aron::detail::DtoAndBoAreSame<DtoKeyT, BoKeyT> and
444  aron::detail::DtoAndBoAreSame<DtoValueT, BoValueT>))
445  void
446  fromAron(const std::map<DtoKeyT, DtoValueT>& dtoMap, std::map<BoKeyT, BoValueT>& boMap)
447  {
448  armarx::aron::fromAron(dtoMap, boMap);
449  }
450 
451  template <class DtoKeyT, class DtoValueT, class BoKeyT, class BoValueT>
452  requires(!(aron::detail::DtoAndBoAreSame<DtoKeyT, BoKeyT> and
453  aron::detail::DtoAndBoAreSame<DtoValueT, BoValueT>))
454  std::map<DtoKeyT, DtoValueT>
455  toAron(const std::map<BoKeyT, BoValueT>& boMap)
456  {
457  armarx::aron::toAron<DtoKeyT, DtoValueT, BoKeyT, BoValueT>(boMap);
458  }
459 
460  /**
461  * @brief Converter function for vector of aron elements to plain cpp type
462  *
463  * You have to provide a converter function for the element with the signature
464  *
465  * PlainCppType fromAron(const AronType&)
466  *
467  * @tparam T the aron vector element
468  * @param v the vector of elements
469  * @return the vector of aron elements
470  */
471  template <typename T>
472  auto
473  fromAron(const std::vector<T>& v) -> std::vector<decltype(fromAron(T()))>
474  {
475  std::vector<decltype(fromAron(T()))> r;
476  r.reserve(v.size());
477 
479  v.begin(), v.end(), std::back_inserter(r), [](const T& t) { return fromAron(t); });
480 
481  return r;
482  }
483 
484  template <typename T>
485  auto
486  toAron(const std::vector<T>& v)
487  {
488  std::vector<decltype(toAron(T()))> r;
489  r.reserve(v.size());
490 
492  v.begin(), v.end(), std::back_inserter(r), [](const T& t) { return toAron(t); });
493 
494  return r;
495  }
496 } // namespace armarx
armarx::requires
requires(!aron::detail::DtoAndBoAreSame< DtoT, BoT >) void toAron(std
Definition: aron_conversions.h:363
detail
Definition: OpenCVUtil.cpp:128
cxxopts::value
std::shared_ptr< Value > value()
Definition: cxxopts.hpp:855
armarx::aron::detail::is_optional
Definition: aron_conversions.h:21
armarx::aron
Definition: DataDisplayVisitor.cpp:5
armarx::aron::requires
requires(!aron::detail::DtoAndBoAreSame< DtoT, BoT >) void toAron(std
Definition: aron_conversions.h:129
armarx::aron::detail::DtoAndBoAreSame
concept DtoAndBoAreSame
Definition: aron_conversions.h:18
armarx::transform
auto transform(const Container< InputT, Alloc > &in, OutputT(*func)(InputT const &)) -> Container< OutputT, typename std::allocator_traits< Alloc >::template rebind_alloc< OutputT >>
Convenience function (with less typing) to transform a container of type InputT into the same contain...
Definition: algorithm.h:351
armarx::ctrlutil::v
double v(double t, double v0, double a0, double j)
Definition: CtrlUtil.h:39
armarx::aron::toAron
void toAron(T &dto, const T &bo)
Framework for converting ARON DTOs (Data Transfer Objects) to C++ BOs (Business Objects) and back.
Definition: aron_conversions.h:77
std
Definition: Application.h:66
armarx::fromAron
void fromAron(const arondto::PackagePath &dto, PackageFileLocation &bo)
armarx::toAron
void toAron(arondto::PackagePath &dto, const PackageFileLocation &bo)
T
float T
Definition: UnscentedKalmanFilterTest.cpp:38
armarx::aron::fromAron
void fromAron(const T &dto, T &bo)
Definition: aron_conversions.h:84
armarx
This file offers overloads of toIce() and fromIce() functions for STL container types.
Definition: ArmarXTimeserver.cpp:27
armarx::aron::bo
const std::optional< BoT > & bo
Definition: aron_conversions.h:174