scenario_simulator_v2 C++ API
element.hpp
Go to the documentation of this file.
1 // Copyright 2015 TIER IV, Inc. All rights reserved.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #ifndef OPENSCENARIO_INTERPRETER__READER__ELEMENT_HPP_
16 #define OPENSCENARIO_INTERPRETER__READER__ELEMENT_HPP_
17 
18 #include <functional>
19 #include <iterator>
20 #include <limits>
24 #include <pugixml.hpp>
25 #include <string>
26 #include <type_traits>
27 #include <unordered_map>
28 #include <utility>
29 
30 /* ---- NOTE -------------------------------------------------------------------
31  *
32  * The functions in the reader header only deal with XML parsing and should
33  * not have any prerequisite knowledge about OpenSCENARIO data structures.
34  *
35  * -------------------------------------------------------------------------- */
36 
38 {
39 struct Scope;
40 
41 inline namespace syntax
42 {
43 struct Orientation;
44 struct Position;
45 struct Properties;
46 struct Range;
47 } // namespace syntax
48 
49 inline namespace reader
50 {
51 using Cardinality =
52  typename std::iterator_traits<typename pugi::xml_node::iterator>::difference_type;
53 
54 constexpr auto unbounded = std::numeric_limits<Cardinality>::max();
55 
56 template <Cardinality MinOccurs, Cardinality MaxOccurs, typename F>
57 auto traverse(const pugi::xml_node & parent, const std::string & name, F && f) -> void
58 {
59  const auto children = parent.children(name.c_str());
60 
61  if (const auto size = iterator::size(children)) {
62  if (MinOccurs != 0 and size < MinOccurs) {
63  throw SyntaxError(
64  parent.name(), " requires class ", name, " at least ", MinOccurs, " element",
65  (1 < MinOccurs ? "s" : ""), ", but ", size, " element", (1 < size ? "s" : ""),
66  " specified");
67  } else if (MaxOccurs < size) {
68  throw SyntaxError(
69  parent.name(), " requires class ", name, " at most ", MaxOccurs, " element",
70  (1 < MaxOccurs ? "s" : ""), ", but ", size, " element", (1 < size ? "s" : ""),
71  " specified");
72  } else {
73  for (const auto & child : children) {
74  f(child);
75  }
76  }
77  } else if (MinOccurs != 0) {
78  throw SyntaxError(
79  parent.name(), " requires class ", name, " at least ", MinOccurs, " element",
80  (1 < MinOccurs ? "s" : ""), ", but there is no specification");
81  }
82 }
83 
84 template <typename T, typename Scope>
85 auto readElement(const std::string & name, const pugi::xml_node & parent, Scope & scope) -> T
86 {
87  if (const auto child = parent.child(name.c_str())) {
88  return T(child, scope);
89  } else {
90  /* ---- NOTE ---------------------------------------------------------------
91  *
92  * If the given XML node does not have a child element (T type) with the
93  * specified name, it assumes that the T type is an optional element and
94  * attempts to build a default T type.
95  *
96  * ---------------------------------------------------------------------- */
98  parent.name(), " requires class ", name, " as element, but there is no declaration"));
99  }
100 }
101 
102 extern template auto readElement(const std::string &, const pugi::xml_node &, Scope &)
103  -> syntax::Orientation;
104 extern template auto readElement(const std::string &, const pugi::xml_node &, Scope &)
105  -> syntax::Position;
106 extern template auto readElement(const std::string &, const pugi::xml_node &, Scope &)
107  -> syntax::Properties;
108 extern template auto readElement(const std::string &, const pugi::xml_node &, Scope &)
109  -> syntax::Range;
110 
111 template <typename T, typename U, typename Scope>
112 auto readElement(const std::string & name, const pugi::xml_node & parent, Scope & scope, U && value)
113 {
114  if constexpr (std::is_same<T, typename std::decay<U>::type>::value) {
115  // use 'value' as a default element.
116  // the default element will be used when current scope has no description about 'name'.
117  if (const auto child = parent.child(name.c_str())) {
118  return T(child, scope);
119  } else {
120  return value;
121  }
122  } else {
123  // use "value" as an additional arguments to the constructor
124  if (const auto child = parent.child(name.c_str())) {
125  return T(child, scope, value);
126  } else {
128  parent.name(), " requires class ", name, " as element, but there is no declaration"));
129  }
130  }
131 }
132 
133 template <typename T, Cardinality MinOccurs, Cardinality MaxOccurs = unbounded, typename... Ts>
134 auto readElements(const std::string & name, const pugi::xml_node & node, Ts &&... xs)
135 {
136  std::list<T> elements;
137 
138  traverse<MinOccurs, MaxOccurs>(node, name, [&](auto && x) {
139  elements.emplace_back(std::forward<decltype(x)>(x), std::forward<decltype(xs)>(xs)...);
140  });
141 
142  return elements;
143 }
144 
145 template <typename GroupT, Cardinality MinOccurs, Cardinality MaxOccurs = unbounded, typename... Ts>
146 auto readGroups(const pugi::xml_node & node, Ts &&... xs)
147 {
148  std::list<GroupT> groups;
149 
150  for (auto child = node.begin(); child != node.end(); ++child) {
151  try {
152  groups.template emplace_back(
153  std::forward<decltype(*child)>(*child), std::forward<decltype(xs)>(xs)...);
154  } catch (...) {
155  }
156  }
157 
158  if (MinOccurs != 0 and groups.size() < MinOccurs) {
159  throw SyntaxError(
160  node.name(), " requires Group ", demangle(typeid(GroupT)), " at least ", MinOccurs,
161  " element", (1 < MinOccurs ? "s" : ""), ", but ", groups.size(), " element",
162  (1 < groups.size() ? "s" : ""), " specified");
163  }
164 
165  if (MaxOccurs < groups.size()) {
166  throw SyntaxError(
167  node.name(), " requires Group ", demangle(typeid(GroupT)), " at most ", MaxOccurs, " element",
168  (1 < MaxOccurs ? "s" : ""), ", but ", groups.size(), " element",
169  (1 < groups.size() ? "s" : ""), " specified");
170  }
171 
172  return groups;
173 }
174 
175 auto choice(
176  const pugi::xml_node & node,
177  const std::unordered_map<std::string, std::function<Object(const pugi::xml_node &)>> & callees)
178  -> Object;
179 } // namespace reader
180 } // namespace openscenario_interpreter
181 
182 #endif // OPENSCENARIO_INTERPRETER__READER__ELEMENT_HPP_
Definition: scope.hpp:154
auto size(const T &range)
Definition: size.hpp:25
auto readElement(const std::string &name, const pugi::xml_node &parent, Scope &scope) -> T
Definition: element.hpp:85
auto choice(const pugi::xml_node &node, const std::unordered_map< std::string, std::function< Object(const pugi::xml_node &)>> &callees) -> Object
Definition: element.cpp:33
auto readElements(const std::string &name, const pugi::xml_node &node, Ts &&... xs)
Definition: element.hpp:134
auto traverse(const pugi::xml_node &parent, const std::string &name, F &&f) -> void
Definition: element.hpp:57
typename std::iterator_traits< typename pugi::xml_node::iterator >::difference_type Cardinality
Definition: element.hpp:52
constexpr auto unbounded
Definition: element.hpp:54
auto readGroups(const pugi::xml_node &node, Ts &&... xs)
Definition: element.hpp:146
auto demangle(const char *name) -> std::string
Definition: demangle.cpp:26
Definition: hypot.hpp:22
Pointer< Expression > Object
Definition: object.hpp:26
Definition: junit5.hpp:25
std::string string
Definition: junit5.hpp:26
Definition: must_be_default_constructible.hpp:27