Version: SMASH-3.2
key.h
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #ifndef SRC_INCLUDE_SMASH_KEY_H_
11 #define SRC_INCLUDE_SMASH_KEY_H_
12 
13 #include <optional>
14 #include <sstream>
15 #include <string>
16 #include <string_view>
17 #include <utility>
18 #include <vector>
19 
20 #include "stringfunctions.h"
21 #include "traits.h"
22 
23 namespace smash {
24 
29 using Version = std::string;
30 
35 using KeyMetadata = std::initializer_list<std::string_view>;
36 
42 using KeyLabels = std::vector<std::string>;
43 
51 inline KeyLabels operator+(const KeyLabels& lhs, std::string_view rhs) {
52  if (lhs.empty()) {
53  return KeyLabels{std::string{rhs}};
54  } else {
55  KeyLabels result{lhs};
56  result.push_back(std::string{rhs});
57  return result;
58  }
59 }
60 
68 inline KeyLabels operator+(KeyLabels&& lhs, std::string_view rhs) {
69  if (lhs.empty()) {
70  return KeyLabels{std::string{rhs}};
71  } else {
72  lhs.push_back(std::string{rhs});
73  return std::move(lhs); // See https://stackoverflow.com/a/14857144
74  }
75 }
76 
80 enum class DefaultType {
82  Null,
84  Value,
86  Dependent
87 };
88 
103 template <typename default_type>
104 class Key {
105  static_assert(!std::is_const_v<default_type>);
106  static_assert(!std::is_volatile_v<default_type>);
107  static_assert(!std::is_reference_v<default_type>);
108  static_assert(!std::is_pointer_v<default_type>);
109  static_assert(!std::is_array_v<default_type>);
110  static_assert(!std::is_function_v<default_type>);
111  static_assert(!std::is_member_object_pointer_v<default_type>);
112  static_assert(!std::is_member_function_pointer_v<default_type>);
113 
114  public:
119  struct WrongNumberOfVersions : public std::runtime_error {
120  using std::runtime_error::runtime_error;
121  };
122 
131  explicit Key(const KeyLabels& labels, const KeyMetadata& versions)
132  : Key{labels, Default<default_type>{}, versions} {}
133 
145  Key(const KeyLabels& labels, default_type value, const KeyMetadata& versions)
146  : Key{labels, Default<default_type>{value}, versions} {}
147 
161  Key(const KeyLabels& labels, DefaultType type_of_default,
162  const KeyMetadata& versions)
163  : Key{labels, Default<default_type>{type_of_default}, versions} {}
164 
168  using type = default_type;
169 
177  default_type default_value() const { return default_.value(); }
178 
185  bool has_dependent_default() const noexcept {
186  return default_.is_dependent();
187  }
188 
194  Version introduced_in() const noexcept { return introduced_in_; }
195 
203  Version deprecated_in() const { return deprecated_in_.value(); }
204 
212  Version removed_in() const { return removed_in_.value(); }
213 
219  bool is_deprecated() const noexcept { return deprecated_in_.has_value(); }
220 
226  bool is_allowed() const noexcept { return !removed_in_.has_value(); }
227 
236  bool has_same_labels(const KeyLabels& labels) const noexcept {
237  return std::equal(std::begin(labels_), std::end(labels_),
238  std::begin(labels), std::end(labels));
239  }
240 
247  explicit operator std::string() const noexcept {
248  return smash::quote(smash::join(labels_, ": "));
249  }
250 
262  std::string as_yaml(
263  std::optional<default_type> value = std::nullopt) const noexcept {
264  std::stringstream value_as_string{};
265  if constexpr (is_writable_to_stream_v<std::stringstream, default_type>) {
266  if (value) {
267  value_as_string << *value;
268  } else if (default_.type_ == DefaultType::Value) {
269  value_as_string << default_value();
270  }
271  }
272  return as_yaml(value_as_string.str());
273  }
274 
285  std::string as_yaml(std::string value) const noexcept {
286  std::stringstream result{};
287  result << "{" << smash::join(labels_, ": {") << ": " << value
288  << smash::join(std::vector<std::string>(labels_.size(), "}"), "");
289  return result.str();
290  }
291 
297  const KeyLabels& labels() const { return labels_; }
298 
299  private:
320  template <typename T>
321  class Default {
322  public:
327  Default() : type_{DefaultType::Null} {}
333  explicit Default(T in) : value_{std::move(in)} {}
344  explicit Default(DefaultType type) : type_{type} {
345  if (type != DefaultType::Dependent) {
346  throw std::logic_error("Default constructor used with invalid type!");
347  }
348  }
349 
357  T value() const { return value_.value(); }
358 
366  bool is_dependent() const noexcept {
367  return type_ == DefaultType::Dependent;
368  }
369 
370  private:
372  DefaultType type_ = DefaultType::Value;
374  std::optional<T> value_ = std::nullopt;
375  // Make nested class friend of enclosing one. This class is anyhow an
376  // implementation detail and part of Key.
377  friend class Key<T>;
378  };
379 
390  Key(const KeyLabels& labels, Default<default_type> value,
391  const KeyMetadata& versions)
392  : default_{std::move(value)}, labels_{labels.begin(), labels.end()} {
393  /*
394  * The following switch statement is a compact way to initialize the
395  * three version member variables without repetition and lots of logic
396  * clauses. The versions variable can have 1, 2 or 3 entries. The use of
397  * the iterator is needed, since std::initializer_list has no access
398  * operator.
399  */
400  switch (auto it = versions.end(); versions.size()) {
401  case 3:
402  removed_in_ = *(--it);
403  [[fallthrough]];
404  case 2:
405  deprecated_in_ = *(--it);
406  [[fallthrough]];
407  case 1:
408  introduced_in_ = *(--it);
409  break;
410  default:
411  throw WrongNumberOfVersions(
412  "Key constructor needs one, two or three version numbers.");
413  }
414  }
415 
417  Version introduced_in_{};
419  std::optional<Version> deprecated_in_{};
421  std::optional<Version> removed_in_{};
425  KeyLabels labels_{};
426 };
427 
428 } // namespace smash
429 
430 #endif // SRC_INCLUDE_SMASH_KEY_H_
Wrapper class around a type with the capability to both store the type of default and its value,...
Definition: key.h:321
T value() const
Retrieve the default value stored in the object.
Definition: key.h:357
Default(T in)
Construct a new Default object storing its default value.
Definition: key.h:333
Default()
Construct a new Default object which denotes a mandatory value without a default.
Definition: key.h:327
Default(DefaultType type)
Construct a new Default object which has a value dependent on external information.
Definition: key.h:344
bool is_dependent() const noexcept
Ask whether the default value depends on other external information.
Definition: key.h:366
Object to store a YAML input file key together with metadata associated to it.
Definition: key.h:104
bool has_dependent_default() const noexcept
Ask whether the default value depends on other other keys.
Definition: key.h:185
bool is_allowed() const noexcept
Get whether the key is still allowed or not.
Definition: key.h:226
bool has_same_labels(const KeyLabels &labels) const noexcept
Check if given labels are the same as those of this object.
Definition: key.h:236
std::string as_yaml(std::string value) const noexcept
Overload of the method taking a string as value.
Definition: key.h:285
Version deprecated_in() const
Get the SMASH version in which the key has been deprecated.
Definition: key.h:203
Key(const KeyLabels &labels, const KeyMetadata &versions)
Construct a new Key object without default value.
Definition: key.h:131
Version removed_in() const
Get the SMASH version in which the key has been removed.
Definition: key.h:212
Key(const KeyLabels &labels, DefaultType type_of_default, const KeyMetadata &versions)
Construct a new Key object which is supposed to have a default value, which however depends on other ...
Definition: key.h:161
bool is_deprecated() const noexcept
Get whether the key is deprecated or not.
Definition: key.h:219
default_type default_value() const
Get the default value of the key.
Definition: key.h:177
const KeyLabels & labels() const
Method to access the Key labels.
Definition: key.h:297
Key(const KeyLabels &labels, default_type value, const KeyMetadata &versions)
Construct a new Key object with default value.
Definition: key.h:145
std::string as_yaml(std::optional< default_type > value=std::nullopt) const noexcept
Build and return a YAML-formatted string in the compact form (using braces as single line).
Definition: key.h:262
default_type type
Let the clients of this class have access to the key type.
Definition: key.h:168
Key(const KeyLabels &labels, Default< default_type > value, const KeyMetadata &versions)
Private constructor of the Key object.
Definition: key.h:390
Version introduced_in() const noexcept
Get the SMASH version in which the key has been introduced.
Definition: key.h:194
Definition: action.h:24
std::vector< std::string > KeyLabels
Descriptive alias for storing key labels, i.e.
Definition: key.h:42
DefaultType
New type to explicit distinguish between mandatory and optional keys.
Definition: key.h:80
@ Dependent
Default value which depends on other keys
@ Value
Normal default with a value associated to it.
@ Null
Default "type" for mandatory keys
std::string quote(const std::string &s)
Add quotes around string.
std::string join(const std::vector< std::string > &v, const std::string &delim)
Join strings using delimiter.
std::initializer_list< std::string_view > KeyMetadata
Descriptive alias for storing keys metadata.
Definition: key.h:35
EnergyMomentumTensor operator+(EnergyMomentumTensor a, const EnergyMomentumTensor &b)
Direct addition operator.
std::string Version
Descriptive alias for storing a SMASH version associated to keys metadata.
Definition: key.h:29
Thrown when too few or too many versions are passed to the constructor.
Definition: key.h:119