2012-01-21 00:14:35 +00:00
|
|
|
#include "configurationmanager.hpp"
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <iostream>
|
2012-01-25 22:55:43 +00:00
|
|
|
#include <algorithm>
|
2016-07-11 00:09:34 +00:00
|
|
|
#include <ctype.h>
|
2012-01-21 00:14:35 +00:00
|
|
|
|
2012-02-19 22:39:37 +00:00
|
|
|
#include <boost/bind.hpp>
|
2012-02-23 22:01:42 +00:00
|
|
|
#include <boost/algorithm/string/erase.hpp>
|
2016-07-11 00:09:34 +00:00
|
|
|
#include <boost/algorithm/string/replace.hpp>
|
2014-01-19 07:34:54 +00:00
|
|
|
#include <boost/filesystem/fstream.hpp>
|
2012-02-19 22:39:37 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \namespace Files
|
|
|
|
*/
|
2012-01-21 00:14:35 +00:00
|
|
|
namespace Files
|
|
|
|
{
|
|
|
|
|
|
|
|
static const char* const openmwCfgFile = "openmw.cfg";
|
|
|
|
|
2014-07-28 19:52:34 +00:00
|
|
|
#if defined(_WIN32) || defined(__WINDOWS__)
|
|
|
|
static const char* const applicationName = "OpenMW";
|
|
|
|
#else
|
|
|
|
static const char* const applicationName = "openmw";
|
|
|
|
#endif
|
|
|
|
|
2012-01-28 10:59:08 +00:00
|
|
|
const char* const localToken = "?local?";
|
2014-06-18 15:18:48 +00:00
|
|
|
const char* const userDataToken = "?userdata?";
|
2012-01-28 10:59:08 +00:00
|
|
|
const char* const globalToken = "?global?";
|
2012-01-21 00:14:35 +00:00
|
|
|
|
2015-01-30 21:11:26 +00:00
|
|
|
ConfigurationManager::ConfigurationManager(bool silent)
|
2014-07-28 19:52:34 +00:00
|
|
|
: mFixedPath(applicationName)
|
2015-01-30 21:11:26 +00:00
|
|
|
, mSilent(silent)
|
2012-01-21 00:14:35 +00:00
|
|
|
{
|
|
|
|
setupTokensMapping();
|
|
|
|
|
2013-12-25 23:28:19 +00:00
|
|
|
boost::filesystem::create_directories(mFixedPath.getUserConfigPath());
|
2013-12-26 00:24:43 +00:00
|
|
|
boost::filesystem::create_directories(mFixedPath.getUserDataPath());
|
2012-07-22 11:52:55 +00:00
|
|
|
|
2013-12-25 23:28:19 +00:00
|
|
|
mLogPath = mFixedPath.getUserConfigPath();
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ConfigurationManager::~ConfigurationManager()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigurationManager::setupTokensMapping()
|
|
|
|
{
|
2012-02-19 22:39:37 +00:00
|
|
|
mTokensMapping.insert(std::make_pair(localToken, &FixedPath<>::getLocalPath));
|
2014-06-18 15:18:48 +00:00
|
|
|
mTokensMapping.insert(std::make_pair(userDataToken, &FixedPath<>::getUserDataPath));
|
2012-01-28 10:59:08 +00:00
|
|
|
mTokensMapping.insert(std::make_pair(globalToken, &FixedPath<>::getGlobalDataPath));
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigurationManager::readConfiguration(boost::program_options::variables_map& variables,
|
2015-06-16 02:48:45 +00:00
|
|
|
boost::program_options::options_description& description, bool quiet)
|
2012-01-21 00:14:35 +00:00
|
|
|
{
|
2015-06-16 02:48:45 +00:00
|
|
|
bool silent = mSilent;
|
|
|
|
mSilent = quiet;
|
|
|
|
|
2013-12-25 23:28:19 +00:00
|
|
|
loadConfig(mFixedPath.getUserConfigPath(), variables, description);
|
2012-01-21 00:14:35 +00:00
|
|
|
boost::program_options::notify(variables);
|
2012-01-25 22:55:43 +00:00
|
|
|
|
2015-12-02 18:21:10 +00:00
|
|
|
// read either local or global config depending on type of installation
|
|
|
|
bool loaded = loadConfig(mFixedPath.getLocalPath(), variables, description);
|
2012-01-21 00:14:35 +00:00
|
|
|
boost::program_options::notify(variables);
|
2015-12-02 18:21:10 +00:00
|
|
|
if (!loaded)
|
|
|
|
{
|
|
|
|
loadConfig(mFixedPath.getGlobalConfigPath(), variables, description);
|
|
|
|
boost::program_options::notify(variables);
|
|
|
|
}
|
2012-01-25 22:55:43 +00:00
|
|
|
|
2015-06-16 02:48:45 +00:00
|
|
|
mSilent = silent;
|
2012-01-25 22:55:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-08 12:31:20 +00:00
|
|
|
void ConfigurationManager::processPaths(Files::PathContainer& dataDirs, bool create)
|
2012-01-25 22:55:43 +00:00
|
|
|
{
|
2012-02-23 22:01:42 +00:00
|
|
|
std::string path;
|
2012-01-25 22:55:43 +00:00
|
|
|
for (Files::PathContainer::iterator it = dataDirs.begin(); it != dataDirs.end(); ++it)
|
|
|
|
{
|
2012-02-23 22:01:42 +00:00
|
|
|
path = it->string();
|
|
|
|
boost::erase_all(path, "\"");
|
2012-02-27 22:56:58 +00:00
|
|
|
*it = boost::filesystem::path(path);
|
2012-01-28 08:49:09 +00:00
|
|
|
|
|
|
|
// Check if path contains a token
|
2012-02-15 21:34:51 +00:00
|
|
|
if (!path.empty() && *path.begin() == '?')
|
2012-01-25 22:55:43 +00:00
|
|
|
{
|
2012-02-15 21:34:51 +00:00
|
|
|
std::string::size_type pos = path.find('?', 1);
|
|
|
|
if (pos != std::string::npos && pos != 0)
|
2012-01-25 22:55:43 +00:00
|
|
|
{
|
2012-02-15 21:34:51 +00:00
|
|
|
TokensMappingContainer::iterator tokenIt = mTokensMapping.find(path.substr(0, pos + 1));
|
|
|
|
if (tokenIt != mTokensMapping.end())
|
2012-01-28 08:49:09 +00:00
|
|
|
{
|
2012-02-15 21:34:51 +00:00
|
|
|
boost::filesystem::path tempPath(((mFixedPath).*(tokenIt->second))());
|
|
|
|
if (pos < path.length() - 1)
|
|
|
|
{
|
|
|
|
// There is something after the token, so we should
|
|
|
|
// append it to the path
|
|
|
|
tempPath /= path.substr(pos + 1, path.length() - pos);
|
|
|
|
}
|
|
|
|
|
2012-02-19 22:39:37 +00:00
|
|
|
*it = tempPath;
|
2012-01-28 08:49:09 +00:00
|
|
|
}
|
|
|
|
else
|
2012-01-25 22:55:43 +00:00
|
|
|
{
|
2012-02-15 21:34:51 +00:00
|
|
|
// Clean invalid / unknown token, it will be removed outside the loop
|
2012-01-28 08:49:09 +00:00
|
|
|
(*it).clear();
|
2012-01-25 22:55:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-02-19 22:39:37 +00:00
|
|
|
|
|
|
|
if (!boost::filesystem::is_directory(*it))
|
|
|
|
{
|
2013-09-08 12:31:20 +00:00
|
|
|
if (create)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
boost::filesystem::create_directories (*it);
|
|
|
|
}
|
|
|
|
catch (...) {}
|
|
|
|
|
|
|
|
if (boost::filesystem::is_directory(*it))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2012-02-19 22:39:37 +00:00
|
|
|
(*it).clear();
|
|
|
|
}
|
2012-01-25 22:55:43 +00:00
|
|
|
}
|
|
|
|
|
2012-02-19 22:39:37 +00:00
|
|
|
dataDirs.erase(std::remove_if(dataDirs.begin(), dataDirs.end(),
|
|
|
|
boost::bind(&boost::filesystem::path::empty, _1)), dataDirs.end());
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
|
2015-12-02 18:21:10 +00:00
|
|
|
bool ConfigurationManager::loadConfig(const boost::filesystem::path& path,
|
2012-01-21 00:14:35 +00:00
|
|
|
boost::program_options::variables_map& variables,
|
|
|
|
boost::program_options::options_description& description)
|
|
|
|
{
|
|
|
|
boost::filesystem::path cfgFile(path);
|
|
|
|
cfgFile /= std::string(openmwCfgFile);
|
|
|
|
if (boost::filesystem::is_regular_file(cfgFile))
|
|
|
|
{
|
2015-01-30 21:11:26 +00:00
|
|
|
if (!mSilent)
|
|
|
|
std::cout << "Loading config file: " << cfgFile.string() << "... ";
|
2012-01-21 00:14:35 +00:00
|
|
|
|
2016-07-11 00:09:34 +00:00
|
|
|
boost::filesystem::ifstream configFileStreamUnfiltered(cfgFile);
|
|
|
|
boost::iostreams::filtering_istream configFileStream;
|
|
|
|
configFileStream.push(escape_hash_filter());
|
|
|
|
configFileStream.push(configFileStreamUnfiltered);
|
|
|
|
if (configFileStreamUnfiltered.is_open())
|
2012-01-21 00:14:35 +00:00
|
|
|
{
|
|
|
|
boost::program_options::store(boost::program_options::parse_config_file(
|
2012-02-22 07:34:47 +00:00
|
|
|
configFileStream, description, true), variables);
|
2012-01-21 00:14:35 +00:00
|
|
|
|
2015-01-30 21:11:26 +00:00
|
|
|
if (!mSilent)
|
|
|
|
std::cout << "done." << std::endl;
|
2015-12-02 18:21:10 +00:00
|
|
|
return true;
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-01-30 21:11:26 +00:00
|
|
|
if (!mSilent)
|
|
|
|
std::cout << "failed." << std::endl;
|
2015-12-02 18:21:10 +00:00
|
|
|
return false;
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
}
|
2015-12-02 18:21:10 +00:00
|
|
|
return false;
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
|
2016-07-11 00:09:34 +00:00
|
|
|
escape_hash_filter::escape_hash_filter() : mNext()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
escape_hash_filter::~escape_hash_filter()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Source>
|
|
|
|
int escape_hash_filter::get(Source & src)
|
|
|
|
{
|
|
|
|
if (mNext.empty())
|
|
|
|
{
|
|
|
|
int character = boost::iostreams::get(src);
|
|
|
|
bool record = true;
|
|
|
|
if (character == boost::iostreams::WOULD_BLOCK)
|
|
|
|
{
|
|
|
|
mNext.push(character);
|
|
|
|
record = false;
|
|
|
|
}
|
|
|
|
else if (character == EOF)
|
|
|
|
{
|
|
|
|
seenNonWhitespace = false;
|
|
|
|
finishLine = false;
|
|
|
|
mNext.push(character);
|
|
|
|
}
|
|
|
|
else if (character == '\n')
|
|
|
|
{
|
|
|
|
seenNonWhitespace = false;
|
|
|
|
finishLine = false;
|
|
|
|
mNext.push(character);
|
|
|
|
}
|
|
|
|
else if (finishLine)
|
|
|
|
{
|
|
|
|
mNext.push(character);
|
|
|
|
}
|
|
|
|
else if (character == '#')
|
|
|
|
{
|
|
|
|
if (seenNonWhitespace)
|
|
|
|
{
|
|
|
|
mNext.push(sEscape);
|
|
|
|
mNext.push(sHashIdentifier);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//it's fine being interpreted by Boost as a comment, and so is anything afterwards
|
|
|
|
mNext.push(character);
|
|
|
|
finishLine = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (mPrevious == sEscape)
|
|
|
|
{
|
|
|
|
mNext.push(sEscape);
|
|
|
|
mNext.push(sEscapeIdentifier);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mNext.push(character);
|
|
|
|
}
|
|
|
|
if (!seenNonWhitespace && !isspace(character))
|
|
|
|
seenNonWhitespace = true;
|
|
|
|
if (record)
|
|
|
|
mPrevious = character;
|
|
|
|
}
|
|
|
|
int retval = mNext.front();
|
|
|
|
mNext.pop();
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string EscapeHashString::processString(const std::string & str)
|
|
|
|
{
|
|
|
|
std::string temp = boost::replace_all_copy<std::string>(str, std::string() + (char)escape_hash_filter::sEscape + (char)escape_hash_filter::sHashIdentifier, "#");
|
|
|
|
boost::replace_all(temp, std::string() + (char)escape_hash_filter::sEscape + (char)escape_hash_filter::sEscapeIdentifier, std::string((char) escape_hash_filter::sEscape, 1));
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
|
2016-07-11 00:59:16 +00:00
|
|
|
|
|
|
|
std::vector<std::string> EscapeHashString::toStdStringVector(const std::vector<EscapeHashString> & vec)
|
|
|
|
{
|
|
|
|
std::vector<std::string> temp = std::vector<std::string>();
|
|
|
|
for (std::vector<EscapeHashString>::const_iterator it = vec.begin(); it != vec.end(); ++it)
|
|
|
|
{
|
|
|
|
temp.push_back(it->toStdString());
|
|
|
|
}
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
|
2016-07-11 00:09:34 +00:00
|
|
|
EscapeHashString::EscapeHashString()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
EscapeHashString::EscapeHashString(const std::string & str) : std::string(EscapeHashString::processString(str))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
EscapeHashString::EscapeHashString(const std::string & str, size_t pos, size_t len) : std::string(EscapeHashString::processString(str), pos, len)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
EscapeHashString::EscapeHashString(const char * s) : std::string(EscapeHashString::processString(std::string(s)))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
EscapeHashString::EscapeHashString(const char * s, size_t n) : std::string(EscapeHashString::processString(std::string(s)), 0, n)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
EscapeHashString::EscapeHashString(size_t n, char c) : std::string(n, c)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class InputIterator>
|
|
|
|
EscapeHashString::EscapeHashString(InputIterator first, InputIterator last) : std::string(EscapeHashString::processString(std::string(first, last)))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-07-11 00:59:16 +00:00
|
|
|
std::string EscapeHashString::toStdString() const
|
|
|
|
{
|
|
|
|
return std::string(* this);
|
|
|
|
}
|
|
|
|
|
2012-01-21 00:14:35 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getGlobalPath() const
|
|
|
|
{
|
2013-12-25 23:28:19 +00:00
|
|
|
return mFixedPath.getGlobalConfigPath();
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
|
2013-12-26 00:24:43 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getUserConfigPath() const
|
2012-01-21 00:14:35 +00:00
|
|
|
{
|
2013-12-25 23:28:19 +00:00
|
|
|
return mFixedPath.getUserConfigPath();
|
2012-01-21 00:14:35 +00:00
|
|
|
}
|
|
|
|
|
2013-12-26 00:24:43 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getUserDataPath() const
|
|
|
|
{
|
|
|
|
return mFixedPath.getUserDataPath();
|
|
|
|
}
|
|
|
|
|
2012-01-21 00:14:35 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getLocalPath() const
|
|
|
|
{
|
|
|
|
return mFixedPath.getLocalPath();
|
|
|
|
}
|
|
|
|
|
2012-01-25 22:55:43 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getGlobalDataPath() const
|
2012-01-21 00:14:35 +00:00
|
|
|
{
|
2012-01-25 22:55:43 +00:00
|
|
|
return mFixedPath.getGlobalDataPath();
|
|
|
|
}
|
|
|
|
|
2012-09-02 17:40:26 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getCachePath() const
|
|
|
|
{
|
|
|
|
return mFixedPath.getCachePath();
|
|
|
|
}
|
|
|
|
|
2012-01-25 22:55:43 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getInstallPath() const
|
|
|
|
{
|
|
|
|
return mFixedPath.getInstallPath();
|
|
|
|
}
|
|
|
|
|
2012-01-21 00:14:35 +00:00
|
|
|
const boost::filesystem::path& ConfigurationManager::getLogPath() const
|
|
|
|
{
|
|
|
|
return mLogPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
} /* namespace Cfg */
|