19 #include "yaml-cpp/yaml.h"
41 std::string_view key) {
43 for (
const auto §ion : node.value()) {
56 if (section.first.Scalar() == key) {
57 node.value().reset(node.value()[key]);
73 std::vector<std::string> to_remove(root.size());
77 if (
n.second.IsMap() &&
n.second.size() == 0) {
78 to_remove.emplace_back(
n.first.Scalar());
81 for (
const auto &key : to_remove) {
95 YAML::Node
operator|=(YAML::Node a,
const YAML::Node &b) {
98 a[n0.first.Scalar()] |= n0.second;
113 return std::accumulate(keys.begin(), keys.end(), std::string{
"{"},
114 [](
const std::string &ss,
const std::string_view &s) {
115 return ss + ((ss.size() == 1) ?
"\"" :
", \"") +
125 Configuration::Configuration(
const std::filesystem::path &path)
130 const std::filesystem::path &filename) {
131 const auto file_path = path / filename;
132 if (!std::filesystem::exists(file_path)) {
135 "', but the file does not exist.");
138 throw std::runtime_error(
139 "The configuration file has CR LF line endings. Please use LF "
143 root_node_ = YAML::LoadFile(file_path.native());
144 }
catch (YAML::ParserException &e) {
145 if (e.msg ==
"illegal map value" || e.msg ==
"end of map not found") {
146 const auto line = std::to_string(e.mark.line + 1);
147 throw ParseError(
"YAML parse error at\n" + file_path.native() +
':' +
148 line +
": " + e.msg +
149 " (check that the indentation of map keys matches)");
156 : root_node_(std::move(other.root_node_)),
157 uncaught_exceptions_(std::move(other.uncaught_exceptions_)),
158 existing_keys_already_taken_(
159 std::move(other.existing_keys_already_taken_)) {
160 other.root_node_.reset();
161 other.uncaught_exceptions_ = 0;
162 other.existing_keys_already_taken_.clear();
171 std::move(other.existing_keys_already_taken_);
172 other.root_node_.reset();
173 other.uncaught_exceptions_ = 0;
174 other.existing_keys_already_taken_.clear();
184 throw std::logic_error(
185 "Configuration object destroyed with unused keys:\n" +
to_string());
195 }
catch (YAML::ParserException &e) {
196 if (e.msg ==
"illegal map value" || e.msg ==
"end of map not found") {
197 const auto line = std::to_string(e.mark.line + 1);
198 throw ParseError(
"YAML parse error in:\n" + yaml +
"\nat line " + line +
200 " (check that the indentation of map keys matches)");
207 std::vector<std::string> r;
210 r.emplace_back(i.first.Scalar());
216 assert(labels.size() > 0);
220 auto last_key_it = labels.end() - 1;
221 auto previous_to_last_node =
223 auto to_be_returned{previous_to_last_node};
225 if (!previous_to_last_node || !to_be_returned) {
226 throw std::runtime_error(
227 "Private Configuration::take method called with not existing key: " +
228 join_quoted(labels) +
". This should not have happened.");
230 previous_to_last_node.value().remove(*last_key_it);
232 if (
const KeyLabels key_labels{labels.begin(), labels.end()};
242 return {to_be_returned.value(), last_key_it->data()};
246 std::vector<std::string_view> labels)
const {
250 return {found_node.value(), labels.back().data()};
252 throw std::runtime_error(
253 "Private Configuration::read method called with not existing key: " +
254 join_quoted(labels) +
". This should not have happened.");
259 const std::string &key,
KeyLabels section) {
262 std::vector<std::string> to_remove{};
263 bool key_exists =
false;
264 for (
auto i : found_node.value()) {
265 if (i.first.Scalar() != key) {
266 to_remove.push_back(i.first.Scalar());
272 std::string section_string{
" section "};
273 if (section.size() > 0) {
274 section_string +=
join_quoted({section.begin(), section.end()}) +
" ";
276 section_string =
" top-level" + section_string;
278 throw std::invalid_argument(
"Attempt to remove all keys in" +
280 "except not existing one: \"" + key +
"\"");
282 for (
auto i : to_remove) {
283 found_node.value().remove(i);
287 throw std::invalid_argument(
288 "Attempt to remove entries in not existing section: " +
296 assert(section.size() > 0);
297 auto last_key_it = section.end() - 1;
298 auto previous_to_section_node =
300 auto sub_conf_root_node{previous_to_section_node};
302 if (!previous_to_section_node || !sub_conf_root_node) {
306 throw std::runtime_error(
"Attempt to extract not existing section " +
312 else if (sub_conf_root_node->IsNull() ||
313 (sub_conf_root_node->IsMap() && sub_conf_root_node->size() == 0)) {
316 throw std::runtime_error(
"Attempt to extract empty section " +
318 }
else if (sub_conf_root_node->IsMap() && sub_conf_root_node->size() != 0) {
320 previous_to_section_node->remove(*last_key_it);
324 throw std::runtime_error(
"Tried to extract configuration section at " +
326 " to get a key value. Use take instead!");
332 auto sub_configuration =
334 sub_configuration.enclose_into_section(section);
335 return sub_configuration;
350 YAML::Node new_root_node{YAML::NodeType::Map};
351 auto last_node = new_root_node;
352 for (
const auto &label : section) {
353 last_node[label] = YAML::Node(YAML::NodeType::Map);
354 last_node.reset(last_node[label]);
367 std::vector<std::string_view> keys)
const {
373 for (
const auto &key : keys) {
380 std::vector<std::string_view> keys)
const {
381 assert(keys.size() > 0);
383 for (
const auto &key : keys) {
385 node.reset(node[key]);
411 std::vector<KeyLabels> &list,
415 for (
const auto &sub_node : root_node) {
416 new_list_entry.push_back(sub_node.first.as<std::string>());
417 if (sub_node.second.IsMap())
421 list.push_back(new_list_entry);
422 new_list_entry.pop_back();
440 std::vector<KeyLabels> list{};
451 template <
typename T>
458 static constexpr
bool value =
false;
468 template <
typename MapKey,
typename MapValue>
474 static constexpr
bool value =
true;
484 std::vector<KeyLabels> labels_of_keys_taken_as_map{};
487 [&labels_of_keys_taken_as_map](
auto &&var) {
498 if constexpr (
IsStdMap<
typename std::decay_t<
499 decltype(var.get())>::type>::value)
500 labels_of_keys_taken_as_map.push_back(var.get().labels());
504 return labels_of_keys_taken_as_map;
520 std::vector<KeyLabels> &list_of_input_key_labels) {
521 const std::vector<KeyLabels> labels_of_keys_taken_as_map =
523 for (
const auto &labels : labels_of_keys_taken_as_map) {
525 list_of_input_key_labels.end(),
526 [&labels](
KeyLabels &labels_of_input_key) {
527 if (std::equal(labels.begin(), labels.end(),
528 labels_of_input_key.begin(),
529 labels_of_input_key.begin() + labels.size()))
530 labels_of_input_key = labels;
535 list_of_input_key_labels.erase(std::unique(list_of_input_key_labels.begin(),
536 list_of_input_key_labels.end()),
537 list_of_input_key_labels.end());
556 auto key_ref_var_it = std::find_if(
558 [&labels](
auto key) {
560 [&labels](
auto &&arg) {
return arg.get().has_same_labels(labels); },
565 " is not a valid SMASH input key.");
570 const auto key_labels =
571 std::visit([](
auto &&var) {
return static_cast<std::string
>(var.get()); },
574 if (std::visit([](
auto &&var) {
return !var.get().is_allowed(); },
576 const auto v_removal = std::visit(
577 [](
auto &&var) {
return var.get().removed_in(); }, found_variant);
579 " has been removed in version ", v_removal,
580 " and it is not valid anymore.");
583 if (std::visit([](
auto &&var) {
return var.get().is_deprecated(); },
585 const auto v_deprecation = std::visit(
586 [](
auto &&var) {
return var.get().deprecated_in(); }, found_variant);
588 "Key ", key_labels,
" has been deprecated in version ", v_deprecation);
609 switch (result_so_far) {
614 result_so_far = new_value;
618 result_so_far = new_value;
629 for (
const auto &key_labels : list) {
631 if (full_validation) {
638 return validation_result;
Proxy object to be used when taking or reading keys in the configuration.
Interface to the SMASH configuration files.
Is
Return type of Configuration::validate which conveys more information that simply a two-state boolean...
void merge_yaml(const std::string &yaml)
Merge the configuration in yaml into the existing tree.
std::string to_string() const
Return a string of the current YAML tree.
Configuration extract_sub_configuration(KeyLabels section, Configuration::GetEmpty empty_if_not_existing=Configuration::GetEmpty::No)
Create a new configuration from a then-removed section of the present object.
T read(const Key< T > &key) const
Additional interface for SMASH to read configuration values without removing them.
Configuration(const std::filesystem::path &path)
Read config.yaml from the specified path.
void enclose_into_section(KeyLabels section)
Enclose the configuration into the given section.
YAML::Node root_node_
The general_config.yaml contents - fully parsed.
YAML::Node find_node_creating_it_if_not_existing(std::vector< std::string_view > keys) const
Descend in and if needed modify the YAML tree from the given node using the provided keys.
int uncaught_exceptions_
Counter to be able to optionally throw in destructor.
bool did_key_exist_and_was_it_already_taken(const KeyLabels &labels) const
Find out whether a key has been already taken.
Is validate(bool full_validation=true) const
Validate content of configuration in terms of YAML keys.
std::optional< YAML::Node > find_existing_node(std::vector< std::string_view > keys) const
Descend in the YAML tree from the given node using the provided keys.
~Configuration() noexcept(false)
Destroy the object, optionally throwing if not all keys were taken.
GetEmpty
Flag to tune method(s) behavior such that it is descriptive from the caller side.
Configuration extract_complete_sub_configuration(KeyLabels section, Configuration::GetEmpty empty_if_not_existing=Configuration::GetEmpty::No)
Alternative method to extract a sub-configuration, which retains the labels from the top-level in the...
T take(const Key< T > &key)
The default interface for SMASH to read configuration values.
void remove_all_entries_in_section_but_one(const std::string &key, KeyLabels section={})
Remove all entries in the given section except for key.
std::vector< std::string > list_upmost_nodes()
Lists all YAML::Nodes from the configuration setup.
std::vector< KeyLabels > existing_keys_already_taken_
List of taken keys to throw on taking same key twice.
Configuration & operator=(const Configuration &)=delete
Prevent Configuration objects from being copy-assigned.
std::array< einhard::Logger<>, std::tuple_size< LogArea::AreaTuple >::value > logg
An array that stores all pre-configured Logger objects.
auto collect_input_keys_taken_as_maps()
Extract from the InputKeys database the labels of keys that have a std::map as type.
void adjust_list_of_labels_dealing_with_keys_taken_as_maps(std::vector< KeyLabels > &list_of_input_key_labels)
Remove last labels of keys that are taken as maps in SMASH and remove duplicates from the resulting l...
std::string join_quoted(std::vector< std::string_view > keys)
Build a string with a list of keys as specified in the code.
void fill_list_of_labels_per_key_in_yaml_tree(const YAML::Node &root_node, std::vector< KeyLabels > &list, KeyLabels &new_list_entry)
Implementation of the algorithm to translate a YAML tree into lists of labels, each identifying a key...
YAML::Node operator|=(YAML::Node a, const YAML::Node &b)
Merge two YAML::Nodes.
void accumulate_validation(Configuration::Is &result_so_far, Configuration::Is new_value)
Utility function to accumulate validation results of keys.
Configuration::Is validate_key(const KeyLabels &labels)
Given some YAML labels (assumed to be in order from the top section), it is checked whether any valid...
void descend_one_existing_level(std::optional< YAML::Node > &node, std::string_view key)
Reset the passed in node to that one at the provided key, which is expected to exist in the node.
YAML::Node remove_empty_maps(YAML::Node root)
Remove all empty maps of a YAML::Node.
auto get_list_of_labels_per_key_in_yaml_tree(const YAML::Node &root_node)
Create a list of lists of key labels present in the passed YAML node considered to be the root one of...
std::vector< std::string > KeyLabels
Descriptive alias for storing key labels, i.e.
UnaryFunction for_each(Container &&c, UnaryFunction &&f)
Convenience wrapper for std::for_each that operates on a complete container.
std::string quote(const std::string &s)
Add quotes around string.
static constexpr int LConfiguration
std::string join(const std::vector< std::string > &v, const std::string &delim)
Join strings using delimiter.
bool has_crlf_line_ending(const std::string in)
Check if a line in the string ends with \r\n.
std::string read_all(std::istream &&input)
Utility function to read a complete input stream (e.g.
Thrown if the file does not exist.
Thrown for YAML parse errors.
std::variant< std::reference_wrapper< const Key< bool > >, std::reference_wrapper< const Key< int > >, std::reference_wrapper< const Key< int64_t > >, std::reference_wrapper< const Key< double > >, std::reference_wrapper< const Key< std::string > >, std::reference_wrapper< const Key< std::array< int, 3 > >>, std::reference_wrapper< const Key< std::array< double, 2 > >>, std::reference_wrapper< const Key< std::array< double, 3 > >>, std::reference_wrapper< const Key< std::pair< double, double > >>, std::reference_wrapper< const Key< std::vector< double > >>, std::reference_wrapper< const Key< std::vector< std::string > >>, std::reference_wrapper< const Key< std::set< ThermodynamicQuantity > >>, std::reference_wrapper< const Key< std::map< PdgCode, int > >>, std::reference_wrapper< const Key< std::map< std::string, std::string > >>, std::reference_wrapper< const Key< einhard::LogLevel > >, std::reference_wrapper< const Key< BoxInitialCondition > >, std::reference_wrapper< const Key< CalculationFrame > >, std::reference_wrapper< const Key< CollisionCriterion > >, std::reference_wrapper< const Key< DensityType > >, std::reference_wrapper< const Key< DerivativesMode > >, std::reference_wrapper< const Key< ExpansionMode > >, std::reference_wrapper< const Key< FermiMotion > >, std::reference_wrapper< const Key< FieldDerivativesMode > >, std::reference_wrapper< const Key< FluidizableProcessesBitSet > >, std::reference_wrapper< const Key< FluidizationType > >, std::reference_wrapper< const Key< MultiParticleReactionsBitSet > >, std::reference_wrapper< const Key< NNbarTreatment > >, std::reference_wrapper< const Key< OutputOnlyFinal > >, std::reference_wrapper< const Key< PdgCode > >, std::reference_wrapper< const Key< PseudoResonance > >, std::reference_wrapper< const Key< ReactionsBitSet > >, std::reference_wrapper< const Key< RestFrameDensityDerivativesMode > >, std::reference_wrapper< const Key< Sampling > >, std::reference_wrapper< const Key< SmearingMode > >, std::reference_wrapper< const Key< SphereInitialCondition > >, std::reference_wrapper< const Key< ThermalizationAlgorithm > >, std::reference_wrapper< const Key< TimeStepMode > >, std::reference_wrapper< const Key< TotalCrossSectionStrategy > >> key_references_variant
Alias for the type to be used in the list of keys.
A utility type to be specialized to check if a type is a std::map .