10 #ifndef SRC_INCLUDE_SMASH_CONFIGURATION_H_
11 #define SRC_INCLUDE_SMASH_CONFIGURATION_H_
26 #include "yaml-cpp/yaml.h"
49 static Node
encode(
const T &x) {
return Node{
static_cast<std::string
>(x)}; }
59 static bool decode(
const Node &node, T &x) {
60 if (!node.IsScalar()) {
63 x =
static_cast<T
>(node.Scalar());
291 using std::runtime_error::runtime_error;
298 using std::runtime_error::runtime_error;
305 using std::runtime_error::runtime_error;
312 using std::logic_error::logic_error;
347 const std::filesystem::path &filename);
361 throw std::runtime_error(
362 "Unknown control flag in Configuration constructor"
363 " with a YAML formatted string. Please, use"
364 " Configuration::InitializeFromYAMLString.");
499 template <typename T>
503 return take({key.labels().begin(), key.labels().end()});
507 return take({key.labels().begin(), key.labels().end()});
509 throw std::logic_error(
510 "Key " + std::string{key} +
511 " was taken, but its value is not a map, although there is a "
512 "section in the configuration with its labels.");
520 return key.default_value();
521 }
catch (std::bad_optional_access &) {
522 throw std::invalid_argument(
523 "Key " + std::string{key} +
524 " without default value taken, but missing in configuration.");
542 template <
typename T>
545 throw std::logic_error(
546 "An input Key without dependent default cannot be taken specifying a "
547 "default value! Either define the key as having a dependent default "
548 "or take it without a default value (which is a Key property).");
553 return default_value;
578 template <
typename T>
588 throw std::logic_error(
589 "Key " + std::string{key} +
590 " was read, but its value is not a map, although there is a "
591 "section in the configuration with its labels.");
596 }
catch (std::bad_optional_access &) {
597 throw std::invalid_argument(
598 "Key " + std::string{key} +
599 " without default value read, but missing in configuration.");
617 template <
typename T>
620 throw std::logic_error(
621 "An input Key without dependent default cannot be read specifying a "
622 "default value! Either define the key as having a dependent default "
623 "or read it without a default value (which is a Key property).");
628 return default_value;
653 template <
typename T,
typename U = remove_cvref_t<T>,
654 typename std::enable_if_t<std::is_convertible_v<T, U>,
bool> = true>
658 node = std::forward<T>(value);
723 template <
typename T>
725 const auto found_node =
727 return found_node.has_value() && !(found_node.value().IsMap());
737 template <
typename T>
739 const auto found_node =
741 return found_node.has_value() && !(found_node.value().IsNull()) &&
742 !(found_node.value().IsMap());
752 return found_node.has_value() && found_node.value().IsMap();
791 Is validate(
bool full_validation =
true)
const;
818 if (!(
n.IsScalar() ||
n.IsSequence() ||
n.IsMap())) {
819 std::stringstream err;
820 err <<
"Configuration value for \"" << key
821 <<
"\" is missing or invalid";
822 throw std::runtime_error(err.str());
840 template <
typename T>
843 return node_.as<T>();
844 }
catch (YAML::TypedBadConversion<T> &e) {
846 "The value for key \"" + std::string(
key_) +
847 "\" cannot be converted to the requested type.");
856 template <
typename T>
857 operator std::vector<T>()
const {
859 return node_.as<std::vector<T>>();
860 }
catch (YAML::TypedBadConversion<T> &e) {
862 "One of the values in the sequence for key \"" + std::string(
key_) +
863 "\" failed to convert to the requested type. E.g. [1 2] is a "
864 "sequence of one string \"1 2\" and [1, 2] is a sequence of two "
865 "integers. Often there is just a comma missing in the config "
867 }
catch (YAML::TypedBadConversion<std::vector<T>> &e) {
869 "The value for key \"" + std::string(
key_) +
870 "\" cannot be converted to the requested type. A sequence was "
871 "expected but apparently not found.");
882 template <
typename T,
size_t N>
883 operator std::array<T, N>()
const {
884 const std::vector<T> vec =
operator std::vector<T>();
885 const size_t n_read = vec.size();
889 std::string(
key_) +
"\". Expected " +
893 std::to_string(n_read) +
".");
895 std::array<T, N> arr;
896 std::copy_n(vec.begin(), N, arr.begin());
908 const std::vector<std::string> v =
operator std::vector<std::string>();
910 for (
const auto &x : v) {
914 }
else if (x ==
"Elastic") {
916 }
else if (x ==
"NN_to_NR") {
918 }
else if (x ==
"NN_to_DR") {
920 }
else if (x ==
"KN_to_KN") {
922 }
else if (x ==
"KN_to_KDelta") {
924 }
else if (x ==
"Strangeness_exchange") {
926 }
else if (x ==
"NNbar") {
928 }
else if (x ==
"PiDeuteron_to_NN") {
930 }
else if (x ==
"PiDeuteron_to_pidprime") {
932 }
else if (x ==
"NDeuteron_to_Ndprime") {
936 "The value for key \"" + std::string(
key_) +
937 "\" should be \"All\", \"Elastic\", \"NN_to_NR\", \"NN_to_DR\","
938 "\"KN_to_KN\", \"KN_to_KDelta\", \"PiDeuteron_to_NN\", "
939 "\"PiDeuteron_to_pidprime\", \"NDeuteron_to_Ndprime\", "
940 "\"Strangeness_exchange\" or "
941 "\"NNbar\", or any combination of these.");
955 const std::vector<std::string> v =
operator std::vector<std::string>();
957 for (
const auto &x : v) {
961 }
else if (x ==
"Meson_3to1") {
963 }
else if (x ==
"Deuteron_3to2") {
965 }
else if (x ==
"NNbar_5to2") {
967 }
else if (x ==
"A3_Nuclei_4to2") {
971 "The value for key \"" + std::string(
key_) +
972 "\" should be \"All\", \"Meson_3to1\", "
973 "\"Deuteron_3to2\" or \"NNbar_5to2\", "
974 "\"A3_Nuclei_4to2\", or any combination of "
988 operator std::set<ThermodynamicQuantity>()
const {
989 const std::vector<std::string> v =
operator std::vector<std::string>();
990 std::set<ThermodynamicQuantity> s;
991 for (
const auto &x : v) {
992 if (x ==
"rho_eckart") {
994 }
else if (x ==
"tmn") {
996 }
else if (x ==
"tmn_landau") {
998 }
else if (x ==
"landau_velocity") {
1000 }
else if (x ==
"j_QBS") {
1004 "The value for key \"" + std::string(
key_) +
1005 "\" should be \"rho_eckart\", \"tmn\""
1006 ", \"tmn_landau\", \"landau_velocity\" or \"j_QBS\".");
1020 const std::string s =
operator std::string();
1021 if (s ==
"center of velocity") {
1024 if (s ==
"center of mass") {
1027 if (s ==
"fixed target") {
1031 "The value for key \"" + std::string(
key_) +
1032 "\" should be \"center of velocity\" or \"center of mass\" "
1033 "or \"fixed target\".");
1044 const std::string s =
operator std::string();
1051 if (s ==
"frozen") {
1055 "The value for key \"" + std::string(
key_) +
1056 "\" should be \"off\" or \"on\" or \"frozen\".");
1067 const std::string s =
operator std::string();
1068 if (s ==
"hadron") {
1071 if (s ==
"baryon") {
1074 if (s ==
"baryonic isospin") {
1080 if (s ==
"total isospin") {
1088 "\" should be \"hadron\" or \"baryon\" "
1089 "or \"baryonic isospin\" or \"pion\" "
1101 const std::string s =
operator std::string();
1102 if (s ==
"NoExpansion") {
1105 if (s ==
"MasslessFRW") {
1108 if (s ==
"MassiveFRW") {
1111 if (s ==
"Exponential") {
1115 "The value for key \"" + std::string(
key_) +
1116 "\" should be \"NoExpansion\", \"MasslessFRW\"," +
1117 "\"MassiveFRW\" or \"Exponential\".");
1124 const std::string s =
operator std::string();
1125 if (s ==
"Covariant Gaussian") {
1128 if (s ==
"Finite difference") {
1135 "The value for key \"" + std::string(
key_) +
1136 "\" should be \"Covariant Gaussian\", \"Finite difference\"," +
1144 const std::string s =
operator std::string();
1145 if (s ==
"Chain Rule") {
1148 if (s ==
"Direct") {
1152 "The value for key \"" + std::string(
key_) +
1153 "\" should be \"Chain Rule\" or \"Direct\".");
1160 const std::string s =
operator std::string();
1161 if (s ==
"Covariant Gaussian") {
1164 if (s ==
"Discrete") {
1167 if (s ==
"Triangular") {
1171 "The value for key \"" + std::string(
key_) +
1172 "\" should be \"Covariant Gaussian\", \"Discrete\"," +
1173 " or \"Triangular\".");
1184 const std::string s =
operator std::string();
1193 "\" should be \"None\" or \"Fixed\".");
1204 const std::string s =
operator std::string();
1205 if (s ==
"thermal momenta") {
1208 if (s ==
"thermal momenta quantum") {
1211 if (s ==
"peaked momenta") {
1215 "The value for key \"" + std::string(
key_) +
1216 "\" should be \"thermal momenta\", \"thermal momenta quantum\", " +
1217 "or \"peaked momenta\".");
1228 const std::string s =
operator std::string();
1229 if (s ==
"thermal momenta") {
1232 if (s ==
"thermal momenta quantum") {
1244 if (s ==
"IC_Massive") {
1248 "The value for key \"" + std::string(
key_) +
1249 "\" should be \"thermal momenta\", \"thermal momenta quantum\", " +
1250 "\"IC_ES\", \"IC_1M\", \"IC_2M\" or" +
"\"IC_Massive\".");
1261 const std::string s =
operator std::string();
1262 if (s ==
"no annihilation") {
1265 if (s ==
"resonances") {
1268 if (s ==
"two to five") {
1271 if (s ==
"strings") {
1275 "The value for key \"" + std::string(
key_) +
"\" should be " +
1276 "\"no annihilation\", \"resonances\", \"two to five\" or " +
1288 const std::string s =
operator std::string();
1289 if (s ==
"quadratic") {
1292 if (s ==
"custom") {
1295 if (s ==
"uniform") {
1299 "The value for key \"" + std::string(
key_) +
1300 "\" should be \"quadratic\", \"uniform\" or \"custom\".");
1311 const std::string s =
operator std::string();
1312 if (s ==
"mode sampling") {
1315 if (s ==
"biased BF") {
1318 if (s ==
"unbiased BF") {
1322 "The value for key \"" + std::string(
key_) +
1323 "\" should be \"mode sampling\", \"biased BF\" or \"unbiased BF\".");
1334 const std::string s =
operator std::string();
1335 if (s ==
"Geometric") {
1338 if (s ==
"Stochastic") {
1341 if (s ==
"Covariant") {
1345 "The value for key \"" + std::string(
key_) +
"\" should be " +
1346 "\"Geometric\", \"Stochastic\" " +
"or \"Covariant\".");
1357 const std::string s =
operator std::string();
1358 if (s ==
"BottomUp") {
1361 if (s ==
"TopDown") {
1364 if (s ==
"TopDownMeasured") {
1368 "The value for key \"" + std::string(
key_) +
"\" should be " +
1369 "\"BottomUp\", \"TopDown\" " +
"or \"TopDownMeasured\".");
1380 const std::string s =
operator std::string();
1384 if (s ==
"Largest") {
1387 if (s ==
"Closest") {
1390 if (s ==
"LargestFromUnstable") {
1393 if (s ==
"ClosestFromUnstable") {
1397 "The value for key \"" + std::string(
key_) +
"\" should be " +
1398 "\"None\", \"Largest\", \"Closest\", \"LargestFromUnstable\", or "
1399 "\"ClosestFromUnstable\".");
1411 const std::string s =
operator std::string();
1412 if (s ==
"Constant_Tau") {
1414 }
else if (s ==
"Dynamic") {
1418 std::string(
key_) +
"\" should be " +
1419 "\"Constant_Tau\" or \"Dynamic\".");
1430 const std::string s =
operator std::string();
1437 if (s ==
"IfNotEmpty") {
1441 std::string(
key_) +
"\" should be " +
1442 "\"Yes\", \"No\" or \"IfNotEmpty\".");
1453 const std::vector<std::string> v =
operator std::vector<std::string>();
1455 for (
const auto &x : v) {
1459 }
else if (x ==
"Elastic") {
1461 }
else if (x ==
"Decay") {
1463 }
else if (x ==
"Inelastic") {
1465 }
else if (x ==
"SoftString") {
1467 }
else if (x ==
"HardString") {
1471 "The value for key \"" + std::string(
key_) +
1472 "\" should be \"All\", \"Elastic\", \"Decay\", "
1473 "\"Inelastic\", \"SoftString\", \"HardString\", "
1474 "or any combination of these.");
1503 std::vector<std::string_view> keys)
const;
1528 std::vector<std::string_view> keys)
const;
1538 template <
class Key,
class Value>
1539 struct isMap<std::map<Key, Value>> : std::true_type {};
1552 Value take(std::vector<std::string_view> labels);
1565 Value read(std::vector<std::string_view> labels)
const;
Proxy object to be used when taking or reading keys in the configuration.
Value(const YAML::Node &n, const char *key)
Construct the Value wrapper from a YAML::Node.
Value(const Value &)=delete
If you want to copy this you're doing it wrong.
const char *const key_
The key to be interpreted.
const YAML::Node node_
a YAML leaf node
Value & operator=(const Value &)=delete
If you want to copy this you're doing it wrong.
Interface to the SMASH configuration files.
Is
Return type of Configuration::validate which conveys more information that simply a two-state boolean...
T take(const Key< T > &key, T default_value)
Alternative method to take a key value, specifying the default value.
void set_value(Key< U > key, T &&value)
Overwrite the value of the YAML node corresponding to the specified key.
Configuration(const Configuration &)=delete
Prevent Configuration objects from being copied.
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.
Configuration(const YAML::Node &node)
Create a sub-object that has its root node at the given node.
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.
Configuration(const char *yaml)
bool has_value(const Key< T > &key) const
Return whether there is a non-empty value behind the requested key (which is supposed not to refer to...
void clear()
Erase the Configuration content.
Configuration(const char *yaml, const char sflag)
Initialize configuration with a YAML formatted string.
T read(const Key< T > &key, T default_value)
Alternative method to read a key value, specifying the default value.
bool has_key(const Key< T > &key) const
Return whether the configuration has a (possibly empty) non-map key.
bool has_section(const KeyLabels &labels) const
Return whether there is a (possibly empty) section with the given labels.
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.
static const char InitializeFromYAMLString
Flag to mark initialization with a YAML formatted string.
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.
Object to store a YAML input file key together with metadata associated to it.
bool has_dependent_default() const noexcept
Ask whether the default value depends on other other keys.
default_type default_value() const
Get the default value of the key.
const KeyLabels & labels() const
Method to access the Key labels.
default_type type
Let the clients of this class have access to the key type.
SmearingMode
Modes of smearing.
FermiMotion
Option to use Fermi Motion.
@ On
Use fermi motion in combination with potentials.
@ Frozen
Use fermi motion without potentials.
@ Off
Don't use fermi motion.
std::bitset< 10 > ReactionsBitSet
Container for the 2 to 2 reactions in the code.
ThermalizationAlgorithm
Defines the algorithm used for the forced thermalization.
FluidizationType
Possible methods to convert SMASH particle into fluid cells.
@ ConstantTau
Hypersurface crossed at a fixed proper time.
@ Dynamic
Dynamic fluidization based on local densities.
NNbarTreatment
Treatment of N Nbar Annihilation.
@ NoAnnihilation
No Annihilation.
@ TwoToFive
Directly create 5 pions, use with multi-particle reactions.
@ Resonances
Use intermediate Resonances.
@ Strings
Use string fragmentation.
TimeStepMode
The time step mode.
@ Fixed
Use fixed time step.
@ None
Don't use time steps; propagate from action to action.
@ EckartDensity
Density in the Eckart frame.
@ Tmn
Energy-momentum tensor in lab frame.
@ LandauVelocity
Velocity of the Landau rest frame.
@ j_QBS
Electric (Q), baryonic (B) and strange (S) currents.
@ TmnLandau
Energy-momentum tensor in Landau rest frame.
std::bitset< 5 > FluidizableProcessesBitSet
TotalCrossSectionStrategy
Determine how total cross sections for collision finding should be computed.
@ TopDownMeasured
Mix the two above, using the parametrizations only for measured processes, and summing up partials fo...
@ TopDown
Use parametrizations based on existing data, rescaling with AQM for unmeasured processes.
@ BottomUp
Sum the existing partial contributions.
Sampling
Possible methods of impact parameter sampling.
@ Quadratic
Sample from areal / quadratic distribution.
@ Custom
Sample from custom, user-defined distribution.
@ Uniform
Sample from uniform distribution.
std::bitset< 4 > MultiParticleReactionsBitSet
Container for the n to m reactions in the code.
CalculationFrame
The calculation frame.
DerivativesMode
Modes of calculating the gradients.
CollisionCriterion
Criteria used to check collisions.
@ Stochastic
Stochastic Criteiron.
@ Geometric
Geometric criterion.
@ Covariant
Covariant Criterion.
FieldDerivativesMode
Modes of calculating the field gradients: chain rule or direct.
SphereInitialCondition
Initial condition for a particle in a sphere.
@ ThermalMomentaBoltzmann
A thermalized ensemble is generated, with momenta sampled from a Maxwell-Boltzmann distribution.
@ IC_ES
Off-equilibrium distribution used in massless comparisons of SMASH to the extended universe metric.
@ ThermalMomentaQuantum
A thermalized ensemble is generated, with momenta of baryons(mesons) sampled from a Fermi(Bose) distr...
@ IC_Massive
A generalization of IC_ES for the non-zero mass case; note that there is currently no analytical comp...
@ IC_2M
Off-equilibrium distribution used in massless comparisons of SMASH to the extended universe metric.
@ IC_1M
Off-equilibrium distribution used in massless comparisons of SMASH to the extended universe metric.
PseudoResonance
Which pseudo-resonance fills the inelastic gap in the transition to string region of cross sections.
@ Closest
Resonance with the pole mass closest from the invariant mass of incoming particles for all processes.
@ ClosestFromUnstable
Closest resonance for a given mass from processes with at least one resonance in the incoming particl...
@ None
No pseudo-resonance is created.
@ LargestFromUnstable
Heaviest possible resonance from processes with at least one resonance in the incoming particles.
@ Largest
Resonance of largest mass for all processes.
DensityType
Allows to choose which kind of density to calculate.
OutputOnlyFinal
Whether and when only final state particles should be printed.
@ IfNotEmpty
Print only final-state particles, and those only if the event is not empty.
@ Yes
Print only final-state particles.
@ No
Print initial, intermediate and final-state particles.
BoxInitialCondition
Initial condition for a particle in a box.
@ ThermalMomentaBoltzmann
A thermalized ensemble is generated, with momenta sampled from a Maxwell-Boltzmann distribution.
@ ThermalMomentaQuantum
A thermalized ensemble is generated, with momenta of baryons(mesons) sampled from a Fermi(Bose) distr...
@ PeakedMomenta
All particles have the same momentum with T being the temperature.
ExpansionMode
Defines properties of expansion for the metric (e.g.
std::vector< std::string > KeyLabels
Descriptive alias for storing key labels, i.e.
Convert from YAML::Node to SMASH-readable (C++) format and vice versa.
static bool decode(const Node &node, T &x)
Deserialization: Converts a YAML::Node to any SMASH-readable data type and returns whether or not thi...
static Node encode(const T &x)
Serialization: Converts x (of any type) to a YAML::Node.
Thrown if the file does not exist.
Thrown when the types in the config file and C++ don't match.
Thrown for YAML parse errors.
Thrown if a Key is taken twice.
Utility type trait (general case) for the take and read public methods.