refs #665 Cache will only load and save the values that it really needs to.

This commit is contained in:
Mathew Sutcliffe
2016-06-04 00:33:22 +01:00
parent 50de252ff6
commit 36cb07cb1f
3 changed files with 50 additions and 14 deletions

View File

@@ -312,10 +312,8 @@ namespace BlackMisc
{ {
return CStatusMessage(this).error("Invalid JSON format in %1") << file.fileName(); return CStatusMessage(this).error("Invalid JSON format in %1") << file.fileName();
} }
CVariantMap storedValues; auto object = json.object();
storedValues.convertFromJson(json.object()); json.setObject(it->mergeToJson(object));
storedValues.insert(*it);
json.setObject(storedValues.toJson());
if (!(file.seek(0) && file.resize(0) && file.write(json.toJson()) > 0 && file.checkedClose())) if (!(file.seek(0) && file.resize(0) && file.write(json.toJson()) > 0 && file.checkedClose()))
{ {
@@ -343,10 +341,25 @@ namespace BlackMisc
return CStatusMessage(this).error("Failed to create directory %1") << dir; return CStatusMessage(this).error("Failed to create directory %1") << dir;
} }
const QStringList entries(QDir(dir).entryList({ "*.json" }, QDir::Files)); QMap<QString, QStringList> keysInFiles;
for (const auto &filename : entries) for (const auto &key : keys)
{ {
QFile file(dir + "/" + filename); keysInFiles[key.section('/', 0, 0)].push_back(key);
}
if (keys.isEmpty())
{
for (const auto &filename : QDir(dir).entryInfoList({ "*.json" }, QDir::Files))
{
keysInFiles.insert(filename.completeBaseName(), {});
}
}
for (auto it = keysInFiles.cbegin(); it != keysInFiles.cend(); ++it)
{
QFile file(dir + "/" + it.key() + ".json");
if (! file.exists())
{
continue;
}
if (! file.open(QFile::ReadOnly | QFile::Text)) if (! file.open(QFile::ReadOnly | QFile::Text))
{ {
return CStatusMessage(this).error("Failed to open %1: %2") << file.fileName() << file.errorString(); return CStatusMessage(this).error("Failed to open %1: %2") << file.fileName() << file.errorString();
@@ -357,11 +370,8 @@ namespace BlackMisc
return CStatusMessage(this).error("Invalid JSON format in %1") << file.fileName(); return CStatusMessage(this).error("Invalid JSON format in %1") << file.fileName();
} }
CVariantMap temp; CVariantMap temp;
temp.convertFromJson(json.object()); temp.convertFromJson(json.object(), it.value());
if (! keys.isEmpty()) if (it.value().isEmpty()) { temp.convertFromJson(json.object()); }
{
temp.removeByKeyIf([&keys](const QString & key) { return ! keys.contains(key); }); // TODO optimize by skipping files
}
temp.removeDuplicates(currentValues); temp.removeDuplicates(currentValues);
o_values.insert(temp, QFileInfo(file).lastModified().toMSecsSinceEpoch()); o_values.insert(temp, QFileInfo(file).lastModified().toMSecsSinceEpoch());
} }

View File

@@ -14,9 +14,8 @@
namespace BlackMisc namespace BlackMisc
{ {
QJsonObject CVariantMap::toJson() const QJsonObject &CVariantMap::mergeToJson(QJsonObject &json) const
{ {
QJsonObject json;
for (auto it = cbegin(); it != cend(); ++it) for (auto it = cbegin(); it != cend(); ++it)
{ {
json.insert(it.key(), it.value().toJson()); json.insert(it.key(), it.value().toJson());
@@ -24,6 +23,13 @@ namespace BlackMisc
return json; return json;
} }
QJsonObject CVariantMap::toJson() const
{
QJsonObject json;
mergeToJson(json);
return json;
}
void CVariantMap::convertFromJson(const QJsonObject &json) void CVariantMap::convertFromJson(const QJsonObject &json)
{ {
clear(); clear();
@@ -35,4 +41,17 @@ namespace BlackMisc
} }
} }
void CVariantMap::convertFromJson(const QJsonObject &json, const QStringList &keys)
{
clear();
for (const auto &key : keys)
{
auto value = json.value(key);
if (value.isUndefined()) { continue; }
CVariant var;
var.convertFromJson(value.toObject());
insert(key, var);
}
}
} }

View File

@@ -56,11 +56,18 @@ namespace BlackMisc
//! Move assignment operator. //! Move assignment operator.
CVariantMap &operator =(CVariantMap &&other) noexcept { CDictionary::operator =(std::move(other)); return *this; } CVariantMap &operator =(CVariantMap &&other) noexcept { CDictionary::operator =(std::move(other)); return *this; }
//! Insert values from this map into an existing JSON object.
QJsonObject &mergeToJson(QJsonObject &json) const;
//! \copydoc BlackMisc::CValueObject::toJson //! \copydoc BlackMisc::CValueObject::toJson
QJsonObject toJson() const; QJsonObject toJson() const;
//! \copydoc BlackMisc::CValueObject::convertFromJson //! \copydoc BlackMisc::CValueObject::convertFromJson
void convertFromJson(const QJsonObject &json); void convertFromJson(const QJsonObject &json);
//! \copydoc BlackMisc::CValueObject::convertFromJson
//! Convert only keys present in list argument.
void convertFromJson(const QJsonObject &json, const QStringList &keys);
}; };
} }