Version: SMASH-3.2
rivetoutput.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2021 Christian Holm Christensen
4  * Copyright (c) 2021-2022,2024
5  * SMASH Team
6  *
7  * GNU General Public License (GPLv3 or later)
8  *
9  */
10 
11 #include "smash/rivetoutput.h"
12 
13 #include "Rivet/Rivet.hh"
14 #include "Rivet/Tools/Logging.hh"
15 
16 #include "smash/logging.h"
17 #include "smash/outputparameters.h"
18 
19 namespace smash {
96 RivetOutput::RivetOutput(const std::filesystem::path& path, std::string name,
97  const bool full_event,
98  const RivetOutputParameters& rivet_par)
99  : HepMcInterface(name, full_event),
100  handler_(),
101  filename_(path / (name + ".yoda")),
102  need_init_(true) {
103  handler_ = std::make_shared<Rivet::AnalysisHandler>();
104  setup(rivet_par);
105 }
106 
108  logg[LOutput].debug() << "Writing Rivet results to " << filename_ << "\n";
109  analysis_handler_proxy()->finalize();
110  analysis_handler_proxy()->writeData(filename_.string());
111 }
112 
113 void RivetOutput::at_eventend(const Particles& particles,
114  const EventLabel& event_label,
115  const EventInfo& event) {
116  HepMcInterface::at_eventend(particles, event_label, event);
117 
118  // Initialize Rivet on first event
119  if (need_init_) {
120  logg[LOutput].debug() << "Initialising Rivet\n";
121  need_init_ = false;
123  }
124 
125  logg[LOutput].debug() << "Analyzing event " << event_label.event_number
126  << "\n";
127  // Let Rivet analyse the event
128  analysis_handler_proxy()->analyze(event_);
129 }
130 
131 void RivetOutput::add_analysis(const std::string& name) {
132  analysis_handler_proxy()->addAnalysis(name);
133 }
134 
135 void RivetOutput::add_path(const std::string& path) {
136  Rivet::addAnalysisLibPath(path);
137  Rivet::addAnalysisDataPath(path);
138 }
139 
140 void RivetOutput::add_preload(const std::string& file) {
141  analysis_handler_proxy()->readData(file);
142 }
143 
144 void RivetOutput::set_ignore_beams(bool ignore) {
145  logg[LOutput].info() << "Ignore beams? " << (ignore ? "yes" : "no") << "\n";
146  analysis_handler_proxy()->setIgnoreBeams(ignore);
147 }
148 
149 void RivetOutput::set_log_level(const std::string& name,
150  const std::string& level) {
151  std::string fname(name);
152  if (fname.rfind("Rivet", 0) != 0) {
153  fname = "Rivet." + fname;
154  }
155 
156  auto upcase = [](const std::string& s) {
157  std::string out(s);
158  std::transform(out.begin(), out.end(), out.begin(),
159  [](char c) { return std::toupper(c); });
160  return out;
161  };
162 
163  try {
164  Rivet::Log::setLevel(fname, Rivet::Log::getLevelFromName(upcase(level)));
165  } catch (...) {
166  }
167 }
168 
169 void RivetOutput::set_cross_section(double xs, double xserr) {
170  analysis_handler_proxy()->setCrossSection(xs, xserr, true);
171 }
172 
174  logg[LOutput].debug() << "Setting up Rivet output:\n";
175 
176  // Paths to analyses libraries and data
177  if (params.paths) {
178  logg[LOutput].info() << "Processing paths\n";
179  for (auto p : params.paths.value())
180  add_path(p);
181  }
182 
183  // Data files to pre-load e.g., for centrality configurations
184  if (params.preloads) {
185  logg[LOutput].info() << "Processing preloads\n";
186  for (auto p : params.preloads.value())
187  add_preload(p);
188  }
189 
190  // Analyses (including options) to add to run
191  if (params.analyses) {
192  logg[LOutput].info() << "Processing analyses\n";
193  for (auto p : params.analyses.value())
194  add_analysis(p);
195  }
196 
197  // Whether Rivet should ignore beams
199 
200  // Cross sections
201  if (params.cross_sections) {
202  set_cross_section(params.cross_sections.value()[0],
203  params.cross_sections.value()[1]);
204  }
205 
206  // Logging in Rivet
207  if (params.logs) {
208  for (auto nl : params.logs.value())
209  set_log_level(nl.first, nl.second);
210  }
211 
212  // Treatment of event weights in Rivet
213  if (params.any_weight_parameter_was_given) {
214  // Do not care about multi weights
215  if (params.no_multi_weight) {
216  analysis_handler_proxy()->skipMultiWeights(
217  params.no_multi_weight.value());
218  }
219 
220  // Set nominal weight name
221  if (params.nominal_weight_name) {
222  analysis_handler_proxy()->setNominalWeightName(
223  params.nominal_weight_name.value());
224  }
225 
226  // Set cap (maximum) on weights
227  if (params.cap_on_weights) {
228  analysis_handler_proxy()->setWeightCap(params.cap_on_weights.value());
229  }
230 
231  // Whether to smear for NLO calculations
232  if (params.nlo_smearing) {
233  analysis_handler_proxy()->setNLOSmearing(params.nlo_smearing.value());
234  }
235 
236  // Select which weights to enable
237  if (params.to_be_enabled_weights) {
238  std::stringstream s;
239  int comma = 0;
240  for (auto w : params.to_be_enabled_weights.value())
241  s << (comma++ ? "," : "") << w;
242  analysis_handler_proxy()->selectMultiWeights(s.str());
243  }
244 
245  // Select weights to disable
246  if (params.to_be_disabled_weights) {
247  std::stringstream s;
248  int comma = 0;
249  for (auto w : params.to_be_disabled_weights.value())
250  s << (comma++ ? "," : "") << w;
251  analysis_handler_proxy()->deselectMultiWeights(s.str());
252  }
253  }
254  logg[LOutput].debug() << "Setup of Rivet output done.\n";
255 }
256 } // namespace smash
Base class for output handlers that need the HepMC3 structure.
void at_eventend(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Add the final particles information of an event to the central vertex.
HepMC3::GenEvent event_
The event.
The Particles class abstracts the storage and manipulation of particles.
Definition: particles.h:33
Proxy analysis_handler_proxy()
Return a proxy that temporarily disables FP exceptions.
Definition: rivetoutput.h:122
void add_path(const std::string &path)
Add a load path to the Rivet handler.
Definition: rivetoutput.cc:135
void add_analysis(const std::string &name)
Add an analysis or analyses to Rivet.
Definition: rivetoutput.cc:131
std::filesystem::path filename_
Output file.
Definition: rivetoutput.h:127
std::shared_ptr< Rivet::AnalysisHandler > handler_
Rivet analysis handler.
Definition: rivetoutput.h:125
bool need_init_
Whether we need initialisation.
Definition: rivetoutput.h:129
~RivetOutput()
Destructor.
Definition: rivetoutput.cc:107
void set_cross_section(double xs, double xserr)
Set X-section.
Definition: rivetoutput.cc:169
void add_preload(const std::string &file)
Add preload to Rivet handler.
Definition: rivetoutput.cc:140
void setup(const RivetOutputParameters &params)
Setup Rivet using SMASH configuration parameters.
Definition: rivetoutput.cc:173
void set_log_level(const std::string &name, const std::string &level)
Set log level in Rivet.
Definition: rivetoutput.cc:149
void at_eventend(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Add the final particles information of an event to the central vertex.
Definition: rivetoutput.cc:113
RivetOutput(const std::filesystem::path &path, std::string name, const bool full_event, const RivetOutputParameters &rivet_par)
Create Rivet output.
Definition: rivetoutput.cc:96
void set_ignore_beams(bool ignore=true)
Do not insist on appropriate beams for analyses.
Definition: rivetoutput.cc:144
std::array< einhard::Logger<>, std::tuple_size< LogArea::AreaTuple >::value > logg
An array that stores all pre-configured Logger objects.
Definition: logging.cc:40
constexpr int p
Proton.
Definition: action.h:24
static constexpr int LOutput
Structure to contain custom data for output.
Structure to contain information about the event and ensemble numbers.
int32_t event_number
The number of the event.
Helper structure for OutputParameters in order to store and hand over Rivet parameters.
std::optional< std::vector< std::string > > to_be_enabled_weights
Weights to be enabled for processing.
std::optional< std::string > nominal_weight_name
Nominal weight name.
std::optional< std::map< std::string, std::string > > logs
Logging in Rivet.
std::optional< std::vector< std::string > > preloads
Data files to pre-load e.g., for centrality configurations.
std::optional< std::vector< std::string > > paths
Paths to analyses libraries and data.
std::optional< double > nlo_smearing
How to smear for NLO calculations.
std::optional< std::vector< std::string > > analyses
Analyses (including options) to add to run.
bool ignore_beams
Whether Rivet should ignore beams.
std::optional< double > cap_on_weights
Cap (maximum) on weights.
std::optional< bool > no_multi_weight
Whether Rivet should not care about multi weights.
std::optional< std::vector< std::string > > to_be_disabled_weights
Weights to be disabled for processing.
bool any_weight_parameter_was_given
Whether any weight parameter was specified.
std::optional< std::array< double, 2 > > cross_sections
Cross sections.