Files
rippled/src/ripple/basics/impl/BasicConfig.cpp
Scott Schurr a1fd579756 Clean up Section in BasicConfig.h:
The following changes were made:
- Removed dependency on template defined in beast detail namespace.
- Removed Section::find() method which had an obsolete interface.
- Made Section::get<>() easier to use for the common case of
  retrieving a std::string.  The revised get() method replaces old
  calls to Section::find().
- Provided a default template parameter to free function
  get<>(Section config, std::string name) so it stays similar to
  Section::get<>().

Then the rest of the code was adapted to these changes.

- Calls to Section::find() were replaced with calls to Section::get.
- Unnecessary get<std::string>() arguments were reduced to get().

These changes dug up an interesting artifact in the SHAMap unit
tests.  I'm not sure why the tests were working before, but there
was a problem with the case of a Section key.  The unit test is
fixed.
2021-07-27 11:35:50 -07:00

197 lines
5.4 KiB
C++

//------------------------------------------------------------------------------
/*
This file is part of rippled: https://github.com/ripple/rippled
Copyright (c) 2012, 2013 Ripple Labs Inc.
Permission to use, copy, modify, and/or distribute this software for any
purpose with or without fee is hereby granted, provided that the above
copyright notice and this permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
//==============================================================================
#include <ripple/basics/BasicConfig.h>
#include <ripple/basics/StringUtilities.h>
#include <boost/regex.hpp>
#include <algorithm>
namespace ripple {
Section::Section(std::string const& name) : name_(name)
{
}
void
Section::set(std::string const& key, std::string const& value)
{
lookup_.insert_or_assign(key, value);
}
void
Section::append(std::vector<std::string> const& lines)
{
// <key> '=' <value>
static boost::regex const re1(
"^" // start of line
"(?:\\s*)" // whitespace (optonal)
"([a-zA-Z][_a-zA-Z0-9]*)" // <key>
"(?:\\s*)" // whitespace (optional)
"(?:=)" // '='
"(?:\\s*)" // whitespace (optional)
"(.*\\S+)" // <value>
"(?:\\s*)" // whitespace (optional)
,
boost::regex_constants::optimize);
lines_.reserve(lines_.size() + lines.size());
for (auto line : lines)
{
auto remove_comment = [](std::string& val) -> bool {
bool removed_trailing = false;
auto comment = val.find('#');
while (comment != std::string::npos)
{
if (comment == 0)
{
// entire value is a comment. In most cases, this
// would have already been handled by the file reader
val = "";
break;
}
else if (val.at(comment - 1) == '\\')
{
// we have an escaped comment char. Erase the escape char
// and keep looking
val.erase(comment - 1, 1);
}
else
{
// this must be a real comment. Extract the value
// as a substring and stop looking.
val = trim_whitespace(val.substr(0, comment));
removed_trailing = true;
break;
}
comment = val.find('#', comment);
}
return removed_trailing;
};
if (remove_comment(line) && !line.empty())
had_trailing_comments_ = true;
if (line.empty())
continue;
boost::smatch match;
if (boost::regex_match(line, match, re1))
set(match[1], match[2]);
else
values_.push_back(line);
lines_.push_back(std::move(line));
}
}
bool
Section::exists(std::string const& name) const
{
return lookup_.find(name) != lookup_.end();
}
std::ostream&
operator<<(std::ostream& os, Section const& section)
{
for (auto const& [k, v] : section.lookup_)
os << k << "=" << v << "\n";
return os;
}
//------------------------------------------------------------------------------
bool
BasicConfig::exists(std::string const& name) const
{
return map_.find(name) != map_.end();
}
Section&
BasicConfig::section(std::string const& name)
{
return map_[name];
}
Section const&
BasicConfig::section(std::string const& name) const
{
static Section none("");
auto const iter = map_.find(name);
if (iter == map_.end())
return none;
return iter->second;
}
void
BasicConfig::overwrite(
std::string const& section,
std::string const& key,
std::string const& value)
{
auto const result = map_.emplace(
std::piecewise_construct,
std::make_tuple(section),
std::make_tuple(section));
result.first->second.set(key, value);
}
void
BasicConfig::deprecatedClearSection(std::string const& section)
{
auto i = map_.find(section);
if (i != map_.end())
i->second = Section(section);
}
void
BasicConfig::legacy(std::string const& section, std::string value)
{
map_[section].legacy(std::move(value));
}
std::string
BasicConfig::legacy(std::string const& sectionName) const
{
return section(sectionName).legacy();
}
void
BasicConfig::build(IniFileSections const& ifs)
{
for (auto const& entry : ifs)
{
auto const result = map_.emplace(
std::piecewise_construct,
std::make_tuple(entry.first),
std::make_tuple(entry.first));
result.first->second.append(entry.second);
}
}
std::ostream&
operator<<(std::ostream& ss, BasicConfig const& c)
{
for (auto const& [k, v] : c.map_)
ss << "[" << k << "]\n" << v;
return ss;
}
} // namespace ripple