goptions
goptions.cc
Go to the documentation of this file.
1 // goptions
2 #include "goptions.h"
3 #include "goptionsConventions.h"
4 #include "gversion.h"
5 
6 // gemc
7 #include "gutilities.h"
8 
9 // c++
10 #include <iostream>
11 #include <cstring>
12 
13 using namespace std;
14 
29 GOptions::GOptions(int argc, char *argv[], const GOptions &user_defined_options) {
30 
31  executableName = gutilities::getFileFromPath(argv[0]);
32  cout << endl;
33 
34  // Add user-defined options.
35  addGOptions(user_defined_options);
36 
37  // switches for all everyone
38  defineSwitch("gui", "use Graphical User Interface");
39  defineSwitch("i", "use interactive batch mode");
40  defineOption(
41  GVariable("conf_yaml", "saved_configuration", "the yaml filename prefix where all used options are saved"),
42  "The default value appends \"_saved_configuration\" to the executable name.");
43 
44  // version is a special option, not settable by the user
45  // it is set by the gversion.h file
46  // we add it here so it can be saved to the yaml file
47  vector <GVariable> version = {
48  {"release", gversion, "release version number"},
49  {"release_date", grelease_date, "release date"},
50  {"Reference", greference, "article reference"},
51  {"Homepage", gweb, "homepage"},
52  {"Author", gauthor, "author"}
53  };
54  defineOption(GVERSION_STRING, "version information", version, "Version information. Not settable by user.");
55 
56  string help = "Levels: \n \n";
57  help += " - 0: (default) shush\n";
58  help += " - 1: normal information\n";
59  help += " - 2: detailed information\n \n";
60  help += "Example: -verbosity.general=1 \n \n";
61  help += "This option can be repeated.\n \n";
62  defineOption("verbosity", "Sets the log verbosity for various classes", option_verbosity_names, help);
63 
64 
65  help = "Debug information Types: \n \n";
66  help += " - false: (default): do not print debug information\n";
67  help += " - true: print debug information\n\n";
68  help += "Example: -debug.general=true \n \n";
69  help += "This option can be repeated.\n \n";
70  defineOption("debug", "Sets the debug level for various classes", option_verbosity_names, help);
71 
72  // Process help/version command-line arguments.
73  for (int i = 1; i < argc; i++) {
74  if (strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--h") == 0 ||
75  strcmp(argv[i], "-help") == 0 || strcmp(argv[i], "--help") == 0) {
76  printHelp();
77  } else if (strcmp(argv[i], "-hweb") == 0) {
78  printWebHelp();
79  } else if (strcmp(argv[i], "-v") == 0 || strcmp(argv[i], "--v") == 0 ||
80  strcmp(argv[i], "-version") == 0 || strcmp(argv[i], "--version") == 0) {
81  print_version();
82  exit(EXIT_SUCCESS);
83  } else if (strcmp(argv[i], "help") == 0) {
84  printOptionOrSwitchHelp(argv[i + 1]);
85  exit(EXIT_SUCCESS);
86  }
87  }
88 
89  // finds and parse the yaml files
90  yaml_files = findYamls(argc, argv);
91  for (auto &yaml_file: yaml_files) {
92  cout << " Parsing " << yaml_file << endl;
93  setOptionsValuesFromYamlFile(yaml_file);
94  }
95 
96  // Parse command-line arguments (supports both standard YAML–style and dot–notation).
97  for (int i = 1; i < argc; i++) {
98  string candidate = argv[i];
99  if (candidate == "") continue;
100  if (find(yaml_files.begin(), yaml_files.end(), candidate) != yaml_files.end()) continue;
101  if (candidate[0] == '-') {
102  string argStr = candidate.substr(1);
103  size_t eqPos = argStr.find('=');
104  if (eqPos != string::npos) {
105  string keyPart = argStr.substr(0, eqPos);
106  string valuePart = argStr.substr(eqPos + 1);
107  size_t dotPos = keyPart.find('.');
108  if (dotPos != string::npos) {
109  // Dot–notation detected (e.g. "debug.general=true")
110  string mainOption = keyPart.substr(0, dotPos);
111  string subOption = keyPart.substr(dotPos + 1);
112  if (doesOptionExist(mainOption)) {
113  auto it = getOptionIterator(mainOption);
114  it->set_sub_option_value(subOption, valuePart);
115  } else {
116  cerr << "The option " << mainOption << " is not known to this system." << endl;
117  exit(EC__NOOPTIONFOUND);
118  }
119  } else {
120  // Standard option syntax.
121  if (doesOptionExist(keyPart)) {
122  setOptionValuesFromCommandLineArgument(keyPart, valuePart);
123  } else {
124  cerr << "The option " << keyPart << " is not known to this system." << endl;
125  exit(EC__NOOPTIONFOUND);
126  }
127  }
128  } else {
129  // Treat as a switch.
130  string possibleSwitch = argStr;
131  if (switches.find(possibleSwitch) != switches.end()) {
132  switches[possibleSwitch].turnOn();
133  } else {
134  cerr << "The switch " << possibleSwitch << " is not known to this system." << endl;
135  exit(EC__NOOPTIONFOUND);
136  }
137  }
138  } else {
139  cerr << "The command-line argument \"" << candidate << "\" is not valid." << endl;
140  exit(EC__NOOPTIONFOUND);
141  }
142  }
143 
144  // Always print version information.
145  print_version();
146 
147  // Save the final configuration to a YAML file.
148  string yamlConf_filename = executableName + "." + getScalarString("conf_yaml") + ".yaml";
149  cout << " Saving options to " << yamlConf_filename << endl << endl;
150  yamlConf = new std::ofstream(yamlConf_filename);
151  saveOptions();
152 }
153 
160 void GOptions::defineSwitch(const std::string &name, const std::string &description) {
161  if (switches.find(name) == switches.end()) {
162  switches[name] = GSwitch(description);
163  } else {
164  std::cerr << FATALERRORL << "The " << YELLOWHHL << name << RSTHHR
165  << " switch is already present." << std::endl;
167  }
168 }
169 
176 void GOptions::defineOption(const GVariable &gvar, const std::string &help) {
177  if (doesOptionExist(gvar.name)) {
178  std::cerr << FATALERRORL << "The " << YELLOWHHL << gvar.name << RSTHHR
179  << " option is already present." << std::endl;
181  } else {
182  goptions.push_back(GOption(gvar, help));
183  }
184 }
185 
194 void GOptions::defineOption(const std::string &name, const std::string &description, const std::vector <GVariable> &g_vars,
195  const std::string &help) {
196  if (doesOptionExist(name)) {
197  std::cerr << FATALERRORL << "The " << YELLOWHHL << name << RSTHHR
198  << " option is already present." << std::endl;
200  } else {
201  goptions.push_back(GOption(name, description, g_vars, help));
202  }
203 }
204 
211 int GOptions::getScalarInt(const std::string &tag) const {
212  auto it = getOptionIterator(tag);
213  if (it == goptions.end()) {
214  cerr << FATALERRORL << "The option " << YELLOWHHL << tag << RSTHHR
215  << " was not found." << endl;
216  exit(EC__NOOPTIONFOUND);
217  }
218  return it->value.begin()->second.as<int>();
219 }
220 
227 float GOptions::getScalarFloat(const std::string &tag) const {
228  auto it = getOptionIterator(tag);
229  if (it == goptions.end()) {
230  cerr << FATALERRORL << "The option " << YELLOWHHL << tag << RSTHHR
231  << " was not found." << endl;
232  exit(EC__NOOPTIONFOUND);
233  }
234  return it->value.begin()->second.as<float>();
235 }
236 
243 double GOptions::getScalarDouble(const std::string &tag) const {
244  auto it = getOptionIterator(tag);
245  if (it == goptions.end()) {
246  cerr << FATALERRORL << "The option " << YELLOWHHL << tag << RSTHHR
247  << " was not found." << endl;
248  exit(EC__NOOPTIONFOUND);
249  }
250  return it->value.begin()->second.as<double>();
251 }
252 
259 string GOptions::getScalarString(const std::string &tag) const {
260  auto it = getOptionIterator(tag);
261  if (it == goptions.end()) {
262  cerr << FATALERRORL << "The option " << YELLOWHHL << tag << RSTHHR
263  << " was not found." << endl;
264  exit(EC__NOOPTIONFOUND);
265  }
266  return it->value.begin()->second.as<string>();
267 }
268 
274 void GOptions::printOptionOrSwitchHelp(const std::string &tag) const {
275  auto switchIt = switches.find(tag);
276  if (switchIt != switches.end()) {
277  cout << KGRN << "-" << tag << RST << ": " << switchIt->second.getDescription() << endl << endl;
278  cout << TPOINTITEM << "Default value is " << (switchIt->second.getStatus() ? "on" : "off") << endl << endl;
279  exit(EXIT_SUCCESS);
280  }
281  for (const auto &goption: goptions) {
282  if (goption.name == tag) {
283  goption.printHelp(true);
284  exit(EXIT_SUCCESS);
285  }
286  }
287  cerr << FATALERRORL << "The " << YELLOWHHL << tag << RSTHHR
288  << " option is not known to this system." << endl;
289  exit(EC__NOOPTIONFOUND);
290 }
291 
299 vector <string> GOptions::findYamls(int argc, char *argv[]) {
300  vector <string> yaml_files;
301  for (int i = 1; i < argc; i++) {
302  string arg = argv[i];
303  size_t pos = arg.find(".yaml");
304  if (pos != string::npos) yaml_files.push_back(arg);
305  pos = arg.find(".yml");
306  if (pos != string::npos) yaml_files.push_back(arg);
307  }
308  return yaml_files;
309 }
310 
311 // checks if the option exists
312 bool GOptions::doesOptionExist(const std::string &tag) const {
313  for (const auto &goption: goptions) {
314  // Use const auto& to avoid copying and to ensure const-correctness
315  if (goption.name == tag) {
316  return true;
317  }
318  }
319  return false;
320 }
321 
327 void GOptions::setOptionsValuesFromYamlFile(const std::string &yaml) {
328  YAML::Node config;
329  try {
330  config = YAML::LoadFile(yaml);
331  } catch (YAML::ParserException &e) {
332  cerr << FATALERRORL << "Error parsing " << YELLOWHHL << yaml << RSTHHR
333  << " yaml file." << endl;
334  cerr << e.what() << endl;
335  cerr << "Try validating the yaml file with an online yaml validator, e.g., https://www.yamllint.com" << endl;
337  }
338 
339  for (auto it = config.begin(); it != config.end(); ++it) {
340  string option_name = it->first.as<std::string>();
341  auto option_it = getOptionIterator(option_name);
342  if (option_it == goptions.end()) {
343  if (switches.find(option_name) == switches.end()) {
344  cerr << FATALERRORL << "The option or switch " << YELLOWHHL << option_name << RSTHHR
345  << " is not known to this system." << endl;
346  exit(EC__NOOPTIONFOUND);
347  } else {
348  switches[option_name].turnOn();
349  }
350  } else {
351  YAML::NodeType::value type = it->second.Type();
352  switch (type) {
353  case YAML::NodeType::Scalar:
354  option_it->set_scalar_value(it->second.as<std::string>());
355  break;
356  case YAML::NodeType::Sequence:
357  option_it->set_value(it->second);
358  break;
359  case YAML::NodeType::Map:
360  option_it->set_value(it->second);
361  break;
362  default:
363  break;
364  }
365  }
366  }
367 }
368 
375 void GOptions::setOptionValuesFromCommandLineArgument(const std::string &optionName, const std::string &possibleYamlNode) {
376  YAML::Node node = YAML::Load(possibleYamlNode);
377  auto option_it = getOptionIterator(optionName);
378  if (node.Type() == YAML::NodeType::Scalar) {
379  option_it->set_scalar_value(possibleYamlNode);
380  } else {
381  option_it->set_value(node);
382  }
383 }
384 
391 std::vector<GOption>::iterator GOptions::getOptionIterator(const std::string &name) {
392  return std::find_if(goptions.begin(), goptions.end(),
393  [&name](GOption &option) { return option.name == name; });
394 }
395 
396 
403 std::vector<GOption>::const_iterator GOptions::getOptionIterator(const std::string &name) const {
404  return std::find_if(goptions.begin(), goptions.end(),
405  [&name](const GOption &option) { return option.name == name; });
406 }
407 
414 bool GOptions::getSwitch(const std::string &tag) const {
415  auto it = switches.find(tag);
416  if (it != switches.end()) {
417  return it->second.getStatus();
418  } else {
419  std::cerr << FATALERRORL << "The switch " << YELLOWHHL << tag << RSTHHR
420  << " was not found." << std::endl;
421  exit(EC__NOOPTIONFOUND);
422  }
423  return false;
424 }
425 
433 YAML::Node GOptions::getOptionMapInNode(string option_name, string map_key) {
434  auto sequence_node = getOptionNode(option_name);
435  for (auto seq_item : sequence_node) {
436  for (auto map_item = seq_item.begin(); map_item != seq_item.end(); ++map_item) {
437  if (map_item->first.as<string>() == map_key) {
438  return map_item->second;
439  }
440  }
441  }
442  cerr << FATALERRORL << "The key " << YELLOWHHL << map_key << RSTHHR
443  << " was not found in " << YELLOWHHL << option_name << RSTHHR << endl;
444  exit(EC__NOOPTIONFOUND);
445  return sequence_node;
446 }
447 
457 template<typename T>
458 T GOptions::get_variable_in_option(const YAML::Node &node, const std::string &variable_name, const T &default_value) {
459  if (node[variable_name]) {
460  return node[variable_name].as<T>();
461  }
462  return default_value;
463 }
464 
465 // Explicit template instantiations.
466 template int GOptions::get_variable_in_option<int>(const YAML::Node &node, const std::string &variable_name, const int &default_value);
467 template float GOptions::get_variable_in_option<float>(const YAML::Node &node, const std::string &variable_name, const float &default_value);
468 template double GOptions::get_variable_in_option<double>(const YAML::Node &node, const std::string &variable_name, const double &default_value);
469 template string GOptions::get_variable_in_option<string>(const YAML::Node &node, const std::string &variable_name, const string &default_value);
470 template bool GOptions::get_variable_in_option<bool>(const YAML::Node &node, const std::string &variable_name, const bool &default_value);
471 
478 int GOptions::getVerbosityFor(const std::string &tag) const {
479  YAML::Node verbosity_node = getOptionNode("verbosity");
480  for (auto v : verbosity_node) {
481  if (v.begin()->first.as<string>() == tag) {
482  return v.begin()->second.as<int>();
483  }
484  }
485 
486  // not found. error
487  std::cerr << KRED << " Invalid verbosity or debug requested: " << tag << RST << std::endl;
488  exit(EC__NOOPTIONFOUND);
489 }
490 
499 int GOptions::getDebugFor(const std::string &tag) const {
500  YAML::Node debug_node = getOptionNode("debug");
501  for (auto d : debug_node) {
502  if (d.begin()->first.as<string>() == tag) {
503  YAML::Node valNode = d.begin()->second;
504  if (valNode.IsScalar()) {
505  string s = valNode.as<string>();
506  if (s == "true") return 1;
507  if (s == "false") return 0;
508  }
509  try {
510  return valNode.as<int>();
511  } catch (const YAML::BadConversion &) {
512  std::cerr << "Invalid debug value for " << tag << std::endl;
513  exit(EC__BAD_CONVERSION);
514  }
515  }
516  }
517  // not found. error
518  std::cerr << KRED << " Invalid verbosity or debug requested: " << tag << RST << std::endl;
519  exit(EC__NOOPTIONFOUND);}
520 
524 void GOptions::printHelp() const {
525  long int fill_width = string(HELPFILLSPACE).size() + 1;
526  cout.fill('.');
527  cout << KGRN << KBOLD << " " << executableName << RST << " [options] [yaml files]" << endl << endl;
528  cout << " Switches: " << endl << endl;
529  for (auto &s : switches) {
530  string help = "-" + s.first + RST + " ";
531  cout << KGRN << " " << left;
532  cout.width(fill_width);
533  cout << help;
534  cout << ": " << s.second.getDescription() << endl;
535  }
536  cout << endl;
537  cout << " Options: " << endl << endl;
538  for (auto &option : goptions) {
539  option.printHelp(false);
540  }
541  cout << endl;
542  cout << endl << " Help / Search / Introspection: " << endl << endl;
543  vector<string> helps = {
544  string("-h, --h, -help, --help") + RST,
545  string("print this help and exit"),
546  string("-hweb") + RST,
547  string("print this help in web format and exit"),
548  string("-v, --v, -version, --version") + RST,
549  string("print the version and exit\n"),
550  string("help <value>") + RST,
551  string("print detailed help for option <value> and exit"),
552  string("search <value>") + RST,
553  string("list all options containing <value> in the description and exit\n")
554  };
555  unsigned half_help = helps.size() / 2;
556  for (unsigned i = 0; i < half_help; i++) {
557  cout << KGRN << " " << left;
558  cout.width(fill_width);
559  cout << helps[i * 2] << ": " << helps[i * 2 + 1] << endl;
560  }
561  cout << endl;
562  cout << " Note: command line options overwrite YAML file(s)." << endl << endl;
563  exit(EXIT_SUCCESS);
564 }
565 
566 
567 
571 void GOptions::printWebHelp() const {
572  exit(EXIT_SUCCESS);
573 }
574 
578 void GOptions::saveOptions() const {
579  for (auto &s : switches) {
580  string status = s.second.getStatus() ? "true" : "false";
581  *yamlConf << s.first + ": " + status << "," << endl;
582  }
583  for (const auto &option : goptions) {
584  option.saveOption(yamlConf);
585  }
586  yamlConf->close();
587 }
588 
592 void GOptions::print_version() {
593  string asterisks = "**************************************************************";
594  cout << endl << asterisks << endl;
595  cout << " " << KGRN << KBOLD << executableName << RST << " version: " << KGRN << gversion << RST << endl;
596  cout << " Released on: " << KGRN << grelease_date << RST << endl;
597  cout << " Reference: " << KGRN << greference << RST << endl;
598  cout << " Homepage: " << KGRN << gweb << RST << endl;
599  cout << " Author: " << KGRN << gauthor << RST << endl << endl;
600  cout << asterisks << endl << endl;
601 }
602 
610 GOptions &operator+=(GOptions &gopts, const GOptions &goptions_to_add) {
611  gopts.addGOptions(goptions_to_add);
612  return gopts;
613 }
Represents a configurable option with a name, value(s), description, and help text.
Definition: goption.h:89
The GOptions class manages command-line options and switches.
Definition: goptions.h:20
void defineOption(const GVariable &gvar, const string &help)
Defines and adds a scalar option.
Definition: goptions.cc:176
bool getSwitch(const string &tag) const
Retrieves the status of a switch.
Definition: goptions.cc:414
int getDebugFor(const string &tag) const
Retrieves the debug level for the specified tag.
Definition: goptions.cc:499
T get_variable_in_option(const YAML::Node &node, const string &variable_name, const T &default_value)
Retrieves a variable from a YAML node within an option.
Definition: goptions.cc:458
int getVerbosityFor(const string &tag) const
Retrieves the verbosity level for the specified tag.
Definition: goptions.cc:478
GOptions()
Default constructor.
Definition: goptions.h:27
int getScalarInt(const string &tag) const
Retrieves the value of a scalar integer option.
Definition: goptions.cc:211
void addGOptions(const GOptions &goptions_to_add)
Adds options from another GOptions object.
Definition: goptions.h:162
bool doesOptionExist(const string &tag) const
Checks if the specified option exists.
Definition: goptions.cc:312
float getScalarFloat(const string &tag) const
Retrieves the value of a scalar float option.
Definition: goptions.cc:227
void defineSwitch(const string &name, const string &description)
Defines and adds a command–line switch.
Definition: goptions.cc:160
string getScalarString(const string &tag) const
Retrieves the value of a scalar string option.
Definition: goptions.cc:259
YAML::Node getOptionMapInNode(string option_name, string map_key)
Retrieves a map option’s value from within a YAML node.
Definition: goptions.cc:433
double getScalarDouble(const string &tag) const
Retrieves the value of a scalar double option.
Definition: goptions.cc:243
Represents a switch with a description and a status.
Definition: gswitch.h:14
#define EC__BAD_CONVERSION
#define EC__NOOPTIONFOUND
#define EC__YAML_PARSING_ERROR
#define HELPFILLSPACE
#define EC__DEFINED_SWITCHALREADYPRESENT
#define EC__DEFINED_OPTION_ALREADY_PRESENT
#define GVERSION_STRING
GOptions & operator+=(GOptions &gopts, const GOptions &goptions_to_add)
Overloaded operator to add options and switches from one GOptions object to another.
Definition: goptions.cc:610
Encapsulates a variable with a name, value, and description.
Definition: goption.h:30
string name
The name of the variable.
Definition: goption.h:31