mirror of
https://github.com/swift-project/pilotclient.git
synced 2026-04-06 10:15:38 +08:00
refs #536 Moved CDataCache serialization into a worker thread.
This commit is contained in:
@@ -26,12 +26,14 @@ namespace BlackCore
|
||||
CLogMessage(this).error("Failed to create directory %1") << persistentStore();
|
||||
}
|
||||
|
||||
connect(this, &CValueCache::valuesChangedByLocal, this, [this](const CValueCachePacket &values) { saveToStore(values.toVariantMap()); });
|
||||
connect(&m_watcher, &QFileSystemWatcher::fileChanged, this, [this] { loadFromStore(); });
|
||||
connect(this, &CValueCache::valuesChangedByLocal, this, &CDataCache::saveToStoreAsync);
|
||||
connect(&m_watcher, &QFileSystemWatcher::fileChanged, this, &CDataCache::loadFromStoreAsync);
|
||||
connect(&m_serializer, &CDataCacheSerializer::valuesLoadedFromStore, this, &CDataCache::changeValuesFromRemote);
|
||||
|
||||
if (! QFile::exists(m_revisionFileName)) { QFile(m_revisionFileName).open(QFile::WriteOnly); }
|
||||
m_watcher.addPath(m_revisionFileName);
|
||||
loadFromStore();
|
||||
m_serializer.start();
|
||||
loadFromStoreAsync();
|
||||
}
|
||||
|
||||
CDataCache *CDataCache::instance()
|
||||
@@ -74,10 +76,37 @@ namespace BlackCore
|
||||
}
|
||||
}
|
||||
|
||||
void CDataCache::saveToStore(const BlackMisc::CVariantMap &values)
|
||||
void CDataCache::saveToStoreAsync(const BlackMisc::CValueCachePacket &values)
|
||||
{
|
||||
QMutexLocker lock(&m_mutex);
|
||||
auto baseline = getAllValues();
|
||||
QTimer::singleShot(0, &m_serializer, [this, baseline, values]
|
||||
{
|
||||
m_serializer.saveToStore(values.toVariantMap(), baseline);
|
||||
});
|
||||
}
|
||||
|
||||
void CDataCache::loadFromStoreAsync()
|
||||
{
|
||||
auto baseline = getAllValues();
|
||||
QTimer::singleShot(0, &m_serializer, [this, baseline]
|
||||
{
|
||||
m_serializer.loadFromStore(baseline);
|
||||
});
|
||||
}
|
||||
|
||||
CDataCacheSerializer::CDataCacheSerializer(CDataCache *owner, const QString &revisionFileName) :
|
||||
CContinuousWorker(owner),
|
||||
m_cache(owner),
|
||||
m_revisionFileName(revisionFileName)
|
||||
{}
|
||||
|
||||
const QString &CDataCacheSerializer::persistentStore() const
|
||||
{
|
||||
return m_cache->persistentStore();
|
||||
}
|
||||
|
||||
void CDataCacheSerializer::saveToStore(const BlackMisc::CVariantMap &values, const BlackMisc::CVariantMap &baseline)
|
||||
{
|
||||
QLockFile revisionFileLock(m_revisionFileName + ".lock");
|
||||
if (! revisionFileLock.lock())
|
||||
{
|
||||
@@ -85,7 +114,7 @@ namespace BlackCore
|
||||
return;
|
||||
}
|
||||
|
||||
loadFromStore(false, true); // last-minute check for remote changes before clobbering the revision file
|
||||
loadFromStore(baseline, false, true); // last-minute check for remote changes before clobbering the revision file
|
||||
for (const auto &key : values.keys()) { m_deferredChanges.remove(key); } // ignore changes that we are about to overwrite
|
||||
|
||||
QFile revisionFile(m_revisionFileName);
|
||||
@@ -97,13 +126,17 @@ namespace BlackCore
|
||||
m_revision = CIdentifier().toUuid();
|
||||
revisionFile.write(m_revision.toByteArray());
|
||||
|
||||
saveToFiles(persistentStore(), values);
|
||||
m_cache->saveToFiles(persistentStore(), values);
|
||||
|
||||
if (! m_deferredChanges.isEmpty()) // apply changes which we grabbed at the last minute above
|
||||
{
|
||||
emit valuesLoadedFromStore(m_deferredChanges, CIdentifier::anonymous());
|
||||
m_deferredChanges.clear();
|
||||
}
|
||||
}
|
||||
|
||||
void CDataCache::loadFromStore(bool revLock, bool defer)
|
||||
void CDataCacheSerializer::loadFromStore(const BlackMisc::CVariantMap &baseline, bool revLock, bool defer)
|
||||
{
|
||||
QMutexLocker lock(&m_mutex);
|
||||
|
||||
QLockFile revisionFileLock(m_revisionFileName + ".lock");
|
||||
if (revLock && ! revisionFileLock.lock())
|
||||
{
|
||||
@@ -127,13 +160,13 @@ namespace BlackCore
|
||||
{
|
||||
m_revision = newRevision;
|
||||
CValueCachePacket newValues;
|
||||
loadFromFiles(persistentStore(), newValues);
|
||||
m_cache->loadFromFiles(persistentStore(), baseline, newValues);
|
||||
m_deferredChanges.insert(newValues);
|
||||
}
|
||||
|
||||
if (! (m_deferredChanges.isEmpty() || defer))
|
||||
{
|
||||
changeValuesFromRemote(m_deferredChanges, CIdentifier::anonymous());
|
||||
emit valuesLoadedFromStore(m_deferredChanges, CIdentifier::anonymous());
|
||||
m_deferredChanges.clear();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -14,12 +14,50 @@
|
||||
|
||||
#include "blackcore/blackcoreexport.h"
|
||||
#include "blackmisc/valuecache.h"
|
||||
#include "blackmisc/worker.h"
|
||||
#include <QUuid>
|
||||
#include <QFileSystemWatcher>
|
||||
|
||||
namespace BlackCore
|
||||
{
|
||||
|
||||
class CDataCache;
|
||||
|
||||
/*!
|
||||
* Worker which performs (de)serialization on behalf of CDataCache, in a separate thread
|
||||
* so that the main thread is not blocked by (de)serialization of large objects.
|
||||
*/
|
||||
class BLACKCORE_EXPORT CDataCacheSerializer : public BlackMisc::CContinuousWorker
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
//! Constructor.
|
||||
CDataCacheSerializer(CDataCache *owner, const QString &revisionFileName);
|
||||
|
||||
//! Save values to persistent store. Called whenever a value is changed locally.
|
||||
void saveToStore(const BlackMisc::CVariantMap &values, const BlackMisc::CVariantMap &baseline);
|
||||
|
||||
//! Load values from persistent store. Called once per second.
|
||||
//! Also called by saveToStore, to ensure that remote changes to unrelated values are not lost.
|
||||
//! \param baseline A snapshot of the currently loaded values, taken when the load is queued.
|
||||
//! \param lock Whether to acquire the revision file lock. Used when called by saveToStore.
|
||||
//! \param defer Whether to defer applying the changes. Used when called by saveToStore.
|
||||
void loadFromStore(const BlackMisc::CVariantMap &baseline, bool lock = true, bool defer = false);
|
||||
|
||||
signals:
|
||||
//! Signal back to the cache when values have been loaded.
|
||||
void valuesLoadedFromStore(const BlackMisc::CValueCachePacket &values, const BlackMisc::CIdentifier &originator);
|
||||
|
||||
private:
|
||||
const QString &persistentStore() const;
|
||||
|
||||
const CDataCache *const m_cache = nullptr;
|
||||
QUuid m_revision;
|
||||
const QString m_revisionFileName;
|
||||
BlackMisc::CValueCachePacket m_deferredChanges;
|
||||
};
|
||||
|
||||
/*!
|
||||
* Singleton derived class of CValueCache, for core dynamic data.
|
||||
*
|
||||
@@ -45,19 +83,14 @@ namespace BlackCore
|
||||
private:
|
||||
CDataCache();
|
||||
|
||||
//! Save values to persistent store. Called whenever a value is changed locally.
|
||||
void saveToStore(const BlackMisc::CVariantMap &values);
|
||||
|
||||
//! Load values from persistent store. Called once per second.
|
||||
//! Also called by saveToStore, to ensure that remote changes to unrelated values are not lost.
|
||||
//! \param lock Whether to acquire the revision file lock. Used when called by saveToStore.
|
||||
//! \param defer Whether to defer applying the changes. Used when called by saveToStore.
|
||||
void loadFromStore(bool lock = true, bool defer = false);
|
||||
void saveToStoreAsync(const BlackMisc::CValueCachePacket &values);
|
||||
void loadFromStoreAsync();
|
||||
|
||||
QFileSystemWatcher m_watcher;
|
||||
QUuid m_revision;
|
||||
const QString m_revisionFileName { persistentStore() + "/.rev" };
|
||||
BlackMisc::CValueCachePacket m_deferredChanges;
|
||||
|
||||
CDataCacheSerializer m_serializer { this, m_revisionFileName };
|
||||
friend class CDataCacheSerializer; // to access protected members of CValueCache
|
||||
};
|
||||
|
||||
/*!
|
||||
|
||||
Reference in New Issue
Block a user