TAMSVIZ
Visualization and annotation tool for ROS
serialization.h
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #pragma once
5 
6 #include "object.h"
7 #include "variant.h"
8 
9 #include <map>
10 
11 class SerializationTypeError : public std::runtime_error {
12 protected:
13  std::string _type_name;
14  SerializationTypeError(const std::string &type_name);
15 
16 public:
17  const std::string &typeName() const { return _type_name; }
18  virtual void createReplacement() const = 0;
19 };
20 
21 template <class T> struct SerializationTypeReplacement : T {};
22 
23 template <class T>
25 
26 public:
27  SerializationTypeErrorImpl(const std::string &type_name)
28  : SerializationTypeError(type_name) {}
29  virtual void createReplacement() const {
30  Type::global<SerializationTypeReplacement<T>>(typeName(), Type::find<T>());
31  }
32 };
33 
34 template <class T>
35 auto serialize(const T &v) ->
36  typename std::enable_if<toStringSupported((T *)nullptr), Variant>::type {
37  std::string ret;
38  toString(v, ret);
39  return Variant(ret);
40 }
41 template <class T>
42 auto deserialize(T &v, const Variant &r) ->
43  typename std::enable_if<fromStringSupported((T *)nullptr), void>::type {
44  fromString(v, r.value<std::string>());
45 }
46 
47 template <class T>
48 auto serialize(const T &v) ->
49  typename std::enable_if<propertiesSupported((T *)nullptr), Variant>::type {
50  std::map<std::string, Variant> r;
51  std::vector<Property> pp;
52  properties(*(T *)&v, pp);
53  for (auto &property : pp) {
54  r[property.displayName()] = property.serialize();
55  }
56  return Variant(r);
57 }
58 template <class T>
59 auto deserialize(T &value, const Variant &variant) ->
60  typename std::enable_if<propertiesSupported((T *)nullptr), void>::type {
61  auto map = variant.value<std::map<std::string, Variant>>();
62  std::vector<Property> pp;
63  properties(*(T *)&value, pp);
64  for (auto &property : pp) {
65  if (map.find(property.name()) != map.end()) {
66  property.deserialize(map[property.name()]);
67  } else if (map.find(property.displayName()) != map.end()) {
68  property.deserialize(map[property.displayName()]);
69  }
70  }
71 }
72 
73 template <class T> Variant serialize(const std::vector<T> &v) {
74  std::vector<Variant> r;
75  for (auto &x : v) {
76  r.push_back(serialize(x));
77  }
78  return Variant(r);
79 }
80 template <class T> void deserialize(std::vector<T> &v, const Variant &x) {
81  auto &r = x.value<std::vector<Variant>>();
82  v.resize(r.size());
83  for (size_t i = 0; i < v.size(); i++) {
84  deserialize(v[i], r[i]);
85  }
86 }
87 
88 template <class T> Variant serialize(const Handle<T> &v) {
89  return Variant(std::to_string(v.id()));
90 }
91 template <class T> void deserialize(Handle<T> &handle, const Variant &variant) {
92  handle.reset(std::stoul(variant.value<std::string>()));
93 }
94 
95 template <class T>
96 static std::shared_ptr<const Type>
97 findSerializationType(const std::shared_ptr<T> &p) {
98  if (p == nullptr || typeid(*p) == typeid(T)) {
99  return nullptr;
100  }
101  return Type::find(typeid(*p));
102 }
103 template <class T> Variant serialize(const std::shared_ptr<T> &v) {
104  if (v) {
105  std::map<std::string, Variant> r;
106  for (auto &property : v->properties()) {
107  r[property.name()] = property.serialize();
108  }
109  if (auto type = findSerializationType(v)) {
110  r["type"] = Variant(std::string(type->name()));
111  }
112  r["id"] = Variant(std::to_string(v->id()));
113  return Variant(r);
114  } else {
115  return Variant(std::string("null"));
116  }
117 }
118 template <class T>
119 void deserialize(std::shared_ptr<T> &object, const Variant &variant) {
120  if (variant.type() == typeid(std::string)) {
121  auto s = variant.value<std::string>();
122  if (s == "null" || s == "") {
123  object = nullptr;
124  } else {
125  throw std::runtime_error("invalid value");
126  }
127  } else {
128  auto map = variant.value<std::map<std::string, Variant>>();
129  if (map.find("type") != map.end()) {
130  std::string type_name = map["type"].value<std::string>();
131  auto type = Type::tryFind(type_name);
132  if (!type) {
133  throw SerializationTypeErrorImpl<T>(type_name);
134  }
135  if (object == nullptr ||
136  std::type_index(typeid(*object)) != type->typeId()) {
137  object = type->instantiate<T>();
138  }
139  } else {
140  std::shared_ptr<Type> type = Type::find<T>();
141  if (object == nullptr ||
142  std::type_index(typeid(*object)) != type->typeId()) {
143  object = type->instantiate<T>();
144  }
145  }
146  if (map.find("id") != map.end()) {
147  object->setId(std::stoull(map["id"].value<std::string>()));
148  }
149  for (auto &property : object->properties()) {
150  if (map.find(property.name()) != map.end()) {
151  property.deserialize(map[property.name()]);
152  }
153  }
154  }
155 }
156 
157 template <class T>
158 Variant PropertyInfoImpl<T>::serialize(const void *value) const {
159  return ::serialize(*(T *)value);
160 }
161 template <class T>
162 void PropertyInfoImpl<T>::deserialize(void *value, const Variant &v) const {
163  ::deserialize(*(T *)value, v);
164 }
165 
166 void toYAML(const Variant &v, std::ostream &stream);
167 std::string toYAML(const Variant &v);
168 
169 Variant parseYAML(const std::string &str);
Definition: type.h:117
Definition: variant.h:9