From 3e65e063253298f04e448e1182959f6c40973d38 Mon Sep 17 00:00:00 2001 From: Jonathan White Date: Sun, 10 May 2020 10:14:33 -0400 Subject: [PATCH] Fix various issues with KeeShare * Fix #3790, shares now use the standard FileWatcher class to detect remote file changes using checksums and file system triggers. * Fix #3985, macOS file selection no longer hangs the app. * Restore saving of KeeShare settings accidentally removed by 596d2cf --- src/core/Database.cpp | 2 +- src/core/FileWatcher.cpp | 191 +-------------------------------- src/core/FileWatcher.h | 53 +-------- src/keeshare/KeeShare.cpp | 4 +- src/keeshare/ShareObserver.cpp | 103 ++++++++---------- src/keeshare/ShareObserver.h | 9 +- 6 files changed, 57 insertions(+), 305 deletions(-) diff --git a/src/core/Database.cpp b/src/core/Database.cpp index 66af624ecb..a59bf03e9a 100644 --- a/src/core/Database.cpp +++ b/src/core/Database.cpp @@ -58,7 +58,7 @@ Database::Database() connect(&m_modifiedTimer, SIGNAL(timeout()), SIGNAL(databaseModified())); connect(this, SIGNAL(databaseOpened()), SLOT(updateCommonUsernames())); connect(this, SIGNAL(databaseSaved()), SLOT(updateCommonUsernames())); - connect(m_fileWatcher, SIGNAL(fileChanged()), SIGNAL(databaseFileChanged())); + connect(m_fileWatcher, &FileWatcher::fileChanged, this, &Database::databaseFileChanged); m_modified = false; m_emitModified = true; diff --git a/src/core/FileWatcher.cpp b/src/core/FileWatcher.cpp index eab27c214f..af66d3d4ca 100644 --- a/src/core/FileWatcher.cpp +++ b/src/core/FileWatcher.cpp @@ -1,6 +1,5 @@ /* - * Copyright (C) 2011 Felix Geyer - * Copyright (C) 2017 KeePassXC Team + * Copyright (C) 2020 KeePassXC Team * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -38,7 +37,7 @@ FileWatcher::FileWatcher(QObject* parent) { connect(&m_fileWatcher, SIGNAL(fileChanged(QString)), SLOT(checkFileChanged())); connect(&m_fileChecksumTimer, SIGNAL(timeout()), SLOT(checkFileChanged())); - connect(&m_fileChangeDelayTimer, SIGNAL(timeout()), SIGNAL(fileChanged())); + connect(&m_fileChangeDelayTimer, &QTimer::timeout, this, [this] { emit fileChanged(m_filePath); }); m_fileChangeDelayTimer.setSingleShot(true); m_fileIgnoreDelayTimer.setSingleShot(true); } @@ -151,189 +150,3 @@ QByteArray FileWatcher::calculateChecksum() // prevents unnecessary merge requests on intermittent network shares return m_fileChecksum; } - -BulkFileWatcher::BulkFileWatcher(QObject* parent) - : QObject(parent) -{ - connect(&m_fileWatcher, SIGNAL(fileChanged(QString)), SLOT(handleFileChanged(QString))); - connect(&m_fileWatcher, SIGNAL(directoryChanged(QString)), SLOT(handleDirectoryChanged(QString))); - connect(&m_watchedFilesIgnoreTimer, SIGNAL(timeout()), this, SLOT(observeFileChanges())); - connect(&m_pendingSignalsTimer, SIGNAL(timeout()), this, SLOT(emitSignals())); - m_watchedFilesIgnoreTimer.setSingleShot(true); - m_pendingSignalsTimer.setSingleShot(true); -} - -void BulkFileWatcher::clear() -{ - for (const QString& path : m_fileWatcher.files() + m_fileWatcher.directories()) { - const QFileInfo info(path); - m_fileWatcher.removePath(info.absoluteFilePath()); - m_fileWatcher.removePath(info.absolutePath()); - } - m_watchedPaths.clear(); - m_watchedFilesInDirectory.clear(); - m_watchedFilesIgnored.clear(); -} - -void BulkFileWatcher::removePath(const QString& path) -{ - const QFileInfo info(path); - const QString filePath = info.absoluteFilePath(); - const QString directoryPath = info.absolutePath(); - m_watchedFilesInDirectory[directoryPath].remove(filePath); - m_fileWatcher.removePath(filePath); - m_watchedPaths.remove(filePath); - if (m_watchedFilesInDirectory[directoryPath].isEmpty()) { - m_fileWatcher.removePath(directoryPath); - m_watchedPaths.remove(directoryPath); - m_watchedFilesInDirectory.remove(directoryPath); - } -} - -void BulkFileWatcher::addPath(const QString& path) -{ - const QFileInfo info(path); - const QString filePath = info.absoluteFilePath(); - const QString directoryPath = info.absolutePath(); - if (!m_watchedPaths.value(filePath)) { - const bool fileSuccess = m_fileWatcher.addPath(filePath); - m_watchedPaths[filePath] = fileSuccess; - } - if (!m_watchedPaths.value(directoryPath)) { - const bool directorySuccess = m_fileWatcher.addPath(directoryPath); - m_watchedPaths[directoryPath] = directorySuccess; - } - m_watchedFilesInDirectory[directoryPath][filePath] = info.exists() ? info.lastModified().toMSecsSinceEpoch() : 0; -} - -void BulkFileWatcher::handleFileChanged(const QString& path) -{ - const QFileInfo info(path); - const QString filePath = info.absoluteFilePath(); - const QString directoryPath = info.absolutePath(); - const QMap& watchedFiles = m_watchedFilesInDirectory[directoryPath]; - const qint64 lastModificationTime = info.lastModified().toMSecsSinceEpoch(); - const bool created = watchedFiles[filePath] == 0 && info.exists(); - const bool deleted = watchedFiles[filePath] != 0 && !info.exists(); - const bool changed = !created && !deleted && lastModificationTime != watchedFiles[filePath]; - - addPath(path); - - if (m_watchedFilesIgnored[info.canonicalFilePath()] > Clock::currentDateTimeUtc()) { - // changes are blocked - return; - } - if (created) { - qDebug("File created %s", qPrintable(path)); - scheduleSignal(Created, filePath); - } - if (changed) { - qDebug("File changed %s", qPrintable(path)); - scheduleSignal(Updated, filePath); - } - if (deleted) { - qDebug("File removed %s", qPrintable(path)); - scheduleSignal(Removed, filePath); - } -} - -void BulkFileWatcher::handleDirectoryChanged(const QString& path) -{ - qDebug("Directory changed %s", qPrintable(path)); - const QFileInfo directoryInfo(path); - const QString directoryPath = directoryInfo.absoluteFilePath(); - QMap& watchedFiles = m_watchedFilesInDirectory[directoryPath]; - for (const QString& filename : watchedFiles.keys()) { - const QFileInfo fileInfo(filename); - const QString filePath = fileInfo.absoluteFilePath(); - const qint64 previousModificationTime = watchedFiles[filePath]; - const qint64 lastModificationTime = fileInfo.lastModified().toMSecsSinceEpoch(); - if (!fileInfo.exists() && previousModificationTime != 0) { - qDebug("Remove watch file %s", qPrintable(fileInfo.absoluteFilePath())); - m_fileWatcher.removePath(filePath); - m_watchedPaths.remove(filePath); - watchedFiles.remove(filePath); - scheduleSignal(Removed, filePath); - } - if (previousModificationTime == 0 && fileInfo.exists()) { - qDebug("Add watch file %s", qPrintable(fileInfo.absoluteFilePath())); - if (!m_watchedPaths.value(filePath)) { - const bool success = m_fileWatcher.addPath(filePath); - m_watchedPaths[filePath] = success; - watchedFiles[filePath] = lastModificationTime; - } - scheduleSignal(Created, filePath); - } - if (fileInfo.exists() && previousModificationTime != lastModificationTime) { - // this case is handled using - qDebug("Refresh watch file %s", qPrintable(fileInfo.absoluteFilePath())); - m_fileWatcher.removePath(fileInfo.absolutePath()); - m_fileWatcher.addPath(fileInfo.absolutePath()); - scheduleSignal(Updated, filePath); - } - m_watchedFilesInDirectory[directoryPath][filePath] = fileInfo.exists() ? lastModificationTime : 0; - } -} - -void BulkFileWatcher::emitSignals() -{ - QMap> queued; - m_pendingSignals.swap(queued); - for (const auto& path : queued.keys()) { - const auto& signal = queued[path]; - if (signal.last() == Removed) { - qDebug("Emit %s removed", qPrintable(path)); - emit fileRemoved(path); - continue; - } - if (signal.first() == Created) { - qDebug("Emit %s created", qPrintable(path)); - emit fileCreated(path); - continue; - } - qDebug("Emit %s changed", qPrintable(path)); - emit fileChanged(path); - } -} - -void BulkFileWatcher::scheduleSignal(Signal signal, const QString& path) -{ - // we need to collect signals since the file watcher API may send multiple signals for a "single" change - // therefore we wait until the event loop finished before starting to import any changes - const QString filePath = QFileInfo(path).absoluteFilePath(); - m_pendingSignals[filePath] << signal; - - if (!m_pendingSignalsTimer.isActive()) { - m_pendingSignalsTimer.start(); - } -} - -void BulkFileWatcher::ignoreFileChanges(const QString& path) -{ - const QFileInfo info(path); - m_watchedFilesIgnored[info.canonicalFilePath()] = Clock::currentDateTimeUtc().addMSecs(FileChangeDelay); -} - -void BulkFileWatcher::observeFileChanges(bool delayed) -{ - int timeout = 0; - if (delayed) { - timeout = FileChangeDelay; - } else { - const QDateTime current = Clock::currentDateTimeUtc(); - for (const QString& key : m_watchedFilesIgnored.keys()) { - if (m_watchedFilesIgnored[key] < current) { - // We assume that there was no concurrent change of the database - // during our block - so no need to reimport - qDebug("Remove block from %s", qPrintable(key)); - m_watchedFilesIgnored.remove(key); - continue; - } - qDebug("Keep block from %s", qPrintable(key)); - timeout = qMin(timeout, static_cast(current.msecsTo(m_watchedFilesIgnored[key]))); - } - } - if (timeout > 0 && !m_watchedFilesIgnoreTimer.isActive()) { - m_watchedFilesIgnoreTimer.start(timeout); - } -} diff --git a/src/core/FileWatcher.h b/src/core/FileWatcher.h index 7a90853606..398124d5d4 100644 --- a/src/core/FileWatcher.h +++ b/src/core/FileWatcher.h @@ -1,5 +1,5 @@ /* - * Copyright (C) 2018 KeePassXC Team + * Copyright (C) 2020 KeePassXC Team * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -37,7 +37,7 @@ class FileWatcher : public QObject bool hasSameFileChecksum(); signals: - void fileChanged(); + void fileChanged(const QString& path); public slots: void pause(); @@ -60,53 +60,4 @@ private slots: bool m_ignoreFileChange = false; }; -class BulkFileWatcher : public QObject -{ - Q_OBJECT - - enum Signal - { - Created, - Updated, - Removed - }; - -public: - explicit BulkFileWatcher(QObject* parent = nullptr); - - void clear(); - - void removePath(const QString& path); - void addPath(const QString& path); - - void ignoreFileChanges(const QString& path); - -signals: - void fileCreated(QString); - void fileChanged(QString); - void fileRemoved(QString); - -public slots: - void observeFileChanges(bool delayed = false); - -private slots: - void handleFileChanged(const QString& path); - void handleDirectoryChanged(const QString& path); - void emitSignals(); - -private: - void scheduleSignal(Signal event, const QString& path); - -private: - QMap m_watchedPaths; - QMap m_watchedFilesIgnored; - QFileSystemWatcher m_fileWatcher; - QMap> m_watchedFilesInDirectory; - // needed for Import/Export-References to prevent update after self-write - QTimer m_watchedFilesIgnoreTimer; - // needed to tolerate multiple signals for same event - QTimer m_pendingSignalsTimer; - QMap> m_pendingSignals; -}; - #endif // KEEPASSXC_FILEWATCHER_H diff --git a/src/keeshare/KeeShare.cpp b/src/keeshare/KeeShare.cpp index 567558bdc2..beff3d9502 100644 --- a/src/keeshare/KeeShare.cpp +++ b/src/keeshare/KeeShare.cpp @@ -48,7 +48,7 @@ KeeShare* KeeShare::instance() KeeShare::KeeShare(QObject* parent) : QObject(parent) { - connect(config(), SIGNAL(changed(Config::ConfigKey)), SLOT(handleSettingsChanged(Config::ConfigKey))); + connect(config(), &Config::changed, this, &KeeShare::handleSettingsChanged); } void KeeShare::init(QObject* parent) @@ -117,7 +117,7 @@ void KeeShare::setReferenceTo(Group* group, const KeeShareSettings::Reference& r return; } const auto serialized = KeeShareSettings::Reference::serialize(reference); - const auto encoded = serialized.toUtf8().toBase64(); + customData->set(KeeShare_Reference, serialized.toUtf8().toBase64()); } bool KeeShare::isEnabled(const Group* group) diff --git a/src/keeshare/ShareObserver.cpp b/src/keeshare/ShareObserver.cpp index c4c37b9161..6dc7a748d3 100644 --- a/src/keeshare/ShareObserver.cpp +++ b/src/keeshare/ShareObserver.cpp @@ -32,12 +32,14 @@ namespace const QFileInfo info(database->filePath()); return info.absoluteDir().absoluteFilePath(path); } + + constexpr int FileWatchPeriod = 30; + constexpr int FileWatchSize = 5; } // End Namespace ShareObserver::ShareObserver(QSharedPointer db, QObject* parent) : QObject(parent) , m_db(std::move(db)) - , m_fileWatcher(new BulkFileWatcher(this)) { connect(KeeShare::instance(), SIGNAL(activeChanged()), SLOT(handleDatabaseChanged())); @@ -48,10 +50,6 @@ ShareObserver::ShareObserver(QSharedPointer db, QObject* parent) connect(m_db.data(), SIGNAL(databaseModified()), SLOT(handleDatabaseChanged())); connect(m_db.data(), SIGNAL(databaseSaved()), SLOT(handleDatabaseSaved())); - connect(m_fileWatcher, SIGNAL(fileCreated(QString)), SLOT(handleFileCreated(QString))); - connect(m_fileWatcher, SIGNAL(fileChanged(QString)), SLOT(handleFileUpdated(QString))); - connect(m_fileWatcher, SIGNAL(fileRemoved(QString)), SLOT(handleFileDeleted(QString))); - handleDatabaseChanged(); } @@ -61,39 +59,33 @@ ShareObserver::~ShareObserver() void ShareObserver::deinitialize() { - m_fileWatcher->clear(); m_groupToReference.clear(); - m_referenceToGroup.clear(); + m_shareToGroup.clear(); + m_fileWatchers.clear(); } void ShareObserver::reinitialize() { - struct Update - { - Group* group; - KeeShareSettings::Reference oldReference; - KeeShareSettings::Reference newReference; - }; - - QList updated; - const QList groups = m_db->rootGroup()->groupsRecursive(true); - for (Group* group : groups) { - const Update couple{group, m_groupToReference.value(group), KeeShare::referenceOf(group)}; - if (couple.oldReference == couple.newReference) { + QList> shares; + for (Group* group : m_db->rootGroup()->groupsRecursive(true)) { + auto oldReference = m_groupToReference.value(group); + auto newReference = KeeShare::referenceOf(group); + if (oldReference == newReference) { continue; } - m_groupToReference.remove(couple.group); - m_referenceToGroup.remove(couple.oldReference); - const auto oldResolvedPath = resolvePath(couple.oldReference.path, m_db); + const auto oldResolvedPath = resolvePath(oldReference.path, m_db); + m_groupToReference.remove(group); m_shareToGroup.remove(oldResolvedPath); - if (couple.newReference.isValid()) { - m_groupToReference[couple.group] = couple.newReference; - m_referenceToGroup[couple.newReference] = couple.group; - const auto newResolvedPath = resolvePath(couple.newReference.path, m_db); - m_shareToGroup[newResolvedPath] = couple.group; + m_fileWatchers.remove(oldResolvedPath); + + if (newReference.isValid()) { + m_groupToReference[group] = newReference; + const auto newResolvedPath = resolvePath(newReference.path, m_db); + m_shareToGroup[newResolvedPath] = group; } - updated << couple; + + shares.append({group, newReference}); } QStringList success; @@ -101,25 +93,27 @@ void ShareObserver::reinitialize() QStringList error; QMap imported; QMap exported; - for (const auto& update : asConst(updated)) { - if (!update.oldReference.path.isEmpty()) { - const auto oldResolvedPath = resolvePath(update.oldReference.path, m_db); - m_fileWatcher->removePath(oldResolvedPath); - } - if (!update.newReference.path.isEmpty() && update.newReference.type != KeeShareSettings::Inactive) { - const auto newResolvedPath = resolvePath(update.newReference.path, m_db); - m_fileWatcher->addPath(newResolvedPath); + for (const auto& share : shares) { + auto group = share.first; + auto& reference = share.second; + + if (!reference.path.isEmpty() && reference.type != KeeShareSettings::Inactive) { + const auto newResolvedPath = resolvePath(reference.path, m_db); + auto fileWatcher = QSharedPointer::create(this); + connect(fileWatcher.data(), &FileWatcher::fileChanged, this, &ShareObserver::handleFileUpdated); + fileWatcher->start(newResolvedPath, FileWatchPeriod, FileWatchSize); + m_fileWatchers.insert(newResolvedPath, fileWatcher); } - if (update.newReference.isExporting()) { - exported[update.newReference.path] << update.group->name(); + if (reference.isExporting()) { + exported[reference.path] << group->name(); // export is only on save } - if (update.newReference.isImporting()) { - imported[update.newReference.path] << update.group->name(); + if (reference.isImporting()) { + imported[reference.path] << group->name(); // import has to occur immediately - const auto result = this->importShare(update.newReference.path); + const auto result = this->importShare(reference.path); if (!result.isValid()) { // tolerable result - blocked import or missing source continue; @@ -136,11 +130,13 @@ void ShareObserver::reinitialize() } } } + for (auto it = imported.cbegin(); it != imported.cend(); ++it) { if (it.value().count() > 1) { warning << tr("Multiple import source path to %1 in %2").arg(it.key(), it.value().join(", ")); } } + for (auto it = exported.cbegin(); it != exported.cend(); ++it) { if (it.value().count() > 1) { error << tr("Conflicting export target path %1 in %2").arg(it.key(), it.value().join(", ")); @@ -184,21 +180,9 @@ void ShareObserver::handleDatabaseChanged() } } -void ShareObserver::handleFileCreated(const QString& path) -{ - // there is currently no difference in handling an added share or updating from one - this->handleFileUpdated(path); -} - -void ShareObserver::handleFileDeleted(const QString& path) -{ - Q_UNUSED(path); - // There is nothing we can or should do for now, ignore deletion -} - void ShareObserver::handleFileUpdated(const QString& path) { - const Result result = this->importShare(path); + const Result result = importShare(path); if (!result.isValid()) { return; } @@ -287,9 +271,16 @@ QList ShareObserver::exportShares() for (auto it = references.cbegin(); it != references.cend(); ++it) { const auto& reference = it.value().first(); const QString resolvedPath = resolvePath(reference.config.path, m_db); - m_fileWatcher->ignoreFileChanges(resolvedPath); + auto watcher = m_fileWatchers.value(resolvedPath); + if (watcher) { + watcher->stop(); + } + results << ShareExport::intoContainer(resolvedPath, reference.config, reference.group); - m_fileWatcher->observeFileChanges(true); + + if (watcher) { + watcher->start(resolvedPath, FileWatchPeriod, FileWatchSize); + } } return results; } diff --git a/src/keeshare/ShareObserver.h b/src/keeshare/ShareObserver.h index df81fb3954..b98d589816 100644 --- a/src/keeshare/ShareObserver.h +++ b/src/keeshare/ShareObserver.h @@ -20,12 +20,13 @@ #include #include +#include #include #include "gui/MessageWidget.h" #include "keeshare/KeeShareSettings.h" -class BulkFileWatcher; +class FileWatcher; class Group; class Database; @@ -67,9 +68,7 @@ class ShareObserver : public QObject private slots: void handleDatabaseChanged(); void handleDatabaseSaved(); - void handleFileCreated(const QString& path); void handleFileUpdated(const QString& path); - void handleFileDeleted(const QString& path); private: Result importShare(const QString& path); @@ -81,11 +80,9 @@ private slots: private: QSharedPointer m_db; - QMap> m_referenceToGroup; QMap, KeeShareSettings::Reference> m_groupToReference; QMap> m_shareToGroup; - - BulkFileWatcher* m_fileWatcher; + QMap> m_fileWatchers; }; #endif // KEEPASSXC_SHAREOBSERVER_H