Version: SMASH-3.2
outputformatter.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_OUTPUTFORMATTER_H_
11 #define SRC_INCLUDE_SMASH_OUTPUTFORMATTER_H_
12 
13 #include <functional>
14 #include <map>
15 #include <sstream>
16 #include <string>
17 #include <vector>
18 
19 #include "smash/particledata.h"
20 
21 namespace smash {
22 
27 struct ToASCII {
29  using type = std::string;
30 
36  type as_integer(int value) const { return std::to_string(value); }
37 
54  type as_double(double value) {
55  constexpr size_t kBufferSize = 13;
56  char buffer[kBufferSize];
57  const auto length = std::snprintf(buffer, kBufferSize, "%g", value);
58  assert(static_cast<size_t>(length) < kBufferSize);
59  assert(length > 0);
60  return std::string{buffer, buffer + length};
61  }
62 
77  type as_precise_double(double value) {
78  constexpr size_t kBufferSize = 16;
79  char buffer[kBufferSize];
80  const auto length = std::snprintf(buffer, kBufferSize, "%.9g", value);
81  assert(static_cast<size_t>(length) < kBufferSize);
82  assert(length > 0);
83  return std::string{buffer, buffer + length};
84  }
85 
92  type as_string(const std::string& str) const { return str; }
93 };
94 
99 class ToBinary {
100  public:
102  using type = std::vector<char>;
103 
110  type as_integer(int value) const { return as_binary_data(value); }
111 
118  type as_double(double value) const { return as_binary_data(value); }
119 
129  type as_precise_double(double value) const { return as_double(value); }
130 
137  type as_string(const std::string& str) const {
138  type binary_data(str.begin(), str.end());
139  return binary_data;
140  }
141 
142  private:
149  template <typename T>
150  type as_binary_data(T value) const {
151  type binary_data(sizeof(T));
152  std::memcpy(binary_data.data(), &value, sizeof(T));
153  return binary_data;
154  }
155 };
156 
175 template <typename Converter,
176  std::enable_if_t<std::is_same_v<Converter, ToASCII> ||
177  std::is_same_v<Converter, ToBinary>,
178  bool> = true>
180  public:
194  explicit OutputFormatter(const std::vector<std::string>& in_quantities)
195  : quantities_(in_quantities) {
197  for (const std::string& quantity : quantities_) {
198  if (quantity == "t") {
199  getters_.push_back([this](const ParticleData& in) {
200  return this->converter_.as_double(in.position()[0]);
201  });
202  } else if (quantity == "x") {
203  getters_.push_back([this](const ParticleData& in) {
204  return this->converter_.as_double(in.position()[1]);
205  });
206  } else if (quantity == "y") {
207  getters_.push_back([this](const ParticleData& in) {
208  return this->converter_.as_double(in.position()[2]);
209  });
210  } else if (quantity == "z") {
211  getters_.push_back([this](const ParticleData& in) {
212  return this->converter_.as_double(in.position()[3]);
213  });
214  } else if (quantity == "mass") {
215  getters_.push_back([this](const ParticleData& in) {
216  return this->converter_.as_double(in.effective_mass());
217  });
218  } else if (quantity == "p0") {
219  getters_.push_back([this](const ParticleData& in) {
220  return this->converter_.as_precise_double(in.momentum()[0]);
221  });
222  } else if (quantity == "px") {
223  getters_.push_back([this](const ParticleData& in) {
224  return this->converter_.as_precise_double(in.momentum()[1]);
225  });
226  } else if (quantity == "py") {
227  getters_.push_back([this](const ParticleData& in) {
228  return this->converter_.as_precise_double(in.momentum()[2]);
229  });
230  } else if (quantity == "pz") {
231  getters_.push_back([this](const ParticleData& in) {
232  return this->converter_.as_precise_double(in.momentum()[3]);
233  });
234  } else if (quantity == "pdg") {
235  getters_.push_back([this](const ParticleData& in) {
236  return this->converter_.as_integer(in.pdgcode().get_decimal());
237  });
238  } else if (quantity == "ID" || quantity == "id") {
239  getters_.push_back([this](const ParticleData& in) {
240  return this->converter_.as_integer(in.id());
241  });
242  } else if (quantity == "charge") {
243  getters_.push_back([this](const ParticleData& in) {
244  return this->converter_.as_integer(in.type().charge());
245  });
246  } else if (quantity == "ncoll") {
247  getters_.push_back([this](const ParticleData& in) {
248  return this->converter_.as_integer(
250  });
251  } else if (quantity == "form_time") {
252  getters_.push_back([this](const ParticleData& in) {
253  return this->converter_.as_double(in.formation_time());
254  });
255  } else if (quantity == "xsecfac") {
256  getters_.push_back([this](const ParticleData& in) {
257  return this->converter_.as_double(in.xsec_scaling_factor());
258  });
259  } else if (quantity == "proc_id_origin") {
260  getters_.push_back([this](const ParticleData& in) {
261  return this->converter_.as_integer(in.get_history().id_process);
262  });
263  } else if (quantity == "proc_type_origin") {
264  getters_.push_back([this](const ParticleData& in) {
265  return this->converter_.as_integer(
266  static_cast<int>(in.get_history().process_type));
267  });
268  } else if (quantity == "time_last_coll") {
269  getters_.push_back([this](const ParticleData& in) {
270  return this->converter_.as_double(
272  });
273  } else if (quantity == "pdg_mother1") {
274  getters_.push_back([this](const ParticleData& in) {
275  return this->converter_.as_integer(in.get_history().p1.get_decimal());
276  });
277  } else if (quantity == "pdg_mother2") {
278  getters_.push_back([this](const ParticleData& in) {
279  return this->converter_.as_integer(in.get_history().p2.get_decimal());
280  });
281  } else if (quantity == "baryon_number") {
282  getters_.push_back([this](const ParticleData& in) {
283  return this->converter_.as_integer(in.pdgcode().baryon_number());
284  });
285  } else if (quantity == "strangeness") {
286  getters_.push_back([this](const ParticleData& in) {
287  return this->converter_.as_integer(in.pdgcode().strangeness());
288  });
289  } else if (quantity == "0") { // for OSCAR1999
290  getters_.push_back([this]([[maybe_unused]] const ParticleData& in) {
291  return this->converter_.as_integer(0);
292  });
293  }
294  }
295  }
303  typename Converter::type binary_chunk(const ParticleData& p) {
304  return std::accumulate(
305  std::begin(getters_), std::end(getters_), std::vector<char>{},
306  [&p](std::vector<char> ss, const auto& getter) {
307  auto binary_data = getter(p);
308  ss.insert(ss.end(), binary_data.begin(), binary_data.end());
309  return ss;
310  });
311  }
312 
319  typename Converter::type data_line(const ParticleData& p) const {
320  return std::accumulate(
321  std::begin(getters_), std::end(getters_), std::string{},
322  [&p](const std::string& ss, const auto& getter) {
323  return ss.empty() ? getter(p) : ss + " " + getter(p);
324  });
325  }
326 
331  typename Converter::type quantities_line() const {
332  return std::accumulate(
333  std::begin(quantities_), std::end(quantities_), std::string{},
334  [this](const std::string& ss, const std::string& s) {
335  return ss.empty() ? this->converter_.as_string(s)
336  : ss + " " + this->converter_.as_string(s);
337  });
338  }
339 
344  typename Converter::type unit_line() const {
345  return std::accumulate(
346  std::begin(quantities_), std::end(quantities_), std::string{},
347  [this](const std::string& ss, const std::string& s) {
348  return ss.empty()
349  ? this->converter_.as_string(units_.at(s))
350  : ss + " " + this->converter_.as_string(units_.at(s));
351  });
352  }
353 
354  private:
356  Converter converter_{};
357 
359  std::vector<std::string> quantities_{};
360 
362  std::vector<std::function<typename Converter::type(const ParticleData&)>>
364 
366  const std::map<std::string, std::string> units_ = {
367  {"t", "fm"},
368  {"x", "fm"},
369  {"y", "fm"},
370  {"z", "fm"},
371  {"mass", "GeV"},
372  {"p0", "GeV"},
373  {"px", "GeV"},
374  {"py", "GeV"},
375  {"pz", "GeV"},
376  {"pdg", "none"},
377  {"ID", "none"},
378  {"id", "none"}, // used in OSCAR1999
379  {"charge", "e"},
380  {"ncoll", "none"},
381  {"form_time", "fm"},
382  {"xsecfac", "none"},
383  {"proc_id_origin", "none"},
384  {"proc_type_origin", "none"},
385  {"time_last_coll", "fm"},
386  {"pdg_mother1", "none"},
387  {"pdg_mother2", "none"},
388  {"baryon_number", "none"},
389  {"strangeness", "none"},
390  {"0", "0"}}; // for OSCAR1999
391 
394  if (quantities_.empty()) {
395  throw std::invalid_argument(
396  "OutputFormatter: Empty quantities handed over to the class.");
397  }
398  std::string error_message{};
399  std::string repeated{};
400  for (const std::string& quantity : quantities_) {
401  if (std::count(quantities_.begin(), quantities_.end(), quantity) > 1) {
402  repeated += "'" + quantity + "',";
403  }
404  }
405  if (!repeated.empty()) {
406  error_message += "Repeated \"Quantities\": " + repeated +
407  " please fix the configuration file.\n";
408  }
409  std::string unknown{};
410  for (const std::string& quantity : quantities_) {
411  if (units_.count(quantity) == 0) {
412  unknown += "'" + quantity + "',";
413  }
414  }
415  if (!unknown.empty()) {
416  error_message += "Unknown \"Quantities\": " + unknown +
417  " please fix the configuration file.\n";
418  }
419  if (!repeated.empty() || !unknown.empty())
420  throw std::invalid_argument(error_message);
421 
422  const bool oscar1999_id_is_given =
423  std::find(quantities_.begin(), quantities_.end(), "id") !=
424  quantities_.end();
425  const bool oscar2013_id_is_given =
426  std::find(quantities_.begin(), quantities_.end(), "ID") !=
427  quantities_.end();
428  if (oscar1999_id_is_given && oscar2013_id_is_given) {
429  throw std::invalid_argument(
430  "Both 'id' and 'ID' cannot be provided in the \"Quantities\" key "
431  "together. Please, fix the configuration file.");
432  }
433  }
434 };
435 
436 } // namespace smash
437 
438 #endif // SRC_INCLUDE_SMASH_OUTPUTFORMATTER_H_
A general-purpose formatter for output, supporting both ASCII and binary formats.
Converter converter_
Member to convert data into the correct output format.
Converter::type quantities_line() const
Produces the line with quantities for the header of the output file.
std::vector< std::string > quantities_
List of quantities to be written.
std::vector< std::function< typename Converter::type(const ParticleData &)> > getters_
List of getters to extract output data from the ParticleData object.
Converter::type data_line(const ParticleData &p) const
Produces the line with formatted data for the body of the output file.
void validate_quantities()
Checks whether the quantities requested are known and unique.
OutputFormatter(const std::vector< std::string > &in_quantities)
Creates the formatter.
Converter::type unit_line() const
Produces the line with units for the header of the output file.
const std::map< std::string, std::string > units_
Map with known quantities and corresponding units.
Converter::type binary_chunk(const ParticleData &p)
Produces a chunk of binary representing a particle line for the output file.
ParticleData contains the dynamic information of a certain particle.
Definition: particledata.h:58
PdgCode pdgcode() const
Get the pdgcode of the particle.
Definition: particledata.h:87
const ParticleType & type() const
Get the type of the particle.
Definition: particledata.h:128
double xsec_scaling_factor(double delta_time=0.) const
Return the cross section scaling factor at a given time.
Definition: particledata.cc:86
const FourVector & momentum() const
Get the particle's 4-momentum.
Definition: particledata.h:158
double formation_time() const
Get the absolute formation time of the particle.
Definition: particledata.h:236
double effective_mass() const
Get the particle's effective mass.
Definition: particledata.cc:24
int32_t id() const
Get the id of the particle.
Definition: particledata.h:76
HistoryData get_history() const
Get history information.
Definition: particledata.h:139
const FourVector & position() const
Get the particle's position in Minkowski space.
Definition: particledata.h:204
int32_t charge() const
The charge of the particle.
Definition: particletype.h:189
int baryon_number() const
Definition: pdgcode.h:391
int strangeness() const
Definition: pdgcode.h:605
int32_t get_decimal() const
Definition: pdgcode.h:831
Structure to convert a given value into binary format, such that all methods return a std::vector<cha...
type as_binary_data(T value) const
Template method to convert numbers into binary format.
type as_double(double value) const
Converts a double to binary format.
type as_precise_double(double value) const
Converts a double to binary format, intended for precise representation.
std::vector< char > type
Return type of this converter.
type as_integer(int value) const
Converts an integer to binary format.
type as_string(const std::string &str) const
Converts a string to binary format.
constexpr int p
Proton.
Definition: action.h:24
double time_last_collision
Time of the last action (excluding walls), time of kinetic freeze_out for HBT analysis this time shou...
Definition: particledata.h:43
int32_t id_process
id of the last action
Definition: particledata.h:34
PdgCode p2
PdgCode of the second parent particles.
Definition: particledata.h:47
PdgCode p1
PdgCode of the first parent particles.
Definition: particledata.h:45
int32_t collisions_per_particle
Collision counter per particle, zero only for initially present particles.
Definition: particledata.h:32
ProcessType process_type
type of the last action
Definition: particledata.h:36
Structure to convert a given value into ASCII format, such that all methods return a std::string.
type as_integer(int value) const
Converts an integer.
std::string type
Return type of this converter.
type as_double(double value)
Converts a double with 6 digits of precision.
type as_precise_double(double value)
Converts a double with 9 digits of precision.
type as_string(const std::string &str) const
Because ToASCII converts into strings, this simply returns the string itself.