autotier
Automatic Tiering Fuse Filesystem
ConfigParser.hpp
1 // -*- C++ -*-
2 /*
3  * Copyright (C) 2021 Joshua Boudreau <jboudreau@45drives.com>
4  *
5  * This file is part of lib45d.
6  *
7  * lib45d is free software: you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation, either version 3 of the License, or
10  * (at your option) any later version.
11  *
12  * lib45d is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with lib45d. If not, see <https://www.gnu.org/licenses/>.
19  */
20 
21 #pragma once
22 
23 #include <45d/config/ConfigNode.hpp>
24 #include <45d/Exceptions.hpp>
25 #include <45d/Quota.hpp>
26 #include <string>
27 #include <unordered_map>
28 #include <vector>
29 #include <sstream>
30 
37 namespace ffd_internal {
48  template<class T>
49  T get(const std::string &key, const std::unordered_map<std::string, ffd::ConfigNode> *config_map) {
50  ffd::ConfigNode node = config_map->at(key);
51  std::stringstream ss(node.value_);
52  ss.exceptions(std::ios::failbit | std::ios::badbit);
53  T result;
54  ss >> result;
55  return result;
56  }
57 }
58 
59 namespace ffd {
67  class ConfigParser {
72  friend class ConfigSubsectionGuard;
73  public:
81  ConfigParser(std::string path);
87  std::string dump_str(void) const;
102  template<class T>
103  T get(const std::string &key) const {
104  return ffd_internal::get<T>(key, config_map_ptr_);
105  }
117  template<class T>
118  T get(const std::string &key, const T &fallback) const noexcept {
119  try {
120  return get<T>(key);
121  } catch (const std::out_of_range &) {
122  // silently return fallback
123  } catch (const std::ios_base::failure &) {
124  report_error("Invalid configuration entry format: " + key + " = " + config_map_.at(key).value_);
125  } catch (const ConfigException &e) {
126  report_error(e.what());
127  } catch (const std::exception &e) {
128  report_error("Unexpected std::exception while getting " + key + ": " + e.what());
129  } catch ( ... ) {
130  report_error("Unexplained exception caught while getting " + key);
131  }
132  return fallback;
133  }
147  template<class T>
148  T get(const std::string &key, bool *fail_flag) const noexcept {
149  try {
150  return get<T>(key);
151  } catch (const std::ios_base::failure &) {
152  report_error("Invalid configuration entry format: " + key + " = " + config_map_.at(key).value_);
153  } catch (const std::out_of_range &) {
154  report_error("Option not in config: " + key);
155  } catch (const ConfigException &e) {
156  report_error(e.what());
157  } catch (const std::exception &e) {
158  report_error("Unexpected std::exception while getting " + key + ": " + e.what());
159  } catch ( ... ) {
160  report_error("Unexplained exception caught while getting " + key);
161  }
162  *fail_flag = true;
163  if (std::is_fundamental<T>::value)
164  return 0;
165  else
166  return T();
167  }
176  template<class T>
177  T get_from(const std::string &section, const std::string &key) {
178  if (guarded_)
179  throw(ffd::ConfigGuardException("Cannot call get_from while ConfigSubsectionGuard is in scope"));
180  set_subsection(section);
181  T result = get<T>(key);
183  return result;
184  }
197  template<class T>
198  T get_from(const std::string &section, const std::string &key, const T &fallback) noexcept {
199  if (guarded_) {
200  std::cerr << "Cannot call get_from while ConfigSubsectionGuard is in scope" << std::endl;
201  return fallback;
202  }
203  try {
204  set_subsection(section);
205  } catch (const std::out_of_range &) {
207  return fallback;
208  }
209  T result = get<T>(key, fallback);
211  return result;
212  }
222  template<class T>
223  T get_from(const std::string &section, const std::string &key, bool *fail_flag) noexcept {
224  if (guarded_) {
225  std::cerr << "Cannot call get_from while ConfigSubsectionGuard is in scope" << std::endl;
226  *fail_flag = true;
227  if (std::is_fundamental<T>::value)
228  return 0;
229  else
230  return T();
231  }
232  try {
233  set_subsection(section);
234  } catch (const std::out_of_range &) {
235  std::cerr << "Section not in config: " << section << std::endl;
236  *fail_flag = true;
238  if (std::is_fundamental<T>::value)
239  return 0;
240  else
241  return T();
242  }
243  T result = get<T>(key, fail_flag);
245  return result;
246  }
262  Quota get_quota(const std::string &key, const Bytes &max) const {
263  return Quota(max, get<std::string>(key));
264  }
273  Quota get_quota(const std::string &key, const Bytes &max, const Quota &fallback) const noexcept {
274  try {
275  return get_quota(key, max);
276  } catch (const std::out_of_range &) {
277  // silently return fallback
278  } catch (const std::ios_base::failure &) {
279  report_error("Invalid configuration entry format: " + key + " = " + config_map_.at(key).value_);
280  } catch (const ConfigException &e) {
281  report_error(e.what());
282  } catch (const std::exception &e) {
283  report_error("Unexpected std::exception while getting " + key + ": " + e.what());
284  } catch ( ... ) {
285  report_error("Unexplained exception caught while getting " + key);
286  }
287  return fallback;
288  }
299  Quota get_quota(const std::string &key, const Bytes &max, bool *fail_flag) const noexcept {
300  try {
301  return get_quota(key, max);
302  } catch (const std::ios_base::failure &) {
303  report_error("Invalid configuration entry format: " + key + " = " + config_map_.at(key).value_);
304  } catch (const std::out_of_range &) {
305  report_error("Option not in config: " + key);
306  } catch (const ConfigException &e) {
307  report_error(e.what());
308  } catch (const std::exception &e) {
309  report_error("Unexpected std::exception while getting " + key + ": " + e.what());
310  } catch ( ... ) {
311  report_error("Unexplained exception caught while getting " + key);
312  }
313  *fail_flag = true;
314  return Quota();
315  }
324  Quota get_quota_from(const std::string &section, const std::string &key, const Bytes &max) {
325  if (guarded_)
326  throw(ffd::ConfigGuardException("Cannot call get_from while ConfigSubsectionGuard is in scope"));
327  set_subsection(section);
328  Quota result = get_quota(key, max);
330  return result;
331  }
344  Quota get_quota_from(const std::string &section, const std::string &key, const Bytes &max, const Quota &fallback) noexcept {
345  if (guarded_) {
346  std::cerr << "Cannot call get_from while ConfigSubsectionGuard is in scope" << std::endl;
347  return fallback;
348  }
349  try {
350  set_subsection(section);
351  } catch (const std::out_of_range &) {
353  return fallback;
354  }
355  Quota result = get_quota(key, max, fallback);
357  return result;
358  }
368  Quota get_quota_from(const std::string &section, const std::string &key, const Bytes &max, bool *fail_flag) noexcept {
369  if (guarded_) {
370  std::cerr << "Cannot call get_from while ConfigSubsectionGuard is in scope" << std::endl;
371  *fail_flag = true;
372  return Quota();
373  }
374  try {
375  set_subsection(section);
376  } catch (const std::out_of_range &) {
377  std::cerr << "Section not in config." << std::endl;
378  *fail_flag = true;
380  return Quota();
381  }
382  Quota result = get_quota(key, max, fail_flag);
384  return result;
385  }
386  protected:
387  std::vector<ConfigNode *> sub_confs_;
388  std::string current_section_;
389  private:
395  bool guarded_;
396  std::unordered_map<std::string, ConfigNode> *config_map_ptr_;
397  std::string path_;
398  std::unordered_map<std::string, ConfigNode> config_map_;
399 
411  void parse(std::ifstream &file);
417  void parse_entry(const std::string &line);
428  void parse_heading(const std::string &line);
434  void set_subsection(const std::string &section) {
435  config_map_ptr_ = config_map_.at(section).sub_map_;
436  if(config_map_ptr_ == nullptr)
437  throw std::out_of_range("ConfigNode has no sub_map_");
438  current_section_ = section;
439  }
444  void reset_subsection(void) noexcept {
446  current_section_ = "";
447  }
454  void report_error(const std::string &message) const noexcept {
456  std::cerr << "[" << current_section_ << "]: ";
457  std::cerr << message << std::endl;
458  }
459  };
460 }
ffd::ConfigParser::ConfigParser
ConfigParser(std::string path)
Construct a new Config Parser object.
ffd::Bytes
Use this class for byte-formatted values. e.g.: "123 KiB".
Definition: Bytes.hpp:32
ffd::ConfigParser::get_from
T get_from(const std::string &section, const std::string &key)
Adapter for ffd::get(). Sets config_map_ptr_ to address of sub config with name section....
Definition: ConfigParser.hpp:177
ffd::ConfigParser::get_quota
Quota get_quota(const std::string &key, const Bytes &max, const Quota &fallback) const noexcept
Try to get Quota from config, default to fallback if fails. Guaranteed no-throw.
Definition: ConfigParser.hpp:273
ffd::Quota
This class extends ffd::Bytes to specify percents of an amount of bytes.
Definition: Quota.hpp:30
ffd::ConfigParser::get
T get(const std::string &key, bool *fail_flag) const noexcept
Try to get value from config. If ffd::get fails, return T() or 0 and set fail_flag....
Definition: ConfigParser.hpp:148
ffd::ConfigParser::dump_str
std::string dump_str(void) const
Dump config to stdout as a string.
ffd::ConfigSubsectionGuard
Use this to switch to a certain config subsection to get a group of values.
Definition: ConfigSubsectionGuard.hpp:49
ffd_internal::get
T get(const std::string &key, const std::unordered_map< std::string, ffd::ConfigNode > *config_map)
Get config entry as type T from configuration map.
Definition: ConfigParser.hpp:49
ffd::ConfigParser::parse_entry
void parse_entry(const std::string &line)
Extract value from config line and insert ConfigNode into config_map_.
ffd::ConfigParser::get_from
T get_from(const std::string &section, const std::string &key, bool *fail_flag) noexcept
Get value from config subsection using ConfigParser::get(const std::string&,bool*) const noexcept,...
Definition: ConfigParser.hpp:223
ffd::ConfigParser::get
T get(const std::string &key, const T &fallback) const noexcept
Try to get value from config, default to fallback if fails. Guaranteed no-throw.
Definition: ConfigParser.hpp:118
ffd::ConfigParser::parse
void parse(std::ifstream &file)
Iterate each line of config file and determine how to parse with l::check_record_type()
ffd::ConfigParser::get
T get(const std::string &key) const
Get value from config map using ffd::get(). This can throw. Use this in a try...catch block.
Definition: ConfigParser.hpp:103
ffd
lib45d documentation (not included in this repo, see lib45d source)
Definition: main-page.dox:21
ffd::ConfigParser::get_quota
Quota get_quota(const std::string &key, const Bytes &max, bool *fail_flag) const noexcept
Try to get Quota from config. If ffd::get fails, return Quota(void) and set fail_flag....
Definition: ConfigParser.hpp:299
ffd::ConfigParser::get_quota_from
Quota get_quota_from(const std::string &section, const std::string &key, const Bytes &max, bool *fail_flag) noexcept
Get value from config subsection using ConfigParser::get(const std::string&,bool*) const noexcept,...
Definition: ConfigParser.hpp:368
ffd::ConfigException::what
const char * what(void) const noexcept
Return string containing explanation message.
Definition: Exceptions.hpp:43
ffd::ConfigParser::get_quota
Quota get_quota(const std::string &key, const Bytes &max) const
Get quota from config map using ffd::get(). This can throw. Use this in a try...catch block.
Definition: ConfigParser.hpp:262
ffd::ConfigParser::config_map_
std::unordered_map< std::string, ConfigNode > config_map_
Map of config keys (std::string) to values (ConfigNode)
Definition: ConfigParser.hpp:398
ffd::ConfigParser::guarded_
bool guarded_
true if a ConfigSubsectionGuard is in scope Set in ConfigSubsectionGuard::ConfigSubsectionGuard() Cle...
Definition: ConfigParser.hpp:395
ffd::ConfigException
Exceptions thrown by this library.
Definition: Exceptions.hpp:30
ffd::ConfigGuardException
Throw this exception when a ConfigGuard is constructed or get_from() is called when the config is alr...
Definition: Exceptions.hpp:73
ffd::ConfigParser::current_section_
std::string current_section_
Name of current section, set by set_subsection()
Definition: ConfigParser.hpp:388
ffd::ConfigParser::set_subsection
void set_subsection(const std::string &section)
Update config_map_ptr_ to the subconfig map for section.
Definition: ConfigParser.hpp:434
ffd::ConfigParser::get_from
T get_from(const std::string &section, const std::string &key, const T &fallback) noexcept
Get value from config subsection using ConfigParser::get(const std::string&,const T&) const noexcept,...
Definition: ConfigParser.hpp:198
ffd::ConfigNode::value_
std::string value_
string from config file after '='
Definition: ConfigNode.hpp:33
ffd::ConfigParser::config_map_ptr_
std::unordered_map< std::string, ConfigNode > * config_map_ptr_
Pointer to current config map.
Definition: ConfigParser.hpp:396
ffd::ConfigParser::sub_confs_
std::vector< ConfigNode * > sub_confs_
Vector of config subsections.
Definition: ConfigParser.hpp:387
ffd::ConfigParser::get_quota_from
Quota get_quota_from(const std::string &section, const std::string &key, const Bytes &max, const Quota &fallback) noexcept
Get value from config subsection using ConfigParser::get(const std::string&,const T&) const noexcept,...
Definition: ConfigParser.hpp:344
ffd::ConfigParser::get_quota_from
Quota get_quota_from(const std::string &section, const std::string &key, const Bytes &max)
Adapter for ffd::get(). Sets config_map_ptr_ to address of sub config with name section....
Definition: ConfigParser.hpp:324
ffd::ConfigParser::reset_subsection
void reset_subsection(void) noexcept
Set config_map_ptr_ back to the address of config_map_.
Definition: ConfigParser.hpp:444
ffd::ConfigParser::report_error
void report_error(const std::string &message) const noexcept
Print error message to stderr, conditionally prepended with current subsection name.
Definition: ConfigParser.hpp:454
ffd::ConfigParser::parse_heading
void parse_heading(const std::string &line)
Create new subconfig.
ffd::ConfigParser
Main configuration parser class to inherit from in your code.
Definition: ConfigParser.hpp:67
ffd::ConfigNode
Class for config_map_ entries.
Definition: ConfigNode.hpp:31
ffd_internal
Namespace for internal use, not to be exposed to ffd.
Definition: ConfigParser.hpp:37
ffd::ConfigParser::path_
std::string path_
Path to config file.
Definition: ConfigParser.hpp:397