Hotfix/2657 prevent share overwrite (#2746)

* Fix problem with export from newly saved database

Newly created/saved databases (or used with DatabaseWidget::saveAs)
were not exported/imported correctly.
Fixed the problem by reinitializing the ShareObserver on
DatabaseWidget::saveAs.

* Introduce warnings and prevent conflicting shares

Introduced several warnings and errors to indicate improper settings.
Prevent export when a path is used multiple times (only the file path is
checked - may ignore multiple similar ways to reference a share).

* Improve KeeShare integration in DatabaseWidget

Moved initial KeeShare association to constructor.
Introduced Q_UNUSED to indicate need for assignment statement.
This commit is contained in:
ckieschnick 2019-03-16 03:39:46 +01:00 committed by Jonathan White
parent ebb87e6379
commit 11ecaf4fa4
8 changed files with 109 additions and 19 deletions

View File

@ -208,6 +208,12 @@ DatabaseWidget::DatabaseWidget(QSharedPointer<Database> db, QWidget* parent)
} }
#endif #endif
#ifdef WITH_XC_KEESHARE
// We need to reregister the database to allow exports
// from a newly created database
KeeShare::instance()->connectDatabase(m_db, {});
#endif
switchToMainView(); switchToMainView();
} }
@ -391,6 +397,9 @@ void DatabaseWidget::replaceDatabase(QSharedPointer<Database> db)
processAutoOpen(); processAutoOpen();
#if defined(WITH_XC_KEESHARE) #if defined(WITH_XC_KEESHARE)
KeeShare::instance()->connectDatabase(m_db, oldDb); KeeShare::instance()->connectDatabase(m_db, oldDb);
#else
// Keep the instance active till the end of this function
Q_UNUSED(oldDb);
#endif #endif
} }

View File

@ -36,9 +36,9 @@ public:
{ {
} }
void set(Group* temporaryGroup) const void set(Group* temporaryGroup, QSharedPointer<Database> database) const
{ {
editPage->set(widget, temporaryGroup); editPage->set(widget, temporaryGroup, database);
} }
void assign() const void assign() const
@ -133,7 +133,7 @@ void EditGroupWidget::loadGroup(Group* group, bool create, const QSharedPointer<
m_editWidgetProperties->setCustomData(m_temporaryGroup->customData()); m_editWidgetProperties->setCustomData(m_temporaryGroup->customData());
for (const ExtraPage& page : asConst(m_extraPages)) { for (const ExtraPage& page : asConst(m_extraPages)) {
page.set(m_temporaryGroup.data()); page.set(m_temporaryGroup.data(), m_db);
} }
setCurrentPage(0); setCurrentPage(0);

View File

@ -43,7 +43,7 @@ public:
virtual QString name() = 0; virtual QString name() = 0;
virtual QIcon icon() = 0; virtual QIcon icon() = 0;
virtual QWidget* createWidget() = 0; virtual QWidget* createWidget() = 0;
virtual void set(QWidget* widget, Group* tempoaryGroup) = 0; virtual void set(QWidget* widget, Group* tempoaryGroup, QSharedPointer<Database> database) = 0;
virtual void assign(QWidget* widget) = 0; virtual void assign(QWidget* widget) = 0;
}; };

View File

@ -25,6 +25,7 @@
#include "core/Entry.h" #include "core/Entry.h"
#include "core/FilePath.h" #include "core/FilePath.h"
#include "core/FileWatcher.h" #include "core/FileWatcher.h"
#include "core/Global.h"
#include "core/Group.h" #include "core/Group.h"
#include "core/Merger.h" #include "core/Merger.h"
#include "core/Metadata.h" #include "core/Metadata.h"
@ -191,7 +192,7 @@ void ShareObserver::reinitialize()
const auto active = KeeShare::active(); const auto active = KeeShare::active();
QList<Update> updated; QList<Update> updated;
QList<Group*> groups = m_db->rootGroup()->groupsRecursive(true); const QList<Group*> groups = m_db->rootGroup()->groupsRecursive(true);
for (Group* group : groups) { for (Group* group : groups) {
Update couple{group, m_groupToReference.value(group), KeeShare::referenceOf(group)}; Update couple{group, m_groupToReference.value(group), KeeShare::referenceOf(group)};
if (couple.oldReference == couple.newReference) { if (couple.oldReference == couple.newReference) {
@ -214,7 +215,9 @@ void ShareObserver::reinitialize()
QStringList success; QStringList success;
QStringList warning; QStringList warning;
QStringList error; QStringList error;
for (const auto& update : updated) { QMap<QString, QStringList> imported;
QMap<QString, QStringList> exported;
for (const auto& update : asConst(updated)) {
if (!update.oldReference.path.isEmpty()) { if (!update.oldReference.path.isEmpty()) {
m_fileWatcher->removePath(update.oldReference.path); m_fileWatcher->removePath(update.oldReference.path);
} }
@ -222,8 +225,12 @@ void ShareObserver::reinitialize()
if (!update.newReference.path.isEmpty() && update.newReference.type != KeeShareSettings::Inactive) { if (!update.newReference.path.isEmpty() && update.newReference.type != KeeShareSettings::Inactive) {
m_fileWatcher->addPath(update.newReference.path); m_fileWatcher->addPath(update.newReference.path);
} }
if (update.newReference.isExporting()) {
exported[update.newReference.path] << update.group->name();
}
if (update.newReference.isImporting()) { if (update.newReference.isImporting()) {
imported[update.newReference.path] << update.group->name();
const auto result = this->importFromReferenceContainer(update.newReference.path); const auto result = this->importFromReferenceContainer(update.newReference.path);
if (!result.isValid()) { if (!result.isValid()) {
// tolerable result - blocked import or missing source // tolerable result - blocked import or missing source
@ -241,6 +248,16 @@ 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(", "));
}
}
notifyAbout(success, warning, error); notifyAbout(success, warning, error);
} }
@ -733,28 +750,55 @@ ShareObserver::Result ShareObserver::exportIntoReferenceUnsignedContainer(const
QList<ShareObserver::Result> ShareObserver::exportIntoReferenceContainers() QList<ShareObserver::Result> ShareObserver::exportIntoReferenceContainers()
{ {
QList<Result> results; QList<Result> results;
struct Reference
{
KeeShareSettings::Reference config;
const Group* group;
};
QMap<QString, QList<Reference>> references;
const auto groups = m_db->rootGroup()->groupsRecursive(true); const auto groups = m_db->rootGroup()->groupsRecursive(true);
for (const auto* group : groups) { for (const auto* group : groups) {
const auto reference = KeeShare::referenceOf(group); const auto reference = KeeShare::referenceOf(group);
if (!reference.isExporting()) { if (!reference.isExporting()) {
continue; continue;
} }
references[reference.path] << Reference{reference, group};
}
m_fileWatcher->ignoreFileChanges(reference.path); for (auto it = references.cbegin(); it != references.cend(); ++it) {
QScopedPointer<Database> targetDb(exportIntoContainer(reference, group)); if (it.value().count() != 1) {
QFileInfo info(reference.path); const auto path = it.value().first().config.path;
QStringList groups;
for (const auto& reference : it.value()) {
groups << reference.group->name();
}
results << Result{
path, Result::Error, tr("Conflicting export target path %1 in %2").arg(path, groups.join(", "))};
}
}
if (!results.isEmpty()) {
// We need to block export due to config
return results;
}
for (auto it = references.cbegin(); it != references.cend(); ++it) {
const auto& reference = it.value().first();
m_fileWatcher->ignoreFileChanges(reference.config.path);
QScopedPointer<Database> targetDb(exportIntoContainer(reference.config, reference.group));
QFileInfo info(reference.config.path);
if (isOfExportType(info, KeeShare::signedContainerFileType())) { if (isOfExportType(info, KeeShare::signedContainerFileType())) {
results << exportIntoReferenceSignedContainer(reference, targetDb.data()); results << exportIntoReferenceSignedContainer(reference.config, targetDb.data());
m_fileWatcher->observeFileChanges(true); m_fileWatcher->observeFileChanges(true);
continue; continue;
} }
if (isOfExportType(info, KeeShare::unsignedContainerFileType())) { if (isOfExportType(info, KeeShare::unsignedContainerFileType())) {
results << exportIntoReferenceUnsignedContainer(reference, targetDb.data()); results << exportIntoReferenceUnsignedContainer(reference.config, targetDb.data());
m_fileWatcher->observeFileChanges(true); m_fileWatcher->observeFileChanges(true);
continue; continue;
} }
Q_ASSERT(false); Q_ASSERT(false);
results << Result{reference.path, Result::Error, tr("Unexpected export error occurred")}; results << Result{reference.config.path, Result::Error, tr("Unexpected export error occurred")};
} }
return results; return results;
} }
@ -767,6 +811,7 @@ void ShareObserver::handleDatabaseSaved()
QStringList error; QStringList error;
QStringList warning; QStringList warning;
QStringList success; QStringList success;
const auto results = exportIntoReferenceContainers(); const auto results = exportIntoReferenceContainers();
for (const Result& result : results) { for (const Result& result : results) {
if (!result.isValid()) { if (!result.isValid()) {

View File

@ -42,10 +42,10 @@ QWidget* EditGroupPageKeeShare::createWidget()
return new EditGroupWidgetKeeShare(); return new EditGroupWidgetKeeShare();
} }
void EditGroupPageKeeShare::set(QWidget* widget, Group* temporaryGroup) void EditGroupPageKeeShare::set(QWidget* widget, Group* temporaryGroup, QSharedPointer<Database> database)
{ {
EditGroupWidgetKeeShare* settingsWidget = reinterpret_cast<EditGroupWidgetKeeShare*>(widget); EditGroupWidgetKeeShare* settingsWidget = reinterpret_cast<EditGroupWidgetKeeShare*>(widget);
settingsWidget->setGroup(temporaryGroup); settingsWidget->setGroup(temporaryGroup, database);
} }
void EditGroupPageKeeShare::assign(QWidget* widget) void EditGroupPageKeeShare::assign(QWidget* widget)

View File

@ -30,7 +30,7 @@ public:
QString name() override; QString name() override;
QIcon icon() override; QIcon icon() override;
QWidget* createWidget() override; QWidget* createWidget() override;
void set(QWidget* widget, Group* temporaryGroup) override; void set(QWidget* widget, Group* temporaryGroup, QSharedPointer<Database> database) override;
void assign(QWidget* widget) override; void assign(QWidget* widget) override;
}; };

View File

@ -85,12 +85,13 @@ EditGroupWidgetKeeShare::~EditGroupWidgetKeeShare()
{ {
} }
void EditGroupWidgetKeeShare::setGroup(Group* temporaryGroup) void EditGroupWidgetKeeShare::setGroup(Group* temporaryGroup, QSharedPointer<Database> database)
{ {
if (m_temporaryGroup) { if (m_temporaryGroup) {
m_temporaryGroup->disconnect(this); m_temporaryGroup->disconnect(this);
} }
m_database = database;
m_temporaryGroup = temporaryGroup; m_temporaryGroup = temporaryGroup;
if (m_temporaryGroup) { if (m_temporaryGroup) {
@ -128,9 +129,43 @@ void EditGroupWidgetKeeShare::showSharingState()
.arg(supportedExtensions.join(", ")), .arg(supportedExtensions.join(", ")),
MessageWidget::Warning); MessageWidget::Warning);
return; return;
} else {
m_ui->messageWidget->hide();
} }
const auto groups = m_database->rootGroup()->groupsRecursive(true);
bool conflictExport = false;
bool multipleImport = false;
bool cycleImportExport = false;
for (const auto* group : groups) {
if (group->uuid() == m_temporaryGroup->uuid()) {
continue;
}
const auto other = KeeShare::referenceOf(group);
if (other.path != reference.path) {
continue;
}
multipleImport |= other.isImporting() && reference.isImporting();
conflictExport |= other.isExporting() && reference.isExporting();
cycleImportExport |=
(other.isImporting() && reference.isExporting()) || (other.isExporting() && reference.isImporting());
}
if (conflictExport) {
m_ui->messageWidget->showMessage(tr("The export container %1 is already referenced.").arg(reference.path),
MessageWidget::Error);
return;
}
if (multipleImport) {
m_ui->messageWidget->showMessage(tr("The import container %1 is already imported.").arg(reference.path),
MessageWidget::Warning);
return;
}
if (cycleImportExport) {
m_ui->messageWidget->showMessage(
tr("The container %1 imported and export by different groups.").arg(reference.path),
MessageWidget::Warning);
return;
}
m_ui->messageWidget->hide();
} }
const auto active = KeeShare::active(); const auto active = KeeShare::active();
if (!active.in && !active.out) { if (!active.in && !active.out) {

View File

@ -37,7 +37,7 @@ public:
explicit EditGroupWidgetKeeShare(QWidget* parent = nullptr); explicit EditGroupWidgetKeeShare(QWidget* parent = nullptr);
~EditGroupWidgetKeeShare(); ~EditGroupWidgetKeeShare();
void setGroup(Group* temporaryGroup); void setGroup(Group* temporaryGroup, QSharedPointer<Database> database);
private slots: private slots:
void showSharingState(); void showSharingState();
@ -55,6 +55,7 @@ private slots:
private: private:
QScopedPointer<Ui::EditGroupWidgetKeeShare> m_ui; QScopedPointer<Ui::EditGroupWidgetKeeShare> m_ui;
QPointer<Group> m_temporaryGroup; QPointer<Group> m_temporaryGroup;
QSharedPointer<Database> m_database;
}; };
#endif // KEEPASSXC_EDITGROUPWIDGETKEESHARE_H #endif // KEEPASSXC_EDITGROUPWIDGETKEESHARE_H