From a1e223ba11bcb2f3d7aa99d4684443f1d8d00b26 Mon Sep 17 00:00:00 2001 From: Christoph Lehmann <christoph.lehmann@ufz.de> Date: Sun, 24 Jan 2016 13:54:48 +0100 Subject: [PATCH] [BL] added config tree builder --- BaseLib/ConfigTreeUtil.cpp | 74 ++++++++++++++++++++++++++++++ BaseLib/ConfigTreeUtil.h | 93 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 167 insertions(+) create mode 100644 BaseLib/ConfigTreeUtil.cpp create mode 100644 BaseLib/ConfigTreeUtil.h diff --git a/BaseLib/ConfigTreeUtil.cpp b/BaseLib/ConfigTreeUtil.cpp new file mode 100644 index 00000000000..6626af0f9b8 --- /dev/null +++ b/BaseLib/ConfigTreeUtil.cpp @@ -0,0 +1,74 @@ +/** + * \copyright + * Copyright (c) 2012-2016, 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 <boost/property_tree/xml_parser.hpp> + +#include "ConfigTreeUtil.h" + +namespace BaseLib +{ + +ConfigTreeTopLevel::ConfigTreeTopLevel( + const std::string& filepath, + const bool be_ruthless, + ConfigTreeNew::PTree&& ptree) + : _ptree(std::move(ptree)) + , _ctree(_ptree, filepath, + be_ruthless ? ConfigTreeNew::onerror : ConfigTreeNew::onwarning) +{ +} + +ConfigTreeNew const& +ConfigTreeTopLevel::operator*() const +{ + return _ctree; +} + +ConfigTreeNew const* +ConfigTreeTopLevel::operator->() const +{ + return &_ctree; +} + +void +ConfigTreeTopLevel::checkAndInvalidate() +{ + ::BaseLib::checkAndInvalidate(_ctree); +} + +ConfigTreeTopLevel +makeConfigTree(const std::string& filepath, const bool be_ruthless, + const std::string& toplevel_tag) +{ + ConfigTreeNew::PTree ptree; + + // note: Trimming whitespace and ignoring comments is crucial in order + // for our configuration tree implementation to work! + try { + read_xml(filepath, ptree, + boost::property_tree::xml_parser::no_comments | + boost::property_tree::xml_parser::trim_whitespace); + } catch (boost::property_tree::xml_parser_error e) { + ERR("Error while parsing XML file `%s' at line %lu: %s.", + e.filename().c_str(), e.line(), e.message().c_str()); + std::abort(); + } + + DBUG("Project configuration from file \'%s\' read.", filepath.c_str()); + + if (auto child = std::move(ptree.get_child_optional(toplevel_tag))) { + return ConfigTreeTopLevel(filepath, be_ruthless, std::move(*child)); + } else { + ERR("Tag <%s> has not been found in file `%s'.", + toplevel_tag.c_str(), filepath.c_str()); + std::abort(); + } +} + +} diff --git a/BaseLib/ConfigTreeUtil.h b/BaseLib/ConfigTreeUtil.h new file mode 100644 index 00000000000..80f568b5581 --- /dev/null +++ b/BaseLib/ConfigTreeUtil.h @@ -0,0 +1,93 @@ +/** + * \copyright + * Copyright (c) 2012-2016, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/project/license + * + */ + +#ifndef BASELIB_CONFIGTREEUTIL +#define BASELIB_CONFIGTREEUTIL + +#include "ConfigTreeNew.h" + +namespace BaseLib +{ + +/*! Manages a ConfigTree and the <tt>boost::property_tree</tt> it depends on. + * + * The whole purpose of this class is making the management of said dependency easy. + */ +class ConfigTreeTopLevel final +{ +public: + /*! Construct a new instance from the given data. + * + * \param filepath stored for use in error/warning messages + * \param be_ruthless if true, then warnings will raise errors, .i.e. lead to program abortion, + * else warnings will only warn + * \param ptree the underlying ptree of the created ConfigTree + */ + explicit ConfigTreeTopLevel(std::string const& filepath, + bool const be_ruthless, + ConfigTreeNew::PTree&& ptree); + + /*! Access the contained ConfigTree. + * + * The non-const version of this method has not been implemented in order to prevent invalidating + * the \c _ctree when it is passed around. In order to check and invalidate \c _ctree use the provided + * member function. + */ + ConfigTreeNew const& operator*() const; + + /*! Access the contained ConfigTree. + * + * The non-const version of this method has not been implemented in order to prevent invalidating + * the \c _ctree when it is passed around. In order to check and invalidate \c _ctree use the provided + * member function. + */ + ConfigTreeNew const* operator->() const; + + /*! Check if the contained ConfigTree has been processed entirely. + * + * This only checks the top level, as usual with ConfigTree instances. + * + * \post Afterwards the contained ConfigTree instance must not be used anymore! + */ + void checkAndInvalidate(); + +private: + ConfigTreeNew::PTree const _ptree; //!< <tt>boost::property_tree</tt> that underlies \c _ctree + ConfigTreeNew _ctree; //!< ConfigTree depending on \c _ptree +}; + +/*! Create a ConfigTree from an XML file. + * + * \param filepath see ConfigTreeTopLevel::ConfigTreeTopLevel() + * \param be_ruthless see ConfigTreeTopLevel::ConfigTreeTopLevel() + * \param toplevel_tag name of the outermost tag in the XML file. The returned ConfigTree is rooted + * one level below that tag. + * + * The parameter \c toplevel_tag is provided for compatibility with our existing configuration + * files whose toplevel tags are written in camel case, which conflicts with the naming rules of + * ConfigTree. Via that parameter the naming rules do not apply to the toplevel tag. + * + * Unfortunately the XML parser shipped with <tt>boost::property_tree</tt> does not fully support + * the XML standard. Additionally there might be encoding issues. From their docs: + * + * > Please note that RapidXML does not understand the encoding specification. If you pass it a + * > character buffer, it assumes the data is already correctly encoded; if you pass it a filename, + * > it will read the file using the character conversion of the locale you give it (or the global + * > locale if you give it none). This means that, in order to parse a UTF-8-encoded XML file into + * > a wptree, you have to supply an alternate locale, either directly or by replacing the global one. + * + * \see http://www.boost.org/doc/libs/1_60_0/doc/html/property_tree/parsers.html + */ +ConfigTreeTopLevel +makeConfigTree(std::string const& filepath, bool const be_ruthless, + std::string const& toplevel_tag); + +} + +#endif -- GitLab