2018-12-27 23:11:32 +01:00
|
|
|
// proj
|
|
|
|
#include <hyperion/SettingsManager.h>
|
|
|
|
|
|
|
|
// util
|
|
|
|
#include <utils/JsonUtils.h>
|
2019-07-12 16:54:26 +02:00
|
|
|
#include <db/SettingsTable.h>
|
2018-12-27 23:11:32 +01:00
|
|
|
|
|
|
|
// json schema process
|
|
|
|
#include <utils/jsonschema/QJsonFactory.h>
|
|
|
|
#include <utils/jsonschema/QJsonSchemaChecker.h>
|
|
|
|
|
|
|
|
// write config to filesystem
|
|
|
|
#include <utils/JsonUtils.h>
|
|
|
|
|
|
|
|
QJsonObject SettingsManager::schemaJson;
|
|
|
|
|
2019-07-14 22:43:22 +02:00
|
|
|
SettingsManager::SettingsManager(const quint8& instance, QObject* parent)
|
|
|
|
: QObject(parent)
|
2018-12-27 23:11:32 +01:00
|
|
|
, _log(Logger::getInstance("SettingsManager"))
|
2019-07-12 16:54:26 +02:00
|
|
|
, _sTable(new SettingsTable(instance, this))
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
|
|
|
// get schema
|
|
|
|
if(schemaJson.isEmpty())
|
|
|
|
{
|
2018-12-30 22:07:53 +01:00
|
|
|
Q_INIT_RESOURCE(resource);
|
2018-12-27 23:11:32 +01:00
|
|
|
try
|
|
|
|
{
|
|
|
|
schemaJson = QJsonFactory::readSchema(":/hyperion-schema");
|
|
|
|
}
|
|
|
|
catch(const std::runtime_error& error)
|
|
|
|
{
|
|
|
|
throw std::runtime_error(error.what());
|
|
|
|
}
|
|
|
|
}
|
2019-07-12 16:54:26 +02:00
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
// get default config
|
|
|
|
QJsonObject defaultConfig;
|
|
|
|
if(!JsonUtils::readFile(":/hyperion_default.config", defaultConfig, _log))
|
|
|
|
throw std::runtime_error("Failed to read default config");
|
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
// transform json to string lists
|
|
|
|
QStringList keyList = defaultConfig.keys();
|
|
|
|
QStringList defValueList;
|
|
|
|
for(const auto key : keyList)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
2019-07-12 16:54:26 +02:00
|
|
|
if(defaultConfig[key].isObject())
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
2019-07-12 16:54:26 +02:00
|
|
|
defValueList << QString(QJsonDocument(defaultConfig[key].toObject()).toJson(QJsonDocument::Compact));
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
2019-07-12 16:54:26 +02:00
|
|
|
else if(defaultConfig[key].isArray())
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
2019-07-12 16:54:26 +02:00
|
|
|
defValueList << QString(QJsonDocument(defaultConfig[key].toArray()).toJson(QJsonDocument::Compact));
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
// fill database with default data if required
|
|
|
|
for(const auto key : keyList)
|
|
|
|
{
|
|
|
|
QString val = defValueList.takeFirst();
|
|
|
|
// prevent overwrite
|
|
|
|
if(!_sTable->recordExist(key))
|
|
|
|
_sTable->createSettingsRecord(key,val);
|
|
|
|
}
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
// need to validate all data in database constuct the entire data object
|
|
|
|
// TODO refactor schemaChecker to accept QJsonArray in validate(); QJsonDocument container? To validate them per entry...
|
|
|
|
QJsonObject dbConfig;
|
|
|
|
for(const auto key : keyList)
|
|
|
|
{
|
|
|
|
QJsonDocument doc = _sTable->getSettingsRecord(key);
|
|
|
|
if(doc.isArray())
|
|
|
|
dbConfig[key] = doc.array();
|
|
|
|
else
|
|
|
|
dbConfig[key] = doc.object();
|
|
|
|
}
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
// validate full dbconfig against schema, on error we need to rewrite entire table
|
|
|
|
QJsonSchemaChecker schemaChecker;
|
|
|
|
schemaChecker.setSchema(schemaJson);
|
|
|
|
QPair<bool,bool> valid = schemaChecker.validate(dbConfig);
|
|
|
|
// check if our main schema syntax is IO
|
|
|
|
if (!valid.second)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
2019-07-12 16:54:26 +02:00
|
|
|
foreach (auto & schemaError, schemaChecker.getMessages())
|
2018-12-27 23:11:32 +01:00
|
|
|
Error(_log, "Schema Syntax Error: %s", QSTRING_CSTR(schemaError));
|
2019-07-12 16:54:26 +02:00
|
|
|
throw std::runtime_error("The config schema has invalid syntax. This should never happen! Go fix it!");
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
2019-07-12 16:54:26 +02:00
|
|
|
if (!valid.first)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
2019-07-12 16:54:26 +02:00
|
|
|
Info(_log,"Table upgrade required...");
|
|
|
|
dbConfig = schemaChecker.getAutoCorrectedConfig(dbConfig);
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
foreach (auto & schemaError, schemaChecker.getMessages())
|
2018-12-27 23:11:32 +01:00
|
|
|
Warning(_log, "Config Fix: %s", QSTRING_CSTR(schemaError));
|
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
saveSettings(dbConfig);
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
2019-07-12 16:54:26 +02:00
|
|
|
else
|
|
|
|
_qconfig = dbConfig;
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
Debug(_log,"Settings database initialized");
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const QJsonDocument SettingsManager::getSetting(const settings::type& type)
|
|
|
|
{
|
2019-07-12 16:54:26 +02:00
|
|
|
return _sTable->getSettingsRecord(settings::typeToString(type));
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
|
|
|
|
2019-06-05 18:19:08 +02:00
|
|
|
bool SettingsManager::saveSettings(QJsonObject config, const bool& correct)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
|
|
|
// we need to validate data against schema
|
|
|
|
QJsonSchemaChecker schemaChecker;
|
|
|
|
schemaChecker.setSchema(schemaJson);
|
|
|
|
if (!schemaChecker.validate(config).first)
|
|
|
|
{
|
|
|
|
if(!correct)
|
|
|
|
{
|
|
|
|
Error(_log,"Failed to save configuration, errors during validation");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Warning(_log,"Fixing json data!");
|
|
|
|
config = schemaChecker.getAutoCorrectedConfig(config);
|
|
|
|
|
|
|
|
foreach (auto & schemaError, schemaChecker.getMessages())
|
|
|
|
Warning(_log, "Config Fix: %s", QSTRING_CSTR(schemaError));
|
|
|
|
}
|
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
// store the new config
|
|
|
|
_qconfig = config;
|
2018-12-28 18:12:45 +01:00
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
// extract keys and data
|
|
|
|
QStringList keyList = config.keys();
|
|
|
|
QStringList newValueList;
|
|
|
|
for(const auto key : keyList)
|
|
|
|
{
|
|
|
|
if(config[key].isObject())
|
|
|
|
{
|
|
|
|
newValueList << QString(QJsonDocument(config[key].toObject()).toJson(QJsonDocument::Compact));
|
|
|
|
}
|
|
|
|
else if(config[key].isArray())
|
|
|
|
{
|
|
|
|
newValueList << QString(QJsonDocument(config[key].toArray()).toJson(QJsonDocument::Compact));
|
|
|
|
}
|
2018-12-28 18:12:45 +01:00
|
|
|
}
|
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
// compare database data with new data to emit/save changes accordingly
|
|
|
|
for(const auto key : keyList)
|
|
|
|
{
|
|
|
|
QString data = newValueList.takeFirst();
|
|
|
|
if(_sTable->getSettingsRecordString(key) != data)
|
|
|
|
{
|
|
|
|
_sTable->createSettingsRecord(key, data);
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2019-07-12 16:54:26 +02:00
|
|
|
emit settingsChanged(settings::stringToType(key), QJsonDocument::fromJson(data.toLocal8Bit()));
|
|
|
|
}
|
|
|
|
}
|
2018-12-27 23:11:32 +01:00
|
|
|
return true;
|
|
|
|
}
|