Version: SMASH-3.2
smash::OutputFormatter< Converter, > Class Template Reference

#include <outputformatter.h>

template<typename Converter, std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
class smash::OutputFormatter< Converter, >

A general-purpose formatter for output, supporting both ASCII and binary formats.

This class allows the output of particle data in a flexible and configurable manner, either in human-readable ASCII format or compact binary format. It uses a template parameter Converter to determine the desired output format, which must conform to the interface of either ToASCII or ToBinary.

New quantities can be added for output by:

  1. Adding their corresponding getter to the constructor, which extracts the value from a ParticleData instance.
  2. Adding the proper key-value pair to the units_ map, specifying the unit of the quantity.
Template Parameters
ConverterThe desired output format. At the moment it must be either ToASCII or ToBinary.

Definition at line 179 of file outputformatter.h.

Public Member Functions

 OutputFormatter (const std::vector< std::string > &in_quantities)
 Creates the formatter. More...
 
Converter::type binary_chunk (const ParticleData &p)
 Produces a chunk of binary representing a particle line for the output file. More...
 
Converter::type data_line (const ParticleData &p) const
 Produces the line with formatted data for the body of the output file. More...
 
Converter::type quantities_line () const
 Produces the line with quantities for the header of the output file. More...
 
Converter::type unit_line () const
 Produces the line with units for the header of the output file. More...
 

Private Member Functions

void validate_quantities ()
 Checks whether the quantities requested are known and unique. More...
 

Private Attributes

Converter converter_ {}
 Member to convert data into the correct output format. More...
 
std::vector< std::string > quantities_ {}
 List of quantities to be written. More...
 
std::vector< std::function< typename Converter::type(const ParticleData &)> > getters_ {}
 List of getters to extract output data from the ParticleData object. More...
 
const std::map< std::string, std::string > units_
 Map with known quantities and corresponding units. More...
 

Constructor & Destructor Documentation

◆ OutputFormatter()

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
smash::OutputFormatter< Converter, >::OutputFormatter ( const std::vector< std::string > &  in_quantities)
inlineexplicit

Creates the formatter.

This reduces the number of literal strings flying around in the codebase, and since this is called only once per output file, in the beginning of the run, there is almost no efficiency lost compared to having fixed strings.

Parameters
[in]in_quantitieslist of quantities to be output.
Exceptions
std::invalid_argumentif the list of quantities is empty
std::invalid_argumentif unknown quantities exist in the list
std::invalid_argumentif incompatible quantities exist in the list
std::invalid_argumentif there are repeated quantities

Definition at line 194 of file outputformatter.h.

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(
249  in.get_history().collisions_per_particle);
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(
271  in.get_history().time_last_collision);
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  }
Converter converter_
Member to convert data into the correct output format.
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.
void validate_quantities()
Checks whether the quantities requested are known and unique.

Member Function Documentation

◆ binary_chunk()

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
Converter::type smash::OutputFormatter< Converter, >::binary_chunk ( const ParticleData p)
inline

Produces a chunk of binary representing a particle line for the output file.

Parameters
[in]pParticle whose information is to be written.
Returns
vector of char of the formatted data.

Definition at line 303 of file outputformatter.h.

303  {
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  }
constexpr int p
Proton.

◆ data_line()

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
Converter::type smash::OutputFormatter< Converter, >::data_line ( const ParticleData p) const
inline

Produces the line with formatted data for the body of the output file.

Parameters
[in]pparticle whose information is to be written.
Returns
string with formatted data separated by a space.

Definition at line 319 of file outputformatter.h.

319  {
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  }

◆ quantities_line()

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
Converter::type smash::OutputFormatter< Converter, >::quantities_line ( ) const
inline

Produces the line with quantities for the header of the output file.

Returns
string with name of quantities separated by a space.

Definition at line 331 of file outputformatter.h.

331  {
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  }

◆ unit_line()

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
Converter::type smash::OutputFormatter< Converter, >::unit_line ( ) const
inline

Produces the line with units for the header of the output file.

Returns
string with units separated by a space.

Definition at line 344 of file outputformatter.h.

344  {
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  }
const std::map< std::string, std::string > units_
Map with known quantities and corresponding units.

◆ validate_quantities()

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
void smash::OutputFormatter< Converter, >::validate_quantities ( )
inlineprivate

Checks whether the quantities requested are known and unique.

Definition at line 393 of file outputformatter.h.

393  {
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  }

Member Data Documentation

◆ converter_

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
Converter smash::OutputFormatter< Converter, >::converter_ {}
private

Member to convert data into the correct output format.

Definition at line 356 of file outputformatter.h.

◆ quantities_

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
std::vector<std::string> smash::OutputFormatter< Converter, >::quantities_ {}
private

List of quantities to be written.

Definition at line 359 of file outputformatter.h.

◆ getters_

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
std::vector<std::function<typename Converter::type(const ParticleData&)> > smash::OutputFormatter< Converter, >::getters_ {}
private

List of getters to extract output data from the ParticleData object.

Definition at line 363 of file outputformatter.h.

◆ units_

template<typename Converter , std::enable_if_t< std::is_same_v< Converter, ToASCII >||std::is_same_v< Converter, ToBinary >, bool > = true>
const std::map<std::string, std::string> smash::OutputFormatter< Converter, >::units_
private
Initial value:
= {
{"t", "fm"},
{"x", "fm"},
{"y", "fm"},
{"z", "fm"},
{"mass", "GeV"},
{"p0", "GeV"},
{"px", "GeV"},
{"py", "GeV"},
{"pz", "GeV"},
{"pdg", "none"},
{"ID", "none"},
{"id", "none"},
{"charge", "e"},
{"ncoll", "none"},
{"form_time", "fm"},
{"xsecfac", "none"},
{"proc_id_origin", "none"},
{"proc_type_origin", "none"},
{"time_last_coll", "fm"},
{"pdg_mother1", "none"},
{"pdg_mother2", "none"},
{"baryon_number", "none"},
{"strangeness", "none"},
{"0", "0"}}

Map with known quantities and corresponding units.

Definition at line 366 of file outputformatter.h.


The documentation for this class was generated from the following file: