config.hpp 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. /**
  2. * @file
  3. * @author Caleb Fangmeier <caleb@fangmeier.tech>
  4. * @version 0.1
  5. *
  6. * @section LICENSE
  7. *
  8. *
  9. * MIT License
  10. *
  11. * Copyright (c) 2017 Caleb Fangmeier
  12. *
  13. * Permission is hereby granted, free of charge, to any person obtaining a copy
  14. * of this software and associated documentation files (the "Software"), to deal
  15. * in the Software without restriction, including without limitation the rights
  16. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  17. * copies of the Software, and to permit persons to whom the Software is
  18. * furnished to do so, subject to the following conditions:
  19. *
  20. * The above copyright notice and this permission notice shall be included in all
  21. * copies or substantial portions of the Software.
  22. *
  23. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  24. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  25. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  26. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  27. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  28. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  29. * SOFTWARE.
  30. *
  31. */
  32. #ifndef config_hpp
  33. #define config_hpp
  34. #include <cstdlib>
  35. #include <cstdio>
  36. #include <string>
  37. #include <fstream>
  38. #include <sstream>
  39. #include <regex>
  40. #include "yaml-cpp/yaml.h"
  41. namespace fv_util {
  42. struct DataFileDescriptor {
  43. std::string filename;
  44. std::string label;
  45. std::string category;
  46. int file_number;
  47. int n_events;
  48. DataFileDescriptor() : filename(""), label(""), category(""), file_number(-1), n_events(-1) {}
  49. DataFileDescriptor(const std::string filename, const std::string label, const std::string category,
  50. const int n_events)
  51. : filename(filename), label(label), category(category), file_number(-1), n_events(n_events) {}
  52. DataFileDescriptor(const std::string filename, const std::string label, const std::string category,
  53. const int file_number, const int n_events)
  54. : filename(filename), label(label), category(category), file_number(file_number), n_events(n_events) {}
  55. };
  56. }
  57. namespace YAML {
  58. template<>
  59. struct convert<fv_util::DataFileDescriptor> {
  60. static Node encode(const fv_util::DataFileDescriptor &rhs) {
  61. Node node;
  62. node["filename"] = rhs.filename;
  63. node["label"] = rhs.label;
  64. node["category"] = rhs.category;
  65. node["n_events"] = rhs.n_events;
  66. return node;
  67. }
  68. static bool decode(const Node &node, fv_util::DataFileDescriptor &rhs) {
  69. if (!node.IsMap()) {
  70. return false;
  71. }
  72. if (!node["filename"]) return false;
  73. rhs.filename = node["filename"].as<std::string>();
  74. if (node["label"]) rhs.label = node["label"].as<std::string>();
  75. if (node["category"]) rhs.category = node["category"].as<std::string>();
  76. if (node["n_events"]) rhs.n_events = node["n_events"].as<int>();
  77. return true;
  78. }
  79. };
  80. }
  81. namespace fv_util {
  82. std::vector<std::string> glob(const std::string& base) {
  83. std::stringstream ss;
  84. ss << "ls -1 " << base << " > __tmp__";
  85. system(ss.str().c_str());
  86. std::vector<std::string> filenames;
  87. std::ifstream f("__tmp__");
  88. std::string line;
  89. while (std::getline(f, line)) {
  90. filenames.push_back(line);
  91. }
  92. std::remove("__tmp__");
  93. return filenames;
  94. }
  95. class Config {
  96. private:
  97. std::string input_filename;
  98. YAML::Node root;
  99. public:
  100. Config()
  101. : root(), input_filename("") {}
  102. Config(const Config &c)
  103. : root(c.root), input_filename(c.input_filename) {}
  104. Config(const std::string &input_filename)
  105. : root(YAML::LoadFile(input_filename)),
  106. input_filename(input_filename) {}
  107. template<typename KEY_TYPE>
  108. YAML::Node get(const KEY_TYPE &key) {
  109. auto val = root[key];
  110. return val;
  111. }
  112. template<typename T>
  113. void update_key(const std::string &key, const T &val) {
  114. root[key] = val;
  115. }
  116. std::string as_string() {
  117. std::stringstream config;
  118. std::ifstream s(input_filename);
  119. for (std::string line; std::getline(s, line);) {
  120. config << line << std::endl;
  121. }
  122. return config.str();
  123. }
  124. size_t get_max_events() {
  125. YAML::Node max_events = root["max-events"];
  126. if (!max_events) {
  127. return 0;
  128. }
  129. return max_events.as<size_t>();
  130. }
  131. bool get_debug() {
  132. YAML::Node debug_node = root["debug"];
  133. if (!debug_node) {
  134. return false;
  135. }
  136. return debug_node.as<bool>();
  137. }
  138. std::string get_output_filename() {
  139. YAML::Node output_file = root["output-file"];
  140. if (output_file) {
  141. return output_file.as<std::string>();
  142. } else {
  143. std::cout << "Must specify output-file in config file." << std::endl;
  144. return "";
  145. }
  146. }
  147. std::vector<DataFileDescriptor> get_source_files() {
  148. std::vector<DataFileDescriptor> source_files;
  149. std::string source_file_key = "source-files";
  150. if (root["source-file-key"]) {
  151. source_file_key = root["source-file-key"].as<std::string>();
  152. }
  153. YAML::Node source_files_nd = root[source_file_key];
  154. if (source_files_nd.IsMap()) {
  155. auto filenames = glob(source_files_nd["files"].as<std::string>());
  156. DataFileDescriptor dfd;
  157. for (const auto& filename : filenames) {
  158. dfd.filename = filename;
  159. source_files.push_back(dfd);
  160. }
  161. } else if (source_files_nd.IsSequence()) {
  162. for (const auto &f : source_files_nd) {
  163. source_files.push_back(f.as<DataFileDescriptor>());
  164. }
  165. } else {
  166. std::cout << "source-files missing or not a sequence" << std::endl;
  167. }
  168. return source_files;
  169. }
  170. };
  171. Config *the_config = nullptr;
  172. bool debug_on = false;
  173. void init_config(const std::string &input_filename) {
  174. if (the_config) delete the_config;
  175. the_config = new Config(input_filename);
  176. debug_on = the_config->get_debug();
  177. }
  178. }
  179. #endif // config_hpp