mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-11-04 02:56:39 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			435 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			435 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include "configurationmanager.hpp"
 | 
						|
 | 
						|
#include <components/debug/debuglog.hpp>
 | 
						|
#include <components/files/configfileparser.hpp>
 | 
						|
#include <components/fallback/validate.hpp>
 | 
						|
 | 
						|
#include <boost/filesystem/fstream.hpp>
 | 
						|
/**
 | 
						|
 * \namespace Files
 | 
						|
 */
 | 
						|
namespace Files
 | 
						|
{
 | 
						|
 | 
						|
static const char* const openmwCfgFile = "openmw.cfg";
 | 
						|
 | 
						|
#if defined(_WIN32) || defined(__WINDOWS__)
 | 
						|
static const char* const applicationName = "OpenMW";
 | 
						|
#else
 | 
						|
static const char* const applicationName = "openmw";
 | 
						|
#endif
 | 
						|
 | 
						|
const char* const localToken = "?local?";
 | 
						|
const char* const userConfigToken = "?userconfig?";
 | 
						|
const char* const userDataToken = "?userdata?";
 | 
						|
const char* const globalToken = "?global?";
 | 
						|
 | 
						|
ConfigurationManager::ConfigurationManager(bool silent)
 | 
						|
    : mFixedPath(applicationName)
 | 
						|
    , mSilent(silent)
 | 
						|
{
 | 
						|
    setupTokensMapping();
 | 
						|
 | 
						|
    // Initialize with fixed paths, will be overridden in `readConfiguration`.
 | 
						|
    mLogPath = mFixedPath.getUserConfigPath();
 | 
						|
    mUserDataPath = mFixedPath.getUserDataPath();
 | 
						|
    mScreenshotPath = mFixedPath.getUserDataPath() / "screenshots";
 | 
						|
}
 | 
						|
 | 
						|
ConfigurationManager::~ConfigurationManager()
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
void ConfigurationManager::setupTokensMapping()
 | 
						|
{
 | 
						|
    mTokensMapping.insert(std::make_pair(localToken, &FixedPath<>::getLocalPath));
 | 
						|
    mTokensMapping.insert(std::make_pair(userConfigToken, &FixedPath<>::getUserConfigPath));
 | 
						|
    mTokensMapping.insert(std::make_pair(userDataToken, &FixedPath<>::getUserDataPath));
 | 
						|
    mTokensMapping.insert(std::make_pair(globalToken, &FixedPath<>::getGlobalDataPath));
 | 
						|
}
 | 
						|
 | 
						|
void ConfigurationManager::readConfiguration(boost::program_options::variables_map& variables,
 | 
						|
    boost::program_options::options_description& description, bool quiet)
 | 
						|
{
 | 
						|
    using ParsedConfigFile = bpo::basic_parsed_options<char>;
 | 
						|
    bool silent = mSilent;
 | 
						|
    mSilent = quiet;
 | 
						|
 | 
						|
    std::optional<ParsedConfigFile> config = loadConfig(mFixedPath.getLocalPath(), description);
 | 
						|
    if (config)
 | 
						|
        mActiveConfigPaths.push_back(mFixedPath.getLocalPath());
 | 
						|
    else
 | 
						|
    {
 | 
						|
        mActiveConfigPaths.push_back(mFixedPath.getGlobalConfigPath());
 | 
						|
        config = loadConfig(mFixedPath.getGlobalConfigPath(), description);
 | 
						|
    }
 | 
						|
    if (!config)
 | 
						|
    {
 | 
						|
        if (!quiet)
 | 
						|
            Log(Debug::Error) << "Neither local config nor global config are available.";
 | 
						|
        mSilent = silent;
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    std::stack<boost::filesystem::path> extraConfigDirs;
 | 
						|
    addExtraConfigDirs(extraConfigDirs, variables);
 | 
						|
    addExtraConfigDirs(extraConfigDirs, *config);
 | 
						|
 | 
						|
    std::vector<ParsedConfigFile> parsedOptions{*std::move(config)};
 | 
						|
    std::set<boost::filesystem::path> alreadyParsedPaths;  // needed to prevent infinite loop in case of a circular link
 | 
						|
    alreadyParsedPaths.insert(boost::filesystem::path(mActiveConfigPaths.front()));
 | 
						|
 | 
						|
    while (!extraConfigDirs.empty())
 | 
						|
    {
 | 
						|
        boost::filesystem::path path = extraConfigDirs.top();
 | 
						|
        extraConfigDirs.pop();
 | 
						|
        if (alreadyParsedPaths.count(path) > 0)
 | 
						|
        {
 | 
						|
            if (!quiet)
 | 
						|
                Log(Debug::Warning) << "Repeated config dir: " << path;
 | 
						|
            continue;
 | 
						|
        }
 | 
						|
        alreadyParsedPaths.insert(path);
 | 
						|
        mActiveConfigPaths.push_back(path);
 | 
						|
        config = loadConfig(path, description);
 | 
						|
        if (!config)
 | 
						|
            continue;
 | 
						|
        addExtraConfigDirs(extraConfigDirs, *config);
 | 
						|
        parsedOptions.push_back(*std::move(config));
 | 
						|
    }
 | 
						|
 | 
						|
    for (auto it = parsedOptions.rbegin(); it != parsedOptions.rend(); ++it)
 | 
						|
    {
 | 
						|
        auto composingVariables = separateComposingVariables(variables, description);
 | 
						|
        boost::program_options::store(std::move(*it), variables);
 | 
						|
        mergeComposingVariables(variables, composingVariables, description);
 | 
						|
    }
 | 
						|
 | 
						|
    mLogPath = mActiveConfigPaths.back();
 | 
						|
    mUserDataPath = variables["user-data"].as<Files::MaybeQuotedPath>();
 | 
						|
    if (mUserDataPath.empty())
 | 
						|
    {
 | 
						|
        if (!quiet)
 | 
						|
            Log(Debug::Warning) << "Error: `user-data` is not specified";
 | 
						|
        mUserDataPath = mFixedPath.getUserDataPath();
 | 
						|
    }
 | 
						|
    processPath(mUserDataPath, true);
 | 
						|
    mScreenshotPath = mUserDataPath / "screenshots";
 | 
						|
 | 
						|
    boost::filesystem::create_directories(getUserConfigPath());
 | 
						|
    boost::filesystem::create_directories(mScreenshotPath);
 | 
						|
 | 
						|
    // probably not necessary but validate the creation of the screenshots directory and fallback to the original behavior if it fails
 | 
						|
    if (!boost::filesystem::is_directory(mScreenshotPath))
 | 
						|
        mScreenshotPath = mUserDataPath;
 | 
						|
 | 
						|
    if (!quiet && !variables["replace"].empty())
 | 
						|
    {
 | 
						|
        for (const std::string& var : variables["replace"].as<std::vector<std::string>>())
 | 
						|
        {
 | 
						|
            if (var == "config")
 | 
						|
            {
 | 
						|
                Log(Debug::Warning) << "replace=config is not allowed and was ignored";
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!quiet)
 | 
						|
    {
 | 
						|
        Log(Debug::Info) << "Logs dir: " << getUserConfigPath().string();
 | 
						|
        Log(Debug::Info) << "User data dir: " << mUserDataPath.string();
 | 
						|
        Log(Debug::Info) << "Screenshots dir: " << mScreenshotPath.string();
 | 
						|
    }
 | 
						|
 | 
						|
    mSilent = silent;
 | 
						|
}
 | 
						|
 | 
						|
void ConfigurationManager::addExtraConfigDirs(std::stack<boost::filesystem::path>& dirs,
 | 
						|
                                              const bpo::basic_parsed_options<char>& options) const
 | 
						|
{
 | 
						|
    boost::program_options::variables_map variables;
 | 
						|
    boost::program_options::store(options, variables);
 | 
						|
    boost::program_options::notify(variables);
 | 
						|
    addExtraConfigDirs(dirs, variables);
 | 
						|
}
 | 
						|
 | 
						|
void ConfigurationManager::addExtraConfigDirs(std::stack<boost::filesystem::path>& dirs,
 | 
						|
                                              const boost::program_options::variables_map& variables) const
 | 
						|
{
 | 
						|
    auto configIt = variables.find("config");
 | 
						|
    if (configIt == variables.end())
 | 
						|
        return;
 | 
						|
    Files::PathContainer newDirs = asPathContainer(configIt->second.as<Files::MaybeQuotedPathContainer>());
 | 
						|
    processPaths(newDirs);
 | 
						|
    for (auto it = newDirs.rbegin(); it != newDirs.rend(); ++it)
 | 
						|
        dirs.push(*it);
 | 
						|
}
 | 
						|
 | 
						|
void ConfigurationManager::addCommonOptions(boost::program_options::options_description& description)
 | 
						|
{
 | 
						|
    description.add_options()
 | 
						|
        ("config", bpo::value<Files::MaybeQuotedPathContainer>()->multitoken()->composing(), "additional config directories")
 | 
						|
        ("user-data", bpo::value<Files::MaybeQuotedPath>(),
 | 
						|
            "set user data directory (used for saves, screenshots, etc)");
 | 
						|
}
 | 
						|
 | 
						|
boost::program_options::variables_map separateComposingVariables(boost::program_options::variables_map & variables,
 | 
						|
    boost::program_options::options_description& description)
 | 
						|
{
 | 
						|
    boost::program_options::variables_map composingVariables;
 | 
						|
    for (auto itr = variables.begin(); itr != variables.end();)
 | 
						|
    {
 | 
						|
        if (description.find(itr->first, false).semantic()->is_composing())
 | 
						|
        {
 | 
						|
            composingVariables.emplace(*itr);
 | 
						|
            itr = variables.erase(itr);
 | 
						|
        }
 | 
						|
        else
 | 
						|
            ++itr;
 | 
						|
    }
 | 
						|
    return composingVariables;
 | 
						|
}
 | 
						|
 | 
						|
void mergeComposingVariables(boost::program_options::variables_map& first, boost::program_options::variables_map& second,
 | 
						|
    boost::program_options::options_description& description)
 | 
						|
{
 | 
						|
    // There are a few places this assumes all variables are present in second, but it's never crashed in the wild, so it looks like that's guaranteed.
 | 
						|
    std::set<std::string> replacedVariables;
 | 
						|
    if (description.find_nothrow("replace", false))
 | 
						|
    {
 | 
						|
        auto replace = second["replace"];
 | 
						|
        if (!replace.defaulted() && !replace.empty())
 | 
						|
        {
 | 
						|
            std::vector<std::string> replaceVector = replace.as<std::vector<std::string>>();
 | 
						|
            replacedVariables.insert(replaceVector.begin(), replaceVector.end());
 | 
						|
        }
 | 
						|
    }
 | 
						|
    for (const auto& option : description.options())
 | 
						|
    {
 | 
						|
        if (option->semantic()->is_composing())
 | 
						|
        {
 | 
						|
            std::string name = option->canonical_display_name();
 | 
						|
 | 
						|
            auto firstPosition = first.find(name);
 | 
						|
            if (firstPosition == first.end())
 | 
						|
            {
 | 
						|
                first.emplace(name, second[name]);
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            if (replacedVariables.count(name) || firstPosition->second.defaulted() || firstPosition->second.empty())
 | 
						|
            {
 | 
						|
                firstPosition->second = second[name];
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            if (second[name].defaulted() || second[name].empty())
 | 
						|
                continue;
 | 
						|
 | 
						|
            boost::any& firstValue = firstPosition->second.value();
 | 
						|
            const boost::any& secondValue = second[name].value();
 | 
						|
            
 | 
						|
            if (firstValue.type() == typeid(Files::MaybeQuotedPathContainer))
 | 
						|
            {
 | 
						|
                auto& firstPathContainer = boost::any_cast<Files::MaybeQuotedPathContainer&>(firstValue);
 | 
						|
                const auto& secondPathContainer = boost::any_cast<const Files::MaybeQuotedPathContainer&>(secondValue);
 | 
						|
                firstPathContainer.insert(firstPathContainer.end(), secondPathContainer.begin(), secondPathContainer.end());
 | 
						|
            }
 | 
						|
            else if (firstValue.type() == typeid(std::vector<std::string>))
 | 
						|
            {
 | 
						|
                auto& firstVector = boost::any_cast<std::vector<std::string>&>(firstValue);
 | 
						|
                const auto& secondVector = boost::any_cast<const std::vector<std::string>&>(secondValue);
 | 
						|
 | 
						|
                firstVector.insert(firstVector.end(), secondVector.begin(), secondVector.end());
 | 
						|
            }
 | 
						|
            else if (firstValue.type() == typeid(Fallback::FallbackMap))
 | 
						|
            {
 | 
						|
                auto& firstMap = boost::any_cast<Fallback::FallbackMap&>(firstValue);
 | 
						|
                const auto& secondMap = boost::any_cast<const Fallback::FallbackMap&>(secondValue);
 | 
						|
 | 
						|
                std::map<std::string, std::string> tempMap(secondMap.mMap);
 | 
						|
                tempMap.merge(firstMap.mMap);
 | 
						|
                firstMap.mMap.swap(tempMap);
 | 
						|
            }
 | 
						|
            else
 | 
						|
                Log(Debug::Error) << "Unexpected composing variable type. Curse boost and their blasted arcane templates.";
 | 
						|
        }
 | 
						|
    }
 | 
						|
    boost::program_options::notify(first);
 | 
						|
}
 | 
						|
 | 
						|
void ConfigurationManager::processPath(boost::filesystem::path& path, bool create) const
 | 
						|
{
 | 
						|
    std::string str = path.string();
 | 
						|
 | 
						|
    // Do nothing if the path doesn't start with a token
 | 
						|
    if (str.empty() || str[0] != '?')
 | 
						|
        return;
 | 
						|
 | 
						|
    std::string::size_type pos = str.find('?', 1);
 | 
						|
    if (pos != std::string::npos && pos != 0)
 | 
						|
    {
 | 
						|
        auto tokenIt = mTokensMapping.find(str.substr(0, pos + 1));
 | 
						|
        if (tokenIt != mTokensMapping.end())
 | 
						|
        {
 | 
						|
            boost::filesystem::path tempPath(((mFixedPath).*(tokenIt->second))());
 | 
						|
            if (pos < str.length() - 1)
 | 
						|
            {
 | 
						|
                // There is something after the token, so we should
 | 
						|
                // append it to the path
 | 
						|
                tempPath /= str.substr(pos + 1, str.length() - pos);
 | 
						|
            }
 | 
						|
 | 
						|
            path = tempPath;
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            if (!mSilent)
 | 
						|
                Log(Debug::Warning) << "Path starts with unknown token: " << path;
 | 
						|
            path.clear();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!boost::filesystem::is_directory(path) && create)
 | 
						|
    {
 | 
						|
        try
 | 
						|
        {
 | 
						|
            boost::filesystem::create_directories(path);
 | 
						|
        }
 | 
						|
        catch (...) {}
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void ConfigurationManager::processPaths(Files::PathContainer& dataDirs, bool create) const
 | 
						|
{
 | 
						|
    for (Files::PathContainer::iterator it = dataDirs.begin(); it != dataDirs.end(); ++it)
 | 
						|
    {
 | 
						|
        processPath(*it, create);
 | 
						|
        if (!boost::filesystem::is_directory(*it))
 | 
						|
        {
 | 
						|
            if (!mSilent)
 | 
						|
                Log(Debug::Warning) << "No such dir: " << *it;
 | 
						|
            (*it).clear();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    dataDirs.erase(std::remove_if(dataDirs.begin(), dataDirs.end(),
 | 
						|
        std::bind(&boost::filesystem::path::empty, std::placeholders::_1)), dataDirs.end());
 | 
						|
}
 | 
						|
 | 
						|
std::optional<bpo::basic_parsed_options<char>> ConfigurationManager::loadConfig(
 | 
						|
    const boost::filesystem::path& path,
 | 
						|
    boost::program_options::options_description& description)
 | 
						|
{
 | 
						|
    boost::filesystem::path cfgFile(path);
 | 
						|
    cfgFile /= std::string(openmwCfgFile);
 | 
						|
    if (boost::filesystem::is_regular_file(cfgFile))
 | 
						|
    {
 | 
						|
        if (!mSilent)
 | 
						|
            Log(Debug::Info) << "Loading config file: " << cfgFile.string();
 | 
						|
 | 
						|
        boost::filesystem::ifstream configFileStream(cfgFile);
 | 
						|
 | 
						|
        if (configFileStream.is_open())
 | 
						|
            return Files::parse_config_file(configFileStream, description, true);
 | 
						|
        else if (!mSilent)
 | 
						|
            Log(Debug::Error) << "Loading failed.";
 | 
						|
    }
 | 
						|
    return std::nullopt;
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getGlobalPath() const
 | 
						|
{
 | 
						|
    return mFixedPath.getGlobalConfigPath();
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getUserConfigPath() const
 | 
						|
{
 | 
						|
    if (mActiveConfigPaths.empty())
 | 
						|
        return mFixedPath.getUserConfigPath();
 | 
						|
    else
 | 
						|
        return mActiveConfigPaths.back();
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getUserDataPath() const
 | 
						|
{
 | 
						|
    return mUserDataPath;
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getLocalPath() const
 | 
						|
{
 | 
						|
    return mFixedPath.getLocalPath();
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getGlobalDataPath() const
 | 
						|
{
 | 
						|
    return mFixedPath.getGlobalDataPath();
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getCachePath() const
 | 
						|
{
 | 
						|
    return mFixedPath.getCachePath();
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getInstallPath() const
 | 
						|
{
 | 
						|
    return mFixedPath.getInstallPath();
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getLogPath() const
 | 
						|
{
 | 
						|
    return mLogPath;
 | 
						|
}
 | 
						|
 | 
						|
const boost::filesystem::path& ConfigurationManager::getScreenshotPath() const
 | 
						|
{
 | 
						|
    return mScreenshotPath;
 | 
						|
}
 | 
						|
 | 
						|
void parseArgs(int argc, const char* const argv[], boost::program_options::variables_map& variables,
 | 
						|
    boost::program_options::options_description& description)
 | 
						|
{
 | 
						|
    boost::program_options::store(
 | 
						|
        boost::program_options::command_line_parser(argc, argv).options(description).allow_unregistered().run(),
 | 
						|
        variables
 | 
						|
    );
 | 
						|
}
 | 
						|
 | 
						|
void parseConfig(std::istream& stream, boost::program_options::variables_map& variables,
 | 
						|
    boost::program_options::options_description& description)
 | 
						|
{
 | 
						|
    boost::program_options::store(
 | 
						|
        Files::parse_config_file(stream, description, true),
 | 
						|
        variables
 | 
						|
    );
 | 
						|
}
 | 
						|
 | 
						|
std::istream& operator>> (std::istream& istream, MaybeQuotedPath& MaybeQuotedPath)
 | 
						|
{
 | 
						|
    // If the stream starts with a double quote, read from stream using boost::filesystem::path rules, then discard anything remaining.
 | 
						|
    // This prevents boost::program_options getting upset that we've not consumed the whole stream.
 | 
						|
    // If it doesn't start with a double quote, read the whole thing verbatim
 | 
						|
    if (istream.peek() == '"')
 | 
						|
    {
 | 
						|
        istream >> static_cast<boost::filesystem::path&>(MaybeQuotedPath);
 | 
						|
        if (istream && !istream.eof() && istream.peek() != EOF)
 | 
						|
        {
 | 
						|
            std::string remainder{std::istreambuf_iterator(istream), {}};
 | 
						|
            Log(Debug::Warning) << "Trailing data in path setting. Used '" << MaybeQuotedPath.string() << "' but '" << remainder << "' remained";
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        std::string intermediate{std::istreambuf_iterator(istream), {}};
 | 
						|
        static_cast<boost::filesystem::path&>(MaybeQuotedPath) = intermediate;
 | 
						|
    }
 | 
						|
    return istream;
 | 
						|
}
 | 
						|
 | 
						|
PathContainer asPathContainer(const MaybeQuotedPathContainer& MaybeQuotedPathContainer)
 | 
						|
{
 | 
						|
    return PathContainer(MaybeQuotedPathContainer.begin(), MaybeQuotedPathContainer.end());
 | 
						|
}
 | 
						|
 | 
						|
} /* namespace Files */
 |