diff options
Diffstat (limited to 'core/config.cc')
| -rw-r--r-- | core/config.cc | 179 |
1 files changed, 179 insertions, 0 deletions
diff --git a/core/config.cc b/core/config.cc new file mode 100644 index 0000000..94a8c1b --- /dev/null +++ b/core/config.cc @@ -0,0 +1,179 @@ +#include "core/pch.hh" +#include "core/config.hh" + +#include "core/cmdline.hh" +#include "core/strtools.hh" +#include "core/version.hh" + +ConfigBoolean::ConfigBoolean(bool default_value) +{ + m_value = default_value; + m_string = ConfigBoolean::to_string(default_value); +} + +void ConfigBoolean::set(const char *value) +{ + m_value = ConfigBoolean::from_string(value); + m_string = ConfigBoolean::to_string(m_value); +} + +const char *ConfigBoolean::get(void) const +{ + return m_string.c_str(); +} + +bool ConfigBoolean::get_value(void) const +{ + return m_value; +} + +void ConfigBoolean::set_value(bool value) +{ + m_value = value; + m_string = ConfigBoolean::to_string(m_value); +} + +const char *ConfigBoolean::to_string(bool value) +{ + if(value) + return "true"; + return "false"; +} + +bool ConfigBoolean::from_string(const char *value) +{ + if(std::strcmp(value, "false") && !std::strcmp(value, "true")) + return true; + return false; +} + +ConfigString::ConfigString(const char *default_value) +{ + m_value = default_value; +} + +void ConfigString::set(const char *value) +{ + m_value = value; +} + +const char *ConfigString::get(void) const +{ + return m_value.c_str(); +} + +void Config::load_cmdline(void) +{ + for(auto it : m_values) { + if(auto value = cmdline::get(it.first.c_str())) { + it.second->set(value); + } + } +} + +bool Config::load_file(const char *path) +{ + if(auto file = PHYSFS_openRead(path)) { + auto source = std::string(PHYSFS_fileLength(file), char(0x00)); + PHYSFS_readBytes(file, source.data(), source.size()); + PHYSFS_close(file); + + std::string line; + std::string kv_string; + std::istringstream stream(source); + + while(std::getline(stream, line)) { + auto comment = line.find_first_of('#'); + + if(comment == std::string::npos) + kv_string = strtools::trim_whitespace(line); + else kv_string = strtools::trim_whitespace(line.substr(0, comment)); + + if(strtools::is_whitespace(kv_string)) { + // Ignore empty or commented out lines + continue; + } + + auto separator = kv_string.find('='); + + if(separator == std::string::npos) { + spdlog::warn("config: {}: invalid line: {}", path, line); + continue; + } + + auto kv_name = strtools::trim_whitespace(kv_string.substr(0, separator)); + auto kv_value = strtools::trim_whitespace(kv_string.substr(separator + 1)); + + auto kv_pair = m_values.find(kv_name); + + if(kv_pair == m_values.cend()) { + spdlog::warn("config: {}: unknown key: {}", path, kv_name); + continue; + } + + kv_pair->second->set(kv_value.c_str()); + } + + return true; + } + + return false; +} + +bool Config::save_file(const char *path) const +{ + std::ostringstream stream; + + auto curtime = std::time(nullptr); + + stream << "# Voxelius " << PROJECT_VERSION_STRING << " configuration file" << std::endl; + stream << "# Generated at: " << std::put_time(std::gmtime(&curtime), "%Y-%m-%d %H:%M:%S %z") << std::endl << std::endl; + + for(const auto &it : m_values) { + stream << it.first << "="; + stream << it.second->get(); + stream << std::endl; + } + + if(auto file = PHYSFS_openWrite(path)) { + auto source = stream.str(); + PHYSFS_writeBytes(file, source.data(), source.size()); + PHYSFS_close(file); + return true; + } + + return false; +} + +bool Config::set_value(const char *name, const char *value) +{ + auto kv_pair = m_values.find(name); + + if(kv_pair != m_values.cend()) { + kv_pair->second->set(value); + return true; + } + + return false; +} + +const char *Config::get_value(const char *name) const +{ + auto kv_pair = m_values.find(name); + if(kv_pair != m_values.cend()) + return kv_pair->second->get(); + return nullptr; +} + +void Config::add_value(const char *name, IConfigValue &vref) +{ + m_values.insert_or_assign(name, &vref); +} + +const IConfigValue *Config::find(const char *name) const +{ + auto kv_pair = m_values.find(name); + if(kv_pair != m_values.cend()) + return kv_pair->second; + return nullptr; +} |
