scenario_simulator_v2 C++ API
attribute.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__ATTRIBUTE_HPP_
16 #define OPENSCENARIO_INTERPRETER__READER__ATTRIBUTE_HPP_
17 
18 #include <ament_index_cpp/get_package_share_directory.hpp>
19 #include <boost/algorithm/string/replace.hpp>
20 #include <concealer/execute.hpp>
21 #include <functional>
25 #include <optional>
26 #include <pugixml.hpp>
27 #include <regex>
29 #include <string>
30 #include <unordered_map>
31 
33 {
34 struct Scope;
35 
36 inline namespace syntax
37 {
38 struct Rule;
39 } // namespace syntax
40 
41 inline namespace reader
42 {
43 auto substitute(const std::string & attribute, const Scope & scope) -> String;
44 
45 template <typename T, typename Node, typename Scope>
46 auto readAttribute(const std::string & name, const Node & node, const Scope & scope) -> T
47 {
48  auto is_openscenario_standard_expression = [](const auto & s) {
49  return s.substr(0, 2) == "${" and s.back() == '}';
50  };
51 
52  auto read_openscenario_standard_expression = [&](const auto & s) {
53  return boost::lexical_cast<T>(evaluate(std::string(std::begin(s) + 2, std::end(s) - 1), scope));
54  };
55 
56  auto is_openscenario_standard_parameter_reference = [](const auto & s) {
57  return s.front() == '$';
58  };
59 
60  auto read_openscenario_standard_parameter_reference = [&](const auto & s) {
61  // TODO Use `return scope.template ref<T>(s.substr(1));`
62  if (auto && object = scope.ref(s.substr(1)); object) {
63  return boost::lexical_cast<T>(boost::lexical_cast<String>(object));
64  } else {
65  throw SyntaxError(
66  "There is no parameter named ", std::quoted(s.substr(1)), " (Attribute ", std::quoted(name),
67  " of class ", std::quoted(node.name()), " references this parameter)");
68  }
69  };
70 
71  auto read_openscenario_standard_literal = [&](const auto & s) {
72  try {
73  return boost::lexical_cast<T>(s);
74  } catch (const boost::bad_lexical_cast &) {
75  throw SyntaxError(
76  "Value ", std::quoted(s), " specified for attribute ", std::quoted(name),
77  " is invalid (Is not value of type ", makeTypename(typeid(T)), ")");
78  }
79  };
80 
81  // NOTE: https://www.asam.net/index.php?eID=dumpFile&t=f&f=4092&token=d3b6a55e911b22179e3c0895fe2caae8f5492467#_parameters
82 
83  if (const auto & attribute = node.attribute(name.c_str())) {
84  // NOTE: `substitute` is TIER IV extension (Non-OpenSCENARIO standard)
85  if (std::string value = substitute(attribute.value(), scope); value.empty()) {
86  return T();
87  } else if (is_openscenario_standard_expression(value)) {
88  return read_openscenario_standard_expression(value);
89  } else if (is_openscenario_standard_parameter_reference(value)) {
90  return read_openscenario_standard_parameter_reference(value);
91  } else {
92  return read_openscenario_standard_literal(value);
93  }
94  } else {
95  throw SyntaxError(
96  "Required attribute ", std::quoted(name), " not specified for class ",
97  std::quoted(node.name()));
98  }
99 }
100 
101 extern template auto readAttribute(const std::string &, const pugi::xml_node &, const Scope &)
102  -> Boolean;
103 extern template auto readAttribute(const std::string &, const pugi::xml_node &, const Scope &)
104  -> Double;
105 extern template auto readAttribute(const std::string &, const pugi::xml_node &, const Scope &)
106  -> syntax::Rule;
107 extern template auto readAttribute(const std::string &, const pugi::xml_node &, const Scope &)
108  -> String;
109 extern template auto readAttribute(const std::string &, const pugi::xml_node &, const Scope &)
110  -> UnsignedShort;
111 extern template auto readAttribute(const std::string &, const pugi::xml_node &, const Scope &)
112  -> UnsignedInteger;
113 
114 template <typename T, typename Node, typename Scope>
115 auto readAttribute(const std::string & name, const Node & node, const Scope & scope, T && value)
116  -> T
117 {
118  if (node.attribute(name.c_str())) {
119  return readAttribute<T>(name, node, scope);
120  } else {
121  return value;
122  }
123 }
124 
125 template <typename T, typename Node, typename Scope>
126 auto readAttribute(const std::string & name, const Node & node, const Scope & scope, std::nullopt_t)
127 {
128  if (node.attribute(name.c_str())) {
129  return std::make_optional(readAttribute<T>(name, node, scope));
130  } else {
131  return std::optional<T>();
132  }
133 }
134 } // namespace reader
135 } // namespace openscenario_interpreter
136 
137 #endif // OPENSCENARIO_INTERPRETER__READER__ATTRIBUTE_HPP_
Definition: scope.hpp:154
auto ref(Ts &&... xs) const -> decltype(auto)
Definition: scope.hpp:210
auto readAttribute(const std::string &name, const Node &node, const Scope &scope) -> T
Definition: attribute.hpp:46
std::string evaluate(const std::string &, const Scope &)
Definition: evaluate.cpp:237
auto substitute(const std::string &attribute, const Scope &scope) -> String
Definition: attribute.cpp:23
std::string String
Definition: string.hpp:24
auto makeTypename(Ts &&... xs)
Definition: demangle.hpp:30
Definition: hypot.hpp:22
std::string string
Definition: junit5.hpp:26
Definition: boolean.hpp:26
Definition: double.hpp:25
Definition: unsigned_integer.hpp:27
Definition: unsigned_short.hpp:26