You've already forked mariadb-columnstore-engine
							
							
				mirror of
				https://github.com/mariadb-corporation/mariadb-columnstore-engine.git
				synced 2025-11-03 17:13:17 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			578 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			578 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* Copyright (C) 2014 InfiniDB, Inc.
 | 
						|
 | 
						|
   This program is free software; you can redistribute it and/or
 | 
						|
   modify it under the terms of the GNU General Public License
 | 
						|
   as published by the Free Software Foundation; version 2 of
 | 
						|
   the License.
 | 
						|
 | 
						|
   This program is distributed in the hope that it will be useful,
 | 
						|
   but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
   GNU General Public License for more details.
 | 
						|
 | 
						|
   You should have received a copy of the GNU General Public License
 | 
						|
   along with this program; if not, write to the Free Software
 | 
						|
   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
 | 
						|
   MA 02110-1301, USA. */
 | 
						|
 | 
						|
/******************************************************************************************
 | 
						|
 * $Id: configcpp.cpp 3899 2013-06-17 20:54:10Z rdempsey $
 | 
						|
 *
 | 
						|
 ******************************************************************************************/
 | 
						|
#include "mcsconfig.h"
 | 
						|
 | 
						|
#include <string>
 | 
						|
#include <stdexcept>
 | 
						|
#include <iostream>
 | 
						|
#include <sstream>
 | 
						|
#include <string>
 | 
						|
using namespace std;
 | 
						|
 | 
						|
#include <boost/thread.hpp>
 | 
						|
#include <boost/filesystem.hpp>
 | 
						|
#include <boost/unordered_map.hpp>
 | 
						|
using namespace boost;
 | 
						|
namespace fs = boost::filesystem;
 | 
						|
 | 
						|
#include <cassert>
 | 
						|
#include <cstring>
 | 
						|
#include <errno.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <libxml/parser.h>
 | 
						|
#include <libxml/xmlmemory.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <sys/types.h>
 | 
						|
#include <unistd.h>
 | 
						|
 | 
						|
#include "configcpp.h"
 | 
						|
 | 
						|
namespace config
 | 
						|
{
 | 
						|
 | 
						|
void Config::checkAndReloadConfig()
 | 
						|
{
 | 
						|
  struct stat statbuf;
 | 
						|
 | 
						|
  if (stat(fConfigFile.c_str(), &statbuf) == 0)
 | 
						|
  {
 | 
						|
    if (statbuf.st_mtime != fMtime)
 | 
						|
    {
 | 
						|
      closeConfig();
 | 
						|
      fMtime = statbuf.st_mtime;
 | 
						|
      parseDoc();
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
Config& Config::globConfigInstance()
 | 
						|
{
 | 
						|
  std::string configFilePath =
 | 
						|
      std::string(MCSSYSCONFDIR) + std::string("/columnstore/") + configDefaultFileName();
 | 
						|
  static Config config(configFilePath);
 | 
						|
  return config;
 | 
						|
}
 | 
						|
 | 
						|
Config* Config::makeConfig(const string& cf)
 | 
						|
{
 | 
						|
  if (cf.empty() || cf == configDefaultFileName())
 | 
						|
  {
 | 
						|
    boost::mutex::scoped_lock lk(instanceMapMutex());
 | 
						|
    globConfigInstance().checkAndReloadConfig();
 | 
						|
    return &globConfigInstance();
 | 
						|
  }
 | 
						|
 | 
						|
  boost::mutex::scoped_lock lk(instanceMapMutex());
 | 
						|
 | 
						|
  if (instanceMap().find(cf) == instanceMap().end())
 | 
						|
  {
 | 
						|
    instanceMap()[cf].reset(new Config(cf));
 | 
						|
  }
 | 
						|
  else
 | 
						|
  {
 | 
						|
    instanceMap()[cf]->checkAndReloadConfig();
 | 
						|
  }
 | 
						|
 | 
						|
  return instanceMap()[cf].get();
 | 
						|
}
 | 
						|
 | 
						|
Config* Config::makeConfig(const char* cf)
 | 
						|
{
 | 
						|
  return cf ? makeConfig(std::string(cf)) : makeConfig(std::string(""));
 | 
						|
}
 | 
						|
 | 
						|
Config::Config(const string& configFile) : fDoc(0), fConfigFile(configFile), fMtime(0), fParser()
 | 
						|
{
 | 
						|
  int i = 0;
 | 
						|
  for (; i < 2; i++)
 | 
						|
  {
 | 
						|
    if (access(fConfigFile.c_str(), R_OK) == 0)
 | 
						|
      break;
 | 
						|
    sleep(1);
 | 
						|
  }
 | 
						|
 | 
						|
  if (i == 2)
 | 
						|
    throw runtime_error("Config::Config: error accessing config file " + fConfigFile);
 | 
						|
 | 
						|
  struct stat statbuf;
 | 
						|
 | 
						|
  if (stat(configFile.c_str(), &statbuf) == 0)
 | 
						|
    fMtime = statbuf.st_mtime;
 | 
						|
 | 
						|
  parseDoc();
 | 
						|
}
 | 
						|
 | 
						|
Config::~Config()
 | 
						|
{
 | 
						|
  if (fDoc != 0)
 | 
						|
    closeConfig();
 | 
						|
}
 | 
						|
 | 
						|
void Config::parseDoc(void)
 | 
						|
{
 | 
						|
  struct flock fl;
 | 
						|
  int fd;
 | 
						|
 | 
						|
  memset(&fl, 0, sizeof(fl));
 | 
						|
  fl.l_type = F_RDLCK;  // read lock
 | 
						|
  fl.l_whence = SEEK_SET;
 | 
						|
  fl.l_start = 0;
 | 
						|
  fl.l_len = 0;  // lock whole file
 | 
						|
 | 
						|
  // lock file if exist
 | 
						|
  if ((fd = open(fConfigFile.c_str(), O_RDONLY)) >= 0)
 | 
						|
  {
 | 
						|
    if (fcntl(fd, F_SETLKW, &fl) != 0)
 | 
						|
    {
 | 
						|
      ostringstream oss;
 | 
						|
      oss << "Config::parseDoc: error locking file " << fConfigFile << ": " << strerror(errno)
 | 
						|
          << ", proceding anyway.";
 | 
						|
      cerr << oss.str() << endl;
 | 
						|
    }
 | 
						|
 | 
						|
    xmlMutex().lock();
 | 
						|
    fDoc = xmlParseFile(fConfigFile.c_str());
 | 
						|
    xmlMutex().unlock();
 | 
						|
 | 
						|
    fl.l_type = F_UNLCK;  // unlock
 | 
						|
    fcntl(fd, F_SETLK, &fl);
 | 
						|
 | 
						|
    close(fd);
 | 
						|
  }
 | 
						|
  else
 | 
						|
  {
 | 
						|
    ostringstream oss;
 | 
						|
    oss << "Config::parseDoc: error opening file " << fConfigFile << ": " << strerror(errno);
 | 
						|
    throw runtime_error(oss.str());
 | 
						|
  }
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
  {
 | 
						|
    throw runtime_error("Config::parseDoc: error parsing config file " + fConfigFile);
 | 
						|
  }
 | 
						|
 | 
						|
  xmlNodePtr cur = xmlDocGetRootElement(fDoc);
 | 
						|
 | 
						|
  if (cur == NULL)
 | 
						|
  {
 | 
						|
    xmlFreeDoc(fDoc);
 | 
						|
    fDoc = 0;
 | 
						|
    throw runtime_error("Config::parseDoc: error parsing config file " + fConfigFile);
 | 
						|
  }
 | 
						|
 | 
						|
  if (xmlStrcmp(cur->name, (const xmlChar*)"Columnstore"))
 | 
						|
  {
 | 
						|
    xmlFreeDoc(fDoc);
 | 
						|
    fDoc = 0;
 | 
						|
    throw runtime_error("Config::parseDoc: error parsing config file " + fConfigFile);
 | 
						|
  }
 | 
						|
 | 
						|
  return;
 | 
						|
}
 | 
						|
 | 
						|
void Config::closeConfig(void)
 | 
						|
{
 | 
						|
  xmlFreeDoc(fDoc);
 | 
						|
  fDoc = 0;
 | 
						|
}
 | 
						|
 | 
						|
const string Config::getConfig(const string& section, const string& name)
 | 
						|
{
 | 
						|
  if (section.length() == 0 || name.length() == 0)
 | 
						|
    throw invalid_argument("Config::getConfig: both section and name must have a length");
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
  {
 | 
						|
    throw runtime_error("Config::getConfig: no XML document!");
 | 
						|
  }
 | 
						|
 | 
						|
  return fParser.getConfig(fDoc, section, name);
 | 
						|
}
 | 
						|
 | 
						|
void Config::getConfig(const string& section, const string& name, vector<string>& values)
 | 
						|
{
 | 
						|
  if (section.length() == 0)
 | 
						|
    throw invalid_argument("Config::getConfig: section must have a length");
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
    throw runtime_error("Config::getConfig: no XML document!");
 | 
						|
 | 
						|
  fParser.getConfig(fDoc, section, name, values);
 | 
						|
}
 | 
						|
 | 
						|
const string Config::getFromActualConfig(const string& section, const string& name)
 | 
						|
{
 | 
						|
  if (section.length() == 0 || name.length() == 0)
 | 
						|
    throw invalid_argument("Config::getFromActualConfig: both section and name must have a length");
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
  {
 | 
						|
    throw runtime_error("Config::getFromActualConfig: no XML document!");
 | 
						|
  }
 | 
						|
 | 
						|
  struct stat statbuf;
 | 
						|
 | 
						|
  if (stat(fConfigFile.c_str(), &statbuf) == 0)
 | 
						|
  {
 | 
						|
    // Config was changed on disk since last read.
 | 
						|
    if (statbuf.st_mtime != fMtime)
 | 
						|
    {
 | 
						|
      boost::recursive_mutex::scoped_lock lk(fLock);
 | 
						|
      // To protect the potential race that happens right after
 | 
						|
      // the config was changed.
 | 
						|
      checkAndReloadConfig();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return fParser.getConfig(fDoc, section, name);
 | 
						|
}
 | 
						|
 | 
						|
// NB The only utility that uses setConfig is setConfig binary.
 | 
						|
// !!!Don't ever ever use this in the engine code b/c it might result in a race
 | 
						|
// b/w getConfig and setConfig methods.!!!
 | 
						|
void Config::setConfig(const string& section, const string& name, const string& value)
 | 
						|
{
 | 
						|
  boost::recursive_mutex::scoped_lock lk(fLock);
 | 
						|
 | 
						|
  if (section.length() == 0 || name.length() == 0)
 | 
						|
    throw invalid_argument("Config::setConfig: all of section and name must have a length");
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
  {
 | 
						|
    throw runtime_error("Config::setConfig: no XML document!");
 | 
						|
  }
 | 
						|
 | 
						|
  struct stat statbuf;
 | 
						|
 | 
						|
  memset(&statbuf, 0, sizeof(statbuf));
 | 
						|
  if (stat(fConfigFile.c_str(), &statbuf) == 0)
 | 
						|
  {
 | 
						|
    if (statbuf.st_mtime != fMtime)
 | 
						|
    {
 | 
						|
      closeConfig();
 | 
						|
      fMtime = statbuf.st_mtime;
 | 
						|
      parseDoc();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  fParser.setConfig(fDoc, section, name, value);
 | 
						|
  return;
 | 
						|
}
 | 
						|
 | 
						|
void Config::delConfig(const string& section, const string& name)
 | 
						|
{
 | 
						|
  boost::recursive_mutex::scoped_lock lk(fLock);
 | 
						|
 | 
						|
  if (section.length() == 0 || name.length() == 0)
 | 
						|
    throw invalid_argument("Config::delConfig: both section and name must have a length");
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
  {
 | 
						|
    throw runtime_error("Config::delConfig: no XML document!");
 | 
						|
  }
 | 
						|
 | 
						|
  checkAndReloadConfig();
 | 
						|
 | 
						|
  fParser.delConfig(fDoc, section, name);
 | 
						|
  return;
 | 
						|
}
 | 
						|
 | 
						|
void Config::writeConfig(const string& configFile) const
 | 
						|
{
 | 
						|
  boost::recursive_mutex::scoped_lock lk(fLock);
 | 
						|
  FILE* fi;
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
    throw runtime_error("Config::writeConfig: no XML document!");
 | 
						|
 | 
						|
  static const fs::path defaultConfigFilePath("Columnstore.xml");
 | 
						|
  static const fs::path defaultConfigFilePathTemp("Columnstore.xml.temp");
 | 
						|
  static const fs::path saveCalpontConfigFileTemp("Columnstore.xml.columnstoreSave");
 | 
						|
  static const fs::path tmpCalpontConfigFileTemp("Columnstore.xml.temp1");
 | 
						|
 | 
						|
  fs::path etcdir = fs::path(MCSSYSCONFDIR) / fs::path("columnstore");
 | 
						|
 | 
						|
  fs::path dcf = etcdir / fs::path(defaultConfigFilePath);
 | 
						|
  fs::path dcft = etcdir / fs::path(defaultConfigFilePathTemp);
 | 
						|
  fs::path scft = etcdir / fs::path(saveCalpontConfigFileTemp);
 | 
						|
  fs::path tcft = etcdir / fs::path(tmpCalpontConfigFileTemp);
 | 
						|
 | 
						|
  // perform a temp write first if Columnstore.xml file to prevent possible corruption
 | 
						|
  if (configFile == dcf)
 | 
						|
  {
 | 
						|
    if (exists(dcft))
 | 
						|
      fs::remove(dcft);
 | 
						|
 | 
						|
    if ((fi = fopen(dcft.string().c_str(), "w+")) == NULL)
 | 
						|
      throw runtime_error("Config::writeConfig: error writing config file " + configFile);
 | 
						|
 | 
						|
    int rc;
 | 
						|
    rc = xmlDocDump(fi, fDoc);
 | 
						|
 | 
						|
    if (rc < 0)
 | 
						|
    {
 | 
						|
      throw runtime_error("Config::writeConfig: error writing config file " + configFile);
 | 
						|
      // cout << "xmlDocDump " << rc << " " << errno << endl;
 | 
						|
    }
 | 
						|
 | 
						|
    fclose(fi);
 | 
						|
 | 
						|
    // check temp file
 | 
						|
    try
 | 
						|
    {
 | 
						|
      Config* c1 = makeConfig(dcft.string().c_str());
 | 
						|
 | 
						|
      string value;
 | 
						|
      value = c1->getConfig("SystemConfig", "SystemName");
 | 
						|
 | 
						|
      // good read, save copy, copy temp file tp tmp then to Columnstore.xml
 | 
						|
      // move to get around a 'same file error' in mv command
 | 
						|
      try
 | 
						|
      {
 | 
						|
        if (exists(scft))
 | 
						|
          fs::remove(scft);
 | 
						|
      }
 | 
						|
      catch (fs::filesystem_error&)
 | 
						|
      {
 | 
						|
      }
 | 
						|
 | 
						|
      fs::copy_file(dcf, scft, fs::copy_options::overwrite_existing);
 | 
						|
 | 
						|
      try
 | 
						|
      {
 | 
						|
        fs::permissions(scft, fs::add_perms | fs::owner_read | fs::owner_write | fs::group_read |
 | 
						|
                                  fs::group_write | fs::others_read | fs::others_write);
 | 
						|
      }
 | 
						|
      catch (fs::filesystem_error&)
 | 
						|
      {
 | 
						|
      }
 | 
						|
 | 
						|
      if (exists(tcft))
 | 
						|
        fs::remove(tcft);
 | 
						|
 | 
						|
      fs::rename(dcft, tcft);
 | 
						|
 | 
						|
      if (exists(dcf))
 | 
						|
        fs::remove(dcf);
 | 
						|
 | 
						|
      fs::rename(tcft, dcf);
 | 
						|
    }
 | 
						|
    catch (...)
 | 
						|
    {
 | 
						|
      throw runtime_error("Config::writeConfig: error writing config file " + configFile);
 | 
						|
    }
 | 
						|
  }
 | 
						|
  else
 | 
						|
  {
 | 
						|
    // non Columnstore.xml, perform update
 | 
						|
    if ((fi = fopen(configFile.c_str(), "w")) == NULL)
 | 
						|
      throw runtime_error("Config::writeConfig: error writing config file " + configFile);
 | 
						|
 | 
						|
    xmlDocDump(fi, fDoc);
 | 
						|
 | 
						|
    fclose(fi);
 | 
						|
  }
 | 
						|
 | 
						|
  return;
 | 
						|
}
 | 
						|
 | 
						|
void Config::write(void) const
 | 
						|
{
 | 
						|
  boost::mutex::scoped_lock lk(writeXmlMutex());
 | 
						|
  write(fConfigFile);
 | 
						|
}
 | 
						|
 | 
						|
void Config::write(const string& configFile) const
 | 
						|
{
 | 
						|
  struct flock fl;
 | 
						|
  int fd;
 | 
						|
 | 
						|
  fl.l_type = F_WRLCK;  // write lock
 | 
						|
  fl.l_whence = SEEK_SET;
 | 
						|
  fl.l_start = 0;
 | 
						|
  fl.l_len = 0;
 | 
						|
  fl.l_pid = getpid();
 | 
						|
 | 
						|
  // lock file if it exists
 | 
						|
  if ((fd = open(configFile.c_str(), O_WRONLY)) >= 0)
 | 
						|
  {
 | 
						|
    if (fcntl(fd, F_SETLKW, &fl) == -1)
 | 
						|
      throw runtime_error("Config::write: file lock error " + configFile);
 | 
						|
 | 
						|
    try
 | 
						|
    {
 | 
						|
      writeConfig(configFile);
 | 
						|
    }
 | 
						|
    catch (...)
 | 
						|
    {
 | 
						|
      fl.l_type = F_UNLCK;  // unlock
 | 
						|
 | 
						|
      if (fcntl(fd, F_SETLK, &fl) == -1)
 | 
						|
        throw runtime_error("Config::write: file unlock error after exception in writeConfig " + configFile);
 | 
						|
 | 
						|
      throw;
 | 
						|
    }
 | 
						|
 | 
						|
    fl.l_type = F_UNLCK;  // unlock
 | 
						|
 | 
						|
    if (fcntl(fd, F_SETLK, &fl) == -1)
 | 
						|
      throw runtime_error("Config::write: file unlock error " + configFile);
 | 
						|
 | 
						|
    close(fd);
 | 
						|
  }
 | 
						|
  else
 | 
						|
  {
 | 
						|
    writeConfig(configFile);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/* static */
 | 
						|
void Config::deleteInstanceMap()
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
/* static */
 | 
						|
int64_t Config::fromText(const std::string& text)
 | 
						|
{
 | 
						|
  if (text.length() == 0)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  int64_t val = 0;
 | 
						|
  char* ctext = static_cast<char*>(alloca(text.length() + 1));
 | 
						|
  strcpy(ctext, text.c_str());
 | 
						|
  char* cptr;
 | 
						|
 | 
						|
  val = strtoll(ctext, &cptr, 0);
 | 
						|
 | 
						|
  switch (*cptr)
 | 
						|
  {
 | 
						|
    case 'T':
 | 
						|
    case 't': val *= 1024;
 | 
						|
 | 
						|
    /* fallthru */
 | 
						|
    case 'G':
 | 
						|
    case 'g': val *= 1024;
 | 
						|
 | 
						|
    /* fallthru */
 | 
						|
    case 'M':
 | 
						|
    case 'm': val *= 1024;
 | 
						|
 | 
						|
    /* fallthru */
 | 
						|
    case 'K':
 | 
						|
    case 'k': val *= 1024;
 | 
						|
 | 
						|
    /* fallthru */
 | 
						|
    case '\0': break;
 | 
						|
 | 
						|
    default:
 | 
						|
      ostringstream oss;
 | 
						|
      oss << "Invalid character '" << *cptr << "' found in numeric parameter '" << text
 | 
						|
          << "'. Since this will not do what you want it is fatal." << endl;
 | 
						|
      throw runtime_error(oss.str());
 | 
						|
      break;
 | 
						|
  }
 | 
						|
 | 
						|
  return val;
 | 
						|
}
 | 
						|
 | 
						|
time_t Config::getCurrentMTime()
 | 
						|
{
 | 
						|
  struct stat statbuf;
 | 
						|
 | 
						|
  if (stat(fConfigFile.c_str(), &statbuf) == 0)
 | 
						|
    return statbuf.st_mtime;
 | 
						|
  else
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
// Utilized by getConfig only
 | 
						|
const vector<string> Config::enumConfig()
 | 
						|
{
 | 
						|
  boost::recursive_mutex::scoped_lock lk(fLock);
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
  {
 | 
						|
    throw runtime_error("Config::getConfig: no XML document!");
 | 
						|
  }
 | 
						|
 | 
						|
  checkAndReloadConfig();
 | 
						|
 | 
						|
  return fParser.enumConfig(fDoc);
 | 
						|
}
 | 
						|
 | 
						|
// Utilized by getConfig only
 | 
						|
const vector<string> Config::enumSection(const string& section)
 | 
						|
{
 | 
						|
  boost::recursive_mutex::scoped_lock lk(fLock);
 | 
						|
 | 
						|
  if (fDoc == 0)
 | 
						|
  {
 | 
						|
    throw runtime_error("Config::getConfig: no XML document!");
 | 
						|
  }
 | 
						|
 | 
						|
  checkAndReloadConfig();
 | 
						|
 | 
						|
  return fParser.enumSection(fDoc, section);
 | 
						|
}
 | 
						|
std::string Config::getTempFileDir(Config::TempDirPurpose what)
 | 
						|
{
 | 
						|
  std::string prefix = getConfig("SystemConfig", "SystemTempFileDir");
 | 
						|
  if (prefix.empty())
 | 
						|
  {
 | 
						|
    prefix.assign("/tmp/columnstore_tmp_files");
 | 
						|
  }
 | 
						|
  prefix.append("/");
 | 
						|
  switch (what)
 | 
						|
  {
 | 
						|
    case TempDirPurpose::Joins: return prefix.append("joins/");
 | 
						|
    case TempDirPurpose::Aggregates: return prefix.append("aggregates/");
 | 
						|
  }
 | 
						|
  // NOTREACHED
 | 
						|
  return {};
 | 
						|
}
 | 
						|
 | 
						|
bool parseBooleanParamValue(const std::string& s)
 | 
						|
{
 | 
						|
  if (s.empty())
 | 
						|
    throw runtime_error("Empty value cannot be parsed");
 | 
						|
  std::string v;
 | 
						|
  v.reserve(s.size());
 | 
						|
  for (char c : s)
 | 
						|
    v.push_back(static_cast<char>(::tolower(static_cast<unsigned char>(c))));
 | 
						|
 | 
						|
  if (v == "Y" || v == "y" || v == "1" || v == "true" || v == "on" || v == "yes" || v == "enable" ||
 | 
						|
      v == "enabled")
 | 
						|
  {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
  if (v == "N" || v == "n" || v == "0" || v == "false" || v == "off" || v == "no" || v == "disable" ||
 | 
						|
      v == "disabled")
 | 
						|
  {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  throw runtime_error("Value " + s + " cannot be parsed as boolean");
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace config
 |