Skip to content
Snippets Groups Projects
Commit 0e758e68 authored by Christoph Lehmann's avatar Christoph Lehmann
Browse files

[BL] Added ConfigTree wrapper.

parent 0e2d9efc
No related branches found
No related tags found
No related merge requests found
/**
* \copyright
* Copyright (c) 2012-2015, OpenGeoSys Community (http://www.opengeosys.org)
* Distributed under a Modified BSD License.
* See accompanying file LICENSE.txt or
* http://www.opengeosys.org/project/license
*
*/
#include "ConfigTreeNew.h"
#include <logog/include/logog.hpp>
namespace BaseLib
{
//! Wraps a pair of iterators for use as a range in range-based for-loops.
template<typename Iterator>
class Range
{
public:
explicit Range(Iterator begin, Iterator end)
: _begin(begin), _end(end)
{}
Iterator begin() const { return _begin; }
Iterator end() const { return _end; }
private:
Iterator _begin;
Iterator _end;
};
template<typename T>
T
ConfigTreeNew::
getConfParam(std::string const& param)
{
auto p = getConfParamOptional<T>(param);
if (p) return *p;
error("Key <" + param + "> has not been found");
return T();
}
template<typename T>
T
ConfigTreeNew::
getConfParam(std::string const& param, T const& default_value)
{
auto p = getConfParamOptional<T>(param);
if (p) return *p;
return default_value;
}
template<typename T>
boost::optional<T>
ConfigTreeNew::
getConfParamOptional(std::string const& param)
{
checkUnique(param);
auto p = _tree->get_child_optional(param);
bool peek_only = p == boost::none;
markVisited<T>(param, peek_only);
if (p) {
auto v = p->get_value_optional<T>();
if (v) {
return v;
} else {
error("Value for key <" + param + "> `" + shortString(p->data())
+ "' not convertible to the desired type.");
}
}
return boost::none;
}
template<typename T>
Range<ConfigTreeNew::ValueIterator<T> >
ConfigTreeNew::
getConfParamList(std::string const& param)
{
checkUnique(param);
markVisited<T>(param, true);
auto p = _tree->equal_range(param);
return Range<ValueIterator<T> >(
ValueIterator<T>(p.first, param, *this),
ValueIterator<T>(p.second, param, *this));
}
template<typename T>
T
ConfigTreeNew::
peekConfParam(std::string const& param)
{
checkKeyname(param);
auto p =_tree->get_child_optional(param);
if (!p) {
error("Key <" + param + "> has not been found");
} else {
try {
return p->get_value<T>();
} catch (boost::property_tree::ptree_bad_data) {
error("Value for key <" + param + "> `" + shortString(p->data())
+ "' not convertible to the desired type.");
}
}
return T();
}
template<typename T>
void
ConfigTreeNew::
checkConfParam(std::string const& param, T const& value)
{
if (getConfParam<T>(param) != value) {
error("The value of key <" + param + "> is not the expected one.");
}
}
template<typename Ch>
void
ConfigTreeNew::
checkConfParam(std::string const& param, Ch const* value)
{
if (getConfParam<std::string>(param) != value) {
error("The value of key <" + param + "> is not the expected one.");
}
}
template<typename T>
ConfigTreeNew::CountType&
ConfigTreeNew::
markVisited(std::string const& key, bool peek_only)
{
auto const type = std::type_index(typeid(T));
auto p = _visited_params.emplace(key, CountType{peek_only ? 0 : 1, type});
if (!p.second) { // no insertion happened
auto& v = p.first->second;
if (v.type == type) {
if (!peek_only) ++v.count;
} else {
error("There already was an attempt to obtain key <" + key
+ "> with type \"" + v.type.name() + "\" (now: \"" + type.name() + "\").");
}
}
return p.first->second;
}
}
/**
* \copyright
* Copyright (c) 2012-2015, OpenGeoSys Community (http://www.opengeosys.org)
* Distributed under a Modified BSD License.
* See accompanying file LICENSE.txt or
* http://www.opengeosys.org/project/license
*
*/
#include "ConfigTreeNew.h"
namespace BaseLib
{
const char ConfigTreeNew::pathseparator = '/';
const std::string ConfigTreeNew::key_chars_start = "abcdefghijklmnopqrstuvwxyz";
const std::string ConfigTreeNew::key_chars = key_chars_start + "_0123456789";
ConfigTreeNew::
ConfigTreeNew(PTree const& tree,
Callback const& error_cb,
Callback const& warning_cb)
: _tree(&tree), _onerror(error_cb), _onwarning(warning_cb)
{
if (!_onerror) {
ERR("ConfigTree: No valid error handler provided.");
std::abort();
}
if (!_onwarning) {
ERR("ConfigTree: No valid warning handler provided.");
std::abort();
}
}
ConfigTreeNew::
ConfigTreeNew(PTree const& tree, ConfigTreeNew const& parent,
std::string const& root)
: _tree(&tree), _path(joinPaths(parent._path, root)),
_onerror(parent._onerror), _onwarning(parent._onwarning)
{
checkKeyname(root);
}
ConfigTreeNew::
ConfigTreeNew(ConfigTreeNew && other)
: _tree(other._tree)
, _path(other._path)
, _visited_params(std::move(other._visited_params))
, _onerror(other._onerror)
, _onwarning(other._onwarning)
{
other._tree = nullptr;
}
ConfigTreeNew::~ConfigTreeNew()
{
if (!_tree) return;
for (auto const& p : *_tree)
{
markVisitedDecrement(p.first);
}
for (auto const& p : _visited_params)
{
if (p.second.count > 0) {
warning("Key <" + p.first + "> has been read " + std::to_string(p.second.count)
+ " time(s) more than it was present in the configuration tree.");
} else if (p.second.count < 0) {
warning("Key <" + p.first + "> has been read " + std::to_string(-p.second.count)
+ " time(s) less than it was present in the configuration tree.");
}
}
}
ConfigTreeNew
ConfigTreeNew::
getConfSubtree(std::string const& root)
{
if (auto t = getConfSubtreeOptional(root)) {
return std::move(*t);
} else {
error("Key <" + root + "> has not been found.");
return ConfigTreeNew(PTree(), *this, ""); // TODO that will crash
}
}
boost::optional<ConfigTreeNew>
ConfigTreeNew::
getConfSubtreeOptional(std::string const& root)
{
checkUnique(root);
auto subtree = _tree->get_child_optional(root);
if (subtree) {
markVisited(root);
boost::optional<ConfigTreeNew> t;
t.emplace(*subtree, *this, root);
return std::move(t);
} else {
markVisited(root, true);
return boost::optional<ConfigTreeNew>();
}
}
Range<ConfigTreeNew::SubtreeIterator>
ConfigTreeNew::
getConfSubtreeList(std::string const& root)
{
checkUnique(root);
markVisited(root, true);
auto p = _tree->equal_range(root);
return Range<SubtreeIterator>(
SubtreeIterator(p.first, root, *this),
SubtreeIterator(p.second, root, *this));
}
void ConfigTreeNew::ignoreConfParam(const std::string &param)
{
checkUnique(param);
// if not found, peek only
bool peek_only = _tree->find(param) == _tree->not_found();
markVisited(param, peek_only);
}
void ConfigTreeNew::ignoreConfParamAll(const std::string &param)
{
checkUnique(param);
auto& ct = markVisited(param, true);
auto p = _tree->equal_range(param);
for (auto it = p.first; it != p.second; ++it) {
++ct.count;
}
}
void ConfigTreeNew::error(const std::string& message)
{
_onerror(_path, message);
}
void ConfigTreeNew::warning(const std::string& message)
{
_onwarning(_path, message);
}
void ConfigTreeNew::onerror(const std::string& path, const std::string& message)
{
ERR("ConfigTree: At path <%s>: %s", path.c_str(), message.c_str());
std::abort();
}
void ConfigTreeNew::onwarning(const std::string& path, const std::string& message)
{
WARN("ConfigTree: At path <%s>: %s", path.c_str(), message.c_str());
}
std::string ConfigTreeNew::shortString(const std::string &s)
{
const std::size_t maxlen = 100;
if (s.size() < maxlen) return s;
return s.substr(0, maxlen-3) + "...";
}
void ConfigTreeNew::checkKeyname(std::string const& key)
{
if (key.empty()) {
error("Search for empty key.");
} else if (key_chars_start.find(key.front()) == std::string::npos) {
error("Key <" + key + "> starts with an illegal character.");
} else if (key.find_first_not_of(key_chars, 1) != std::string::npos) {
error("Key <" + key + "> contains illegal characters.");
}
}
std::string ConfigTreeNew::
joinPaths( const std::string &p1, const std::string &p2)
{
if (p2.empty()) {
error("Second path to be joined is empty.");
}
if (p1.empty()) return p2;
return p1 + pathseparator + p2;
}
void ConfigTreeNew::checkUnique(const std::string &key)
{
checkKeyname(key);
if (_visited_params.find(key) != _visited_params.end()) {
error("Key <" + key + "> has already been processed.");
}
}
ConfigTreeNew::CountType&
ConfigTreeNew::
markVisited(std::string const& key, bool peek_only)
{
return markVisited<ConfigTreeNew>(key, peek_only);
}
void
ConfigTreeNew::
markVisitedDecrement(std::string const& key)
{
auto const type = std::type_index(typeid(nullptr));
auto p = _visited_params.emplace(key, CountType{-1, type});
if (!p.second) { // no insertion happened
auto& v = p.first->second;
--v.count;
}
}
}
/**
* \copyright
* Copyright (c) 2012-2015, OpenGeoSys Community (http://www.opengeosys.org)
* Distributed under a Modified BSD License.
* See accompanying file LICENSE.txt or
* http://www.opengeosys.org/project/license
*
*/
#pragma once
#include "ConfigTree.h"
#include <typeindex>
#include <map>
#include <functional>
namespace BaseLib
{
template<typename Iterator> class Range;
/*!
* Wrapper around a Boost Property Tree with some basic error reporting features.
*
* Features. This class:
* * makes sure that every configuration setting in a Property Tree is read
* exactly once. If some settings is not read (e.g. due to a typo), a warning message
* is generated. The message contains a hint where it occured.
* * enforces a naming scheme of settings: letters a-z, numbers 0-9, underscore
* * provides some functionality to read lists of values using range-based for loops.
* * has rather long method names that are easily greppable from the source code. So a list
* of supported configuration options can be easily obtained from the source code.
*
* The purpose of this class is to reduce or completely avoid the amount of error-handling
* code in routines that take configuration parameters.
*
* Most methods of this class check that they have not been called before for the same
* \c ConfigTree and the same parameter. This behaviour helps to enforce that every parameter
* is read exactly once during parsing of the configuration settings.
*
* The most notable restriction of this class when compared to plain tree traversal is, that
* one must know all the XML tags (i.e. configuration parameters) at compile time. It is not
* possible to read from this class, which configuration parameters are present in the tree.
* This restriction, however, is intended, because it provides the possibility to get all
* existing configuration parameters from the source code.
*/
class ConfigTreeNew final
{
public:
/*!
* A wrapper around a Boost Iterator for iterating over ranges of subtrees.
*
* The methods of this class tell the associated (parent) \c ConfigTree object when
* a setting has been parsed.
*/
class SubtreeIterator
: public std::iterator<std::input_iterator_tag, ConfigTreeNew>
{
public:
using Iterator = boost::property_tree::ptree::const_assoc_iterator;
explicit SubtreeIterator(Iterator it, std::string const& root,
ConfigTreeNew& parent)
: _it(it), _root(root), _parent(parent)
{}
SubtreeIterator& operator++() {
++_it;
_has_incremented = true;
return *this;
}
ConfigTreeNew operator*() {
// if this iterator has been incremented since the last dereference,
// tell the _parent instance that a subtree now has been parsed.
if (_has_incremented) {
_has_incremented = false;
_parent.markVisited(_root);
}
return ConfigTreeNew(_it->second, _parent, _root);
}
bool operator==(SubtreeIterator const& other) const {
return _it == other._it;
}
bool operator!=(SubtreeIterator const& other) const {
return _it != other._it;
}
private:
bool _has_incremented = true;
Iterator _it;
std::string const _root;
ConfigTreeNew& _parent;
};
/*!
* A wrapper around a Boost Iterator for iterating over ranges of values.
*
* The methods of this class tell the associated (parent) \c ConfigTree object when
* a setting has been parsed.
*/
template<typename ValueType>
class ValueIterator
: public std::iterator<std::input_iterator_tag, ValueType>
{
public:
using Iterator = boost::property_tree::ptree::const_assoc_iterator;
explicit ValueIterator(Iterator it, std::string const& root,
ConfigTreeNew& parent)
: _it(it), _root(root), _parent(parent)
{}
ValueIterator<ValueType>& operator++() {
++_it;
_has_incremented = true;
return *this;
}
ValueType operator*() {
// if this iterator has been incremented since the last dereference,
// tell the _parent instance that a setting now has been parsed.
if (_has_incremented) {
_has_incremented = false;
_parent.markVisited<ValueType>(_root);
}
if (_it->second.begin() != _it->second.end()) {
_parent.error("Configuration at key " + _root + " has subitems.");
return ValueType();
}
auto v = _it->second.get_value_optional<ValueType>();
if (v) return *v;
// TODO: change error method
_parent.error("Could not get value out of key " + _root + ".");
return ValueType();
}
bool operator==(ValueIterator<ValueType> const& other) const {
return _it == other._it;
}
bool operator!=(ValueIterator<ValueType> const& other) const {
return _it != other._it;
}
private:
bool _has_incremented = true;
Iterator _it;
std::string const _root;
ConfigTreeNew& _parent;
};
using PTree = boost::property_tree::ptree;
//! Type of the function objects used as callbacks.
//! The first argument denotes the path in the tree at which an event (warning/error)
//! occured, the second argument is the associated message
using Callback = std::function<void(const std::string& path,
const std::string& message)>;
/*! Creates a new instance wrapping the given Boost Property Tree.
*
* \param tree the Boost Property Tree to be wrapped
* \param onerror callback function to be called on error.
* \param onwarning callback function to be called on warning.
*
* The callback functions must be valid callable functions, i.e. not nullptr's.
* They are configurable in order to make unit tests of this class easier.
* They should not be provided in production code!
*
* If a custom error callback is provided, this function should break out of
* the normal execution order, e.g., by throwing or by calling std::abort(),
* because otherwise this class will effectively treat errors as no-errors.
*/
explicit ConfigTreeNew(PTree const& tree,
Callback const& error_cb = onerror,
Callback const& warning_cb = onwarning);
//! Used for wrapping a subtree
explicit ConfigTreeNew(PTree const& tree, ConfigTreeNew const& parent, std::string const& root);
//! copying is not compatible with the semantics of this class
ConfigTreeNew(ConfigTreeNew const&) = delete;
//! After being moved from, \c other is in an undefined state and must not be
//! used anymore!
ConfigTreeNew(ConfigTreeNew && other);
ConfigTreeNew() = delete;
void operator=(ConfigTreeNew const&) = delete;
void operator=(ConfigTreeNew &&) = delete;
/*! Get parameter \c param of type \c T from the configuration tree.
*
* \return the value looked for or a default constructed value \c T() in the case of
* an error. For the behaviour in case of an error, see also the documentation
* of the method error().
*
* \pre \c param must not have been read before from this ConfigTree.
*/
template<typename T> T
getConfParam(std::string const& param);
/*! Get parameter \c param of type \c T from the configuration tree or the \c default_value.
*
* This method has a similar behaviour as getConfParam(std::string const&) except in case
* of errors the \c default_value is returned.
*
* \pre \c param must not have been read before from this ConfigTree.
*/
template<typename T> T
getConfParam(std::string const& param, T const& default_value);
/*! Get parameter \c param of type \c T from the configuration tree if present
*
* This method has a similar behaviour as getConfParam(std::string const&) except
* no errors are raised. Rather it can be told from the return value if the
* parameter could be read.
*
* \pre \c param must not have been read before from this ConfigTree.
*/
template<typename T> boost::optional<T>
getConfParamOptional(std::string const& param);
/*! Returns all parameters with the name \c param from the current level of the tree.
*
* The return value is suitable to be used with range-base for-loops.
*
* \pre \c param must not have been read before from this ConfigTree.
*/
template<typename T> Range<ValueIterator<T> >
getConfParamList(std::string const& param);
/*! Peek at a parameter \c param of type \c T from the configuration tree.
*
* This method is an exception to the single-read rule. It is meant to be used to
* tell from a ConfigTree instance where to pass that instance on for further processing.
*
* Return value and error behaviour are the same as for getConfParam(std::string const&).
*/
template<typename T> T
peekConfParam(std::string const& param);
/*! Assert that \c param has the given \c value.
*
* Convenience method combining getConfParam(std::string const&) with a check.
*/
template<typename T> void
checkConfParam(std::string const& param, T const& value);
//! Make checkConfParam() work for string literals.
template<typename Ch> void
checkConfParam(std::string const& param, Ch const* value);
/*! Get the subtree rooted at \c root
*
* If \c root is not found error() is called.
*
* \pre \c root must not have been read before from this ConfigTree.
*/
ConfigTreeNew
getConfSubtree(std::string const& root);
/*! Get the subtree rooted at \c root if present
*
* \pre \c root must not have been read before from this ConfigTree.
*/
boost::optional<ConfigTreeNew>
getConfSubtreeOptional(std::string const& root);
/*! Get all subtrees that have a root \c root from the current level of the tree.
*
* The return value is suitable to be used with range-base for-loops.
*
* \pre \c root must not have been read before from this ConfigTree.
*/
Range<SubtreeIterator>
getConfSubtreeList(std::string const& root);
/*! Tell this instance to ignore parameter \c param.
*
* This method is used to avoid warning messages.
*
* \pre \c root must not have been read before from this ConfigTree.
*/
void ignoreConfParam(std::string const& param);
/*! Tell this instance to ignore all parameters \c param on the current level of the tree.
*
* This method is used to avoid warning messages.
*
* \pre \c root must not have been read before from this ConfigTree.
*/
void ignoreConfParamAll(std::string const& param);
//! The destructor performs the check if all nodes at the current level of the tree
//! have been read.
~ConfigTreeNew();
private:
struct CountType
{
int count;
std::type_index type;
};
//! Called if an error occurs. Will call the error callback.
//! This method only acts as a helper method.
void error(std::string const& message);
//! Called for printing warning messages. Will call the warning callback.
//! This method only acts as a helper method.
void warning(std::string const& message);
//! Checks if \c key complies with the rules [a-z0-9_].
void checkKeyname(std::string const& key);
//! Used to generate the path of a subtree.
std::string joinPaths(std::string const& p1, std::string const& p2);
//! Asserts that the \c key has not been read yet.
void checkUnique(std::string const& key);
/*! Keeps track of the key \c key and its value type \c T.
*
* This method asserts that a key is read always with the same type.
*
* \c param peek_only if true, do not change the read-count of the given key.
*/
template<typename T>
CountType& markVisited(std::string const& key, bool peek_only = false);
/*! Keeps track of the key \c key and its value type ConfigTree.
*
* This method asserts that a key is read always with the same type.
*
* \c param peek_only if true, do not change the read-count of the given key.
*/
CountType& markVisited(std::string const& key, bool peek_only = false);
//! Used in the destructor to compute the difference between number of reads of a parameter
//! and the number of times it exists in the ConfigTree
void markVisitedDecrement(std::string const& key);
//! Default error callback function
//! Will print an error message and call std::abort()
static void onerror(std::string const& path, std::string const& message);
//! Default warning callback function
//! Will print a warning message
static void onwarning(std::string const& path, std::string const& message);
//! returns a short string at suitable for error/warning messages
static std::string shortString(std::string const& s);
//! The wrapped tree.
boost::property_tree::ptree const* _tree;
//! A path printed in error/warning messages.
std::string const _path;
//! A map key -> (count, type) keeping track which parameters have been read how often
//! and which datatype they have.
std::map<std::string, CountType> _visited_params;
const Callback _onerror;
const Callback _onwarning;
//! Character separating two path components.
static const char pathseparator;
//! Set of allowed characters as the first letter of a key name.
static const std::string key_chars_start;
//! Set of allowed characters in a key name.
static const std::string key_chars;
};
}
#include "ConfigTreeNew-impl.h"
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment