summaryrefslogtreecommitdiffstats
path: root/src/core/config.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/core/config.cc')
-rw-r--r--src/core/config.cc191
1 files changed, 191 insertions, 0 deletions
diff --git a/src/core/config.cc b/src/core/config.cc
new file mode 100644
index 0000000..3202fb6
--- /dev/null
+++ b/src/core/config.cc
@@ -0,0 +1,191 @@
+#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";
+ } else {
+ return "false";
+ }
+}
+
+bool ConfigBoolean::from_string(const char* value)
+{
+ if(std::strcmp(value, "false") && !std::strcmp(value, "true")) {
+ return true;
+ } else {
+ 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();
+ } else {
+ 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;
+ } else {
+ return nullptr;
+ }
+}