mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 00:26:42 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			291 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			291 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file's contents is largely lifted from boost::program_options with only minor modification.
 | |
| // Its original preamble (without updated dates) from those source files is below:
 | |
| 
 | |
| // Copyright Vladimir Prus 2002-2004.
 | |
| // Distributed under the Boost Software License, Version 1.0.
 | |
| // (See accompanying file LICENSE_1_0.txt
 | |
| // or copy at http://www.boost.org/LICENSE_1_0.txt)
 | |
| 
 | |
| #include "configfileparser.hpp"
 | |
| 
 | |
| #include <boost/program_options/detail/config_file.hpp>
 | |
| #include <boost/program_options/detail/convert.hpp>
 | |
| 
 | |
| namespace Files
 | |
| {
 | |
|     namespace
 | |
|     {
 | |
|         /** Standalone parser for config files in ini-line format.
 | |
|             The parser is a model of single-pass lvalue iterator, and
 | |
|             default constructor creates past-the-end-iterator. The typical usage is:
 | |
|             config_file_iterator i(is, ... set of options ...), e;
 | |
|             for(; i !=e; ++i) {
 | |
|                 *i;
 | |
|             }
 | |
| 
 | |
|             Syntax conventions:
 | |
| 
 | |
|             - config file can not contain positional options
 | |
|             - '#' is comment character: it is ignored together with
 | |
|             the rest of the line.
 | |
|             - variable assignments are in the form
 | |
|             name '=' value.
 | |
|             spaces around '=' are trimmed.
 | |
|             - Section names are given in brackets.
 | |
| 
 | |
|             The actual option name is constructed by combining current section
 | |
|             name and specified option name, with dot between. If section_name
 | |
|             already contains dot at the end, new dot is not inserted. For example:
 | |
|             @verbatim
 | |
|             [gui.accessibility]
 | |
|             visual_bell=yes
 | |
|             @endverbatim
 | |
|             will result in option "gui.accessibility.visual_bell" with value
 | |
|             "yes" been returned.
 | |
| 
 | |
|             TODO: maybe, we should just accept a pointer to options_description
 | |
|             class.
 | |
|         */
 | |
|         class common_config_file_iterator : public boost::eof_iterator<common_config_file_iterator, bpo::option>
 | |
|         {
 | |
|         public:
 | |
|             common_config_file_iterator() { found_eof(); }
 | |
|             common_config_file_iterator(const std::set<std::string>& allowed_options, bool allow_unregistered = false);
 | |
| 
 | |
|             virtual ~common_config_file_iterator() {}
 | |
| 
 | |
|         public: // Method required by eof_iterator
 | |
|             void get();
 | |
| 
 | |
| #if BOOST_WORKAROUND(_MSC_VER, <= 1900)
 | |
|             void decrement() {}
 | |
|             void advance(difference_type) {}
 | |
| #endif
 | |
| 
 | |
|         protected: // Stubs for derived classes
 | |
|             // Obtains next line from the config file
 | |
|             // Note: really, this design is a bit ugly
 | |
|             // The most clean thing would be to pass 'line_iterator' to
 | |
|             // constructor of this class, but to avoid templating this class
 | |
|             // we'd need polymorphic iterator, which does not exist yet.
 | |
|             virtual bool getline(std::string&)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|         protected:
 | |
|             /** Adds another allowed option. If the 'name' ends with
 | |
|                 '*', then all options with the same prefix are
 | |
|                 allowed. For example, if 'name' is 'foo*', then 'foo1' and
 | |
|                 'foo_bar' are allowed. */
 | |
|             void add_option(const char* name);
 | |
| 
 | |
|             // Returns true if 's' is a registered option name.
 | |
|             bool allowed_option(const std::string& s) const;
 | |
| 
 | |
|             // That's probably too much data for iterator, since
 | |
|             // it will be copied, but let's not bother for now.
 | |
|             std::set<std::string> allowed_options;
 | |
|             // Invariant: no element is prefix of other element.
 | |
|             std::set<std::string> allowed_prefixes;
 | |
|             std::string m_prefix;
 | |
|             bool m_allow_unregistered = false;
 | |
|         };
 | |
| 
 | |
|         common_config_file_iterator::common_config_file_iterator(
 | |
|             const std::set<std::string>& allowed_options, bool allow_unregistered)
 | |
|             : allowed_options(allowed_options)
 | |
|             , m_allow_unregistered(allow_unregistered)
 | |
|         {
 | |
|             for (std::set<std::string>::const_iterator i = allowed_options.begin(); i != allowed_options.end(); ++i)
 | |
|             {
 | |
|                 add_option(i->c_str());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void common_config_file_iterator::add_option(const char* name)
 | |
|         {
 | |
|             std::string s(name);
 | |
|             assert(!s.empty());
 | |
|             if (*s.rbegin() == '*')
 | |
|             {
 | |
|                 s.resize(s.size() - 1);
 | |
|                 bool bad_prefixes(false);
 | |
|                 // If 's' is a prefix of one of allowed suffix, then
 | |
|                 // lower_bound will return that element.
 | |
|                 // If some element is prefix of 's', then lower_bound will
 | |
|                 // return the next element.
 | |
|                 std::set<std::string>::iterator i = allowed_prefixes.lower_bound(s);
 | |
|                 if (i != allowed_prefixes.end())
 | |
|                 {
 | |
|                     if (i->find(s) == 0)
 | |
|                         bad_prefixes = true;
 | |
|                 }
 | |
|                 if (i != allowed_prefixes.begin())
 | |
|                 {
 | |
|                     --i;
 | |
|                     if (s.find(*i) == 0)
 | |
|                         bad_prefixes = true;
 | |
|                 }
 | |
|                 if (bad_prefixes)
 | |
|                     boost::throw_exception(
 | |
|                         bpo::error("options '" + std::string(name) + "' and '" + *i
 | |
|                             + "*' will both match the same "
 | |
|                               "arguments from the configuration file"));
 | |
|                 allowed_prefixes.insert(s);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         std::string trim_ws(const std::string& s)
 | |
|         {
 | |
|             std::string::size_type n, n2;
 | |
|             n = s.find_first_not_of(" \t\r\n");
 | |
|             if (n == std::string::npos)
 | |
|                 return std::string();
 | |
|             else
 | |
|             {
 | |
|                 n2 = s.find_last_not_of(" \t\r\n");
 | |
|                 return s.substr(n, n2 - n + 1);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void common_config_file_iterator::get()
 | |
|         {
 | |
|             std::string s;
 | |
|             std::string::size_type n;
 | |
|             bool found = false;
 | |
| 
 | |
|             while (this->getline(s))
 | |
|             {
 | |
| 
 | |
|                 // strip '#' comments and whitespace
 | |
|                 if (s.find('#') == s.find_first_not_of(" \t\r\n"))
 | |
|                     continue;
 | |
|                 s = trim_ws(s);
 | |
| 
 | |
|                 if (!s.empty())
 | |
|                 {
 | |
|                     // Handle section name
 | |
|                     if (*s.begin() == '[' && *s.rbegin() == ']')
 | |
|                     {
 | |
|                         m_prefix = s.substr(1, s.size() - 2);
 | |
|                         if (*m_prefix.rbegin() != '.')
 | |
|                             m_prefix += '.';
 | |
|                     }
 | |
|                     else if ((n = s.find('=')) != std::string::npos)
 | |
|                     {
 | |
| 
 | |
|                         std::string name = m_prefix + trim_ws(s.substr(0, n));
 | |
|                         std::string value = trim_ws(s.substr(n + 1));
 | |
| 
 | |
|                         bool registered = allowed_option(name);
 | |
|                         if (!registered && !m_allow_unregistered)
 | |
|                             boost::throw_exception(bpo::unknown_option(name));
 | |
| 
 | |
|                         found = true;
 | |
|                         this->value().string_key = name;
 | |
|                         this->value().value.clear();
 | |
|                         this->value().value.push_back(value);
 | |
|                         this->value().unregistered = !registered;
 | |
|                         this->value().original_tokens.clear();
 | |
|                         this->value().original_tokens.push_back(name);
 | |
|                         this->value().original_tokens.push_back(value);
 | |
|                         break;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         boost::throw_exception(
 | |
|                             bpo::invalid_config_file_syntax(s, bpo::invalid_syntax::unrecognized_line));
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             if (!found)
 | |
|                 found_eof();
 | |
|         }
 | |
| 
 | |
|         bool common_config_file_iterator::allowed_option(const std::string& s) const
 | |
|         {
 | |
|             std::set<std::string>::const_iterator i = allowed_options.find(s);
 | |
|             if (i != allowed_options.end())
 | |
|                 return true;
 | |
|             // If s is "pa" where "p" is allowed prefix then
 | |
|             // lower_bound should find the element after "p".
 | |
|             // This depends on 'allowed_prefixes' invariant.
 | |
|             i = allowed_prefixes.lower_bound(s);
 | |
|             if (i != allowed_prefixes.begin() && s.find(*--i) == 0)
 | |
|                 return true;
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         template <class charT>
 | |
|         class basic_config_file_iterator : public Files::common_config_file_iterator
 | |
|         {
 | |
|         public:
 | |
|             basic_config_file_iterator() { found_eof(); }
 | |
| 
 | |
|             /** Creates a config file parser for the specified stream.
 | |
|              */
 | |
|             basic_config_file_iterator(std::basic_istream<charT>& is, const std::set<std::string>& allowed_options,
 | |
|                 bool allow_unregistered = false);
 | |
| 
 | |
|         private: // base overrides
 | |
|             bool getline(std::string&);
 | |
| 
 | |
|         private: // internal data
 | |
|             std::shared_ptr<std::basic_istream<charT>> is;
 | |
|         };
 | |
| 
 | |
|         template <class charT>
 | |
|         basic_config_file_iterator<charT>::basic_config_file_iterator(
 | |
|             std::basic_istream<charT>& is, const std::set<std::string>& allowed_options, bool allow_unregistered)
 | |
|             : common_config_file_iterator(allowed_options, allow_unregistered)
 | |
|         {
 | |
|             this->is.reset(&is, bpo::detail::null_deleter());
 | |
|             get();
 | |
|         }
 | |
| 
 | |
|         template <class charT>
 | |
|         bool basic_config_file_iterator<charT>::getline(std::string& s)
 | |
|         {
 | |
|             std::basic_string<charT> in;
 | |
|             if (std::getline(*is, in))
 | |
|             {
 | |
|                 s = bpo::to_internal(in);
 | |
|                 return true;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     template <class charT>
 | |
|     bpo::basic_parsed_options<charT> parse_config_file(
 | |
|         std::basic_istream<charT>& is, const bpo::options_description& desc, bool allow_unregistered)
 | |
|     {
 | |
|         std::set<std::string> allowed_options;
 | |
| 
 | |
|         const std::vector<boost::shared_ptr<bpo::option_description>>& options = desc.options();
 | |
|         for (unsigned i = 0; i < options.size(); ++i)
 | |
|         {
 | |
|             const bpo::option_description& d = *options[i];
 | |
| 
 | |
|             if (d.long_name().empty())
 | |
|                 boost::throw_exception(
 | |
|                     bpo::error("abbreviated option names are not permitted in options configuration files"));
 | |
| 
 | |
|             allowed_options.insert(d.long_name());
 | |
|         }
 | |
| 
 | |
|         // Parser return char strings
 | |
|         bpo::parsed_options result(&desc);
 | |
|         copy(basic_config_file_iterator<charT>(is, allowed_options, allow_unregistered),
 | |
|             basic_config_file_iterator<charT>(), back_inserter(result.options));
 | |
|         // Convert char strings into desired type.
 | |
|         return bpo::basic_parsed_options<charT>(result);
 | |
|     }
 | |
| 
 | |
|     template bpo::basic_parsed_options<char> parse_config_file(
 | |
|         std::basic_istream<char>& is, const bpo::options_description& desc, bool allow_unregistered);
 | |
| }
 |