Project
Loading...
Searching...
No Matches
PropertyTreeHelpers.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
12#include "PropertyTreeHelpers.h"
17
18#include <boost/program_options/variables_map.hpp>
19
20#include <vector>
21#include <string>
22
23namespace o2::framework
24{
25namespace
26{
28template <VariantType T>
29inline Variant fromString(std::string const& str)
30{
31 std::stringstream ss;
32 ss.str(str);
33 return VariantJSONHelpers::read<T>(ss);
34}
35} // namespace
36void PropertyTreeHelpers::populateDefaults(std::vector<ConfigParamSpec> const& schema,
37 boost::property_tree::ptree& pt,
38 boost::property_tree::ptree& provenance)
39{
40 for (auto const& spec : schema) {
41 std::string key = spec.name.substr(0, spec.name.find(','));
42 try {
43 if (spec.defaultValue.type() == VariantType::Empty) {
44 continue;
45 }
46 switch (spec.type) {
48 pt.put(key, spec.defaultValue.get<int>());
49 break;
51 pt.put(key, spec.defaultValue.get<int8_t>());
52 break;
54 pt.put(key, spec.defaultValue.get<int16_t>());
55 break;
57 pt.put(key, spec.defaultValue.get<uint8_t>());
58 break;
60 pt.put(key, spec.defaultValue.get<uint16_t>());
61 break;
63 pt.put(key, spec.defaultValue.get<uint32_t>());
64 break;
66 pt.put(key, spec.defaultValue.get<uint64_t>());
67 break;
69 pt.put(key, spec.defaultValue.get<int64_t>());
70 break;
72 pt.put(key, spec.defaultValue.get<float>());
73 break;
75 pt.put(key, spec.defaultValue.get<double>());
76 break;
78 pt.put(key, spec.defaultValue.get<std::string>());
79 break;
81 pt.put(key, spec.defaultValue.get<bool>());
82 break;
84 pt.put_child(key, boost::property_tree::ptree{});
85 break;
87 pt.put_child(key, vectorToBranch(spec.defaultValue.get<int*>(), spec.defaultValue.size()));
88 break;
90 pt.put_child(key, vectorToBranch(spec.defaultValue.get<float*>(), spec.defaultValue.size()));
91 break;
93 pt.put_child(key, vectorToBranch(spec.defaultValue.get<double*>(), spec.defaultValue.size()));
94 break;
96 pt.put_child(key, vectorToBranch(spec.defaultValue.get<bool*>(), spec.defaultValue.size()));
97 break;
99 pt.put_child(key, vectorToBranch(spec.defaultValue.get<std::string*>(), spec.defaultValue.size()));
100 break;
102 pt.put_child(key, array2DToBranch(spec.defaultValue.get<Array2D<int>>()));
103 break;
105 pt.put_child(key, array2DToBranch(spec.defaultValue.get<Array2D<float>>()));
106 break;
108 pt.put_child(key, array2DToBranch(spec.defaultValue.get<Array2D<double>>()));
109 break;
111 pt.put_child(key, labeledArrayToBranch(spec.defaultValue.get<LabeledArray<int>>()));
112 break;
114 pt.put_child(key, labeledArrayToBranch(spec.defaultValue.get<LabeledArray<float>>()));
115 break;
117 pt.put_child(key, labeledArrayToBranch(spec.defaultValue.get<LabeledArray<double>>()));
118 break;
120 pt.put_child(key, labeledArrayToBranch(spec.defaultValue.get<LabeledArray<std::string>>()));
121 break;
124 default:
125 throw runtime_error_f("Unknown variant type", spec.type);
126 }
127 provenance.put(key, "default");
128 } catch (std::runtime_error& re) {
129 throw;
130 } catch (std::exception& e) {
131 throw std::invalid_argument(std::string("missing option: ") + key + " (" + e.what() + ")");
132 } catch (...) {
133 throw std::invalid_argument(std::string("missing option: ") + key);
134 }
135 }
136}
137
138void PropertyTreeHelpers::populate(std::vector<ConfigParamSpec> const& schema,
139 boost::property_tree::ptree& pt,
140 boost::program_options::variables_map const& vmap,
141 boost::property_tree::ptree& provenance)
142{
143 for (auto const& spec : schema) {
144 // strip short version to get the correct key
145 std::string key = spec.name.substr(0, spec.name.find(','));
146 if (vmap.count(key) == 0) {
147 continue;
148 }
149 try {
150 switch (spec.type) {
151 case VariantType::Int:
152 pt.put(key, vmap[key].as<int>());
153 break;
155 pt.put(key, vmap[key].as<int8_t>());
156 break;
158 pt.put(key, vmap[key].as<int16_t>());
159 break;
161 pt.put(key, vmap[key].as<uint8_t>());
162 break;
164 pt.put(key, vmap[key].as<uint16_t>());
165 break;
167 pt.put(key, vmap[key].as<uint32_t>());
168 break;
170 pt.put(key, vmap[key].as<uint64_t>());
171 break;
173 pt.put(key, vmap[key].as<int64_t>());
174 break;
176 pt.put(key, vmap[key].as<float>());
177 break;
179 pt.put(key, vmap[key].as<double>());
180 break;
182 if (auto const* v = boost::any_cast<std::string>(&vmap[key].value())) {
183 pt.put(key, *v);
184 }
185 break;
187 pt.put(key, vmap[key].as<bool>());
188 break;
190 auto v = fromString<VariantType::ArrayInt>(vmap[key].as<std::string>());
191 pt.put_child(key, vectorToBranch<int>(v.get<int*>(), v.size()));
192 } break;
194 auto v = fromString<VariantType::ArrayFloat>(vmap[key].as<std::string>());
195 pt.put_child(key, vectorToBranch<float>(v.get<float*>(), v.size()));
196 } break;
198 auto v = fromString<VariantType::ArrayDouble>(vmap[key].as<std::string>());
199 pt.put_child(key, vectorToBranch<double>(v.get<double*>(), v.size()));
200 } break;
202 // pt.put_child(key, vectorToBranch<bool>(stringToVector<bool>(vmap[key].as<std::string>())));
203 break;
205 auto v = fromString<VariantType::ArrayString>(vmap[key].as<std::string>());
206 pt.put_child(key, vectorToBranch<std::string>(v.get<std::string*>(), v.size()));
207 } break;
209 auto v = fromString<VariantType::Array2DInt>(vmap[key].as<std::string>());
210 pt.put_child(key, array2DToBranch<int>(v.get<Array2D<int>>()));
211 } break;
213 auto v = fromString<VariantType::Array2DFloat>(vmap[key].as<std::string>());
214 pt.put_child(key, array2DToBranch<float>(v.get<Array2D<float>>()));
215 } break;
217 auto v = fromString<VariantType::Array2DDouble>(vmap[key].as<std::string>());
218 pt.put_child(key, array2DToBranch<double>(v.get<Array2D<double>>()));
219 } break;
221 auto v = fromString<VariantType::LabeledArrayInt>(vmap[key].as<std::string>());
222 pt.put_child(key, labeledArrayToBranch(v.get<LabeledArray<int>>()));
223 } break;
225 auto v = fromString<VariantType::LabeledArrayFloat>(vmap[key].as<std::string>());
226 pt.put_child(key, labeledArrayToBranch(v.get<LabeledArray<float>>()));
227 } break;
229 auto v = fromString<VariantType::LabeledArrayDouble>(vmap[key].as<std::string>());
230 pt.put_child(key, labeledArrayToBranch(v.get<LabeledArray<double>>()));
231 } break;
233 auto v = fromString<VariantType::LabeledArrayString>(vmap[key].as<std::string>());
235 } break;
237 pt.put_child(key, vmap[key].as<boost::property_tree::ptree>());
238 break;
241 default:
242 throw runtime_error("Unknown variant type");
243 }
244 provenance.put(key, "fairmq");
245 } catch (std::runtime_error& re) {
246 throw;
247 } catch (std::exception& e) {
248 throw std::invalid_argument(std::string("missing option: ") + key + " (" + e.what() + ")");
249 } catch (...) {
250 throw std::invalid_argument(std::string("missing option: ") + key);
251 }
252 }
253}
254
255template <typename T>
257{
258 if (!input.getLabelsCols().empty() || !input.getLabelsRows().empty()) {
259 return false;
260 }
261 if (spec.getLabelsCols().empty() == false) {
262 input.replaceLabelsCols(spec.getLabelsCols());
263 }
264 if (spec.getLabelsRows().empty() == false) {
265 input.replaceLabelsRows(spec.getLabelsRows());
266 }
267 return true;
268}
269
270void PropertyTreeHelpers::populate(std::vector<ConfigParamSpec> const& schema,
271 boost::property_tree::ptree& pt,
272 boost::property_tree::ptree const& in,
273 boost::property_tree::ptree& provenance,
274 std::string const& provenanceLabel)
275{
276 for (auto const& spec : schema) {
277 // strip short version to get the correct key
278 std::string key = spec.name.substr(0, spec.name.find(','));
279 auto it = in.get_child_optional(key);
280 if (!it) {
281 it = in.get_child_optional(boost::property_tree::path(key, '/'));
282 }
283 if (!it) {
284 continue;
285 }
286 try {
287 switch (spec.type) {
288 case VariantType::Int:
289 pt.put(key, (*it).get_value<int>());
290 break;
292 pt.put(key, (*it).get_value<int8_t>());
293 break;
295 pt.put(key, (*it).get_value<int16_t>());
296 break;
298 pt.put(key, (*it).get_value<uint8_t>());
299 break;
301 pt.put(key, (*it).get_value<uint16_t>());
302 break;
304 pt.put(key, (*it).get_value<uint32_t>());
305 break;
307 pt.put(key, (*it).get_value<uint64_t>());
308 break;
310 pt.put(key, (*it).get_value<int64_t>());
311 break;
313 pt.put(key, (*it).get_value<float>());
314 break;
316 pt.put(key, (*it).get_value<double>());
317 break;
319 pt.put(key, (*it).get_value<std::string>());
320 break;
322 pt.put(key, (*it).get_value<bool>());
323 break;
333 pt.put_child(key, *it);
334 break;
336 auto v = labeledArrayFromBranch<int>(it.value());
337 if (!replaceLabels(v, spec.defaultValue.get<LabeledArray<int>>())) {
338 pt.put_child(key, *it);
339 } else {
340 pt.put_child(key, labeledArrayToBranch(std::move(v)));
341 }
342 }; break;
344 auto v = labeledArrayFromBranch<float>(it.value());
345 if (!replaceLabels(v, spec.defaultValue.get<LabeledArray<float>>())) {
346 pt.put_child(key, *it);
347 } else {
348 pt.put_child(key, labeledArrayToBranch(std::move(v)));
349 }
350 }; break;
352 auto v = labeledArrayFromBranch<double>(it.value());
353 if (!replaceLabels(v, spec.defaultValue.get<LabeledArray<double>>())) {
354 pt.put_child(key, *it);
355 } else {
356 pt.put_child(key, labeledArrayToBranch(std::move(v)));
357 }
358 }; break;
360 auto v = labeledArrayFromBranch<std::string>(it.value());
361 if (!replaceLabels(v, spec.defaultValue.get<LabeledArray<std::string>>())) {
362 pt.put_child(key, *it);
363 } else {
364 pt.put_child(key, labeledArrayToBranch(std::move(v)));
365 }
366 }; break;
369 default:
370 throw std::runtime_error("Unknown variant type");
371 }
372 provenance.put(key, provenanceLabel);
373 } catch (std::runtime_error& re) {
374 throw;
375 } catch (std::exception& e) {
376 throw std::invalid_argument(std::string("missing option: ") + key + " (" + e.what() + ")");
377 } catch (...) {
378 throw std::invalid_argument(std::string("missing option: ") + key);
379 }
380 }
381}
382
383namespace
384{
385void traverseRecursive(const boost::property_tree::ptree& parent,
386 const boost::property_tree::ptree::path_type& childPath,
387 const boost::property_tree::ptree& child,
389{
390 using boost::property_tree::ptree;
391
392 method(parent, childPath, child);
393 for (ptree::const_iterator it = child.begin(); it != child.end(); ++it) {
394 ptree::path_type curPath = childPath / ptree::path_type(it->first);
395 traverseRecursive(parent, curPath, it->second, method);
396 }
397}
398} // namespace
399
400template <>
401void PropertyTreeHelpers::traverse<boost::property_tree::ptree>(const boost::property_tree::ptree& parent, PropertyTreeHelpers::WalkerFunction<boost::property_tree::ptree>& method)
402{
403 traverseRecursive(parent, "", parent, method);
404}
405
406} // namespace o2::framework
template o2::framework::LabeledArray< std::string > o2::framework::labeledArrayFromBranch< std::string >(boost::property_tree::ptree const &tree)
StringRef key
void replaceLabelsRows(std::vector< std::string > const &labels)
Definition Array2D.h:260
void replaceLabelsCols(std::vector< std::string > const &labels)
Definition Array2D.h:265
const GLdouble * v
Definition glcorearb.h:832
GLsizei const GLfloat * value
Definition glcorearb.h:819
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
RuntimeErrorRef runtime_error(const char *)
boost::property_tree::ptree labeledArrayToBranch(LabeledArray< T > &&array)
auto replaceLabels(LabeledArray< T > &input, LabeledArray< T > &&spec)
boost::property_tree::ptree vectorToBranch(T *values, size_t size)
boost::property_tree::ptree array2DToBranch(Array2D< T > &&array)
RuntimeErrorRef runtime_error_f(const char *,...)
auto getLabelsRows() const
Definition Array2D.h:147
auto getLabelsCols() const
Definition Array2D.h:152
static void populate(std::vector< ConfigParamSpec > const &schema, boost::property_tree::ptree &tree, boost::program_options::variables_map const &vmap, boost::property_tree::ptree &provenance)
std::function< void(boost::property_tree::ptree const &, typename T::path_type, boost::property_tree::ptree const &)> WalkerFunction
static void populateDefaults(std::vector< ConfigParamSpec > const &schema, boost::property_tree::ptree &tree, boost::property_tree::ptree &provenance)
const std::string str