mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-04-06 05:57:37 +03:00
Correct behaviors when saving database fails
* Mark database dirty if saving fails * Restore database file from backup if unsafe save fails between deleting database file and copying temporary file into place * Improve error message display for opening and saving database files * Do not automatically retry saving after failure. This prevents deletion of the backup database file and improves user awareness of issues.
This commit is contained in:
parent
ec82931573
commit
3b0b5d85e9
7 changed files with 50 additions and 12 deletions
|
@ -89,6 +89,7 @@ DatabaseWidget::DatabaseWidget(QSharedPointer<Database> db, QWidget* parent)
|
|||
, m_databaseOpenWidget(new DatabaseOpenWidget(this))
|
||||
, m_keepass1OpenWidget(new KeePass1OpenWidget(this))
|
||||
, m_groupView(new GroupView(m_db.data(), m_mainSplitter))
|
||||
, m_saveAttempts(0)
|
||||
, m_fileWatcher(new DelayingFileWatcher(this))
|
||||
{
|
||||
m_messageWidget->setHidden(true);
|
||||
|
@ -859,6 +860,7 @@ void DatabaseWidget::loadDatabase(bool accepted)
|
|||
replaceDatabase(openWidget->database());
|
||||
switchToMainView();
|
||||
m_fileWatcher->restart();
|
||||
m_saveAttempts = 0;
|
||||
emit databaseUnlocked();
|
||||
} else {
|
||||
m_fileWatcher->stop();
|
||||
|
@ -1512,7 +1514,7 @@ EntryView* DatabaseWidget::entryView()
|
|||
* @param attempt current save attempt or -1 to disable attempts
|
||||
* @return true on success
|
||||
*/
|
||||
bool DatabaseWidget::save(int attempt)
|
||||
bool DatabaseWidget::save()
|
||||
{
|
||||
// Never allow saving a locked database; it causes corruption
|
||||
Q_ASSERT(!isLocked());
|
||||
|
@ -1527,6 +1529,8 @@ bool DatabaseWidget::save(int attempt)
|
|||
}
|
||||
|
||||
blockAutoReload(true);
|
||||
++m_saveAttempts;
|
||||
|
||||
// TODO: Make this async, but lock out the database widget to prevent re-entrance
|
||||
bool useAtomicSaves = config()->get("UseAtomicSaves", true).toBool();
|
||||
QString errorMessage;
|
||||
|
@ -1534,14 +1538,11 @@ bool DatabaseWidget::save(int attempt)
|
|||
blockAutoReload(false);
|
||||
|
||||
if (ok) {
|
||||
m_saveAttempts = 0;
|
||||
return true;
|
||||
}
|
||||
|
||||
if (attempt >= 0 && attempt <= 2) {
|
||||
return save(attempt + 1);
|
||||
}
|
||||
|
||||
if (attempt > 2 && useAtomicSaves) {
|
||||
if (m_saveAttempts > 2 && useAtomicSaves) {
|
||||
// Saving failed 3 times, issue a warning and attempt to resolve
|
||||
auto result = MessageBox::question(this,
|
||||
tr("Disable safe saves?"),
|
||||
|
@ -1552,11 +1553,15 @@ bool DatabaseWidget::save(int attempt)
|
|||
MessageBox::Disable);
|
||||
if (result == MessageBox::Disable) {
|
||||
config()->set("UseAtomicSaves", false);
|
||||
return save(attempt + 1);
|
||||
return save();
|
||||
}
|
||||
}
|
||||
|
||||
showMessage(tr("Writing the database failed.\n%1").arg(errorMessage), MessageWidget::Error);
|
||||
showMessage(tr("Writing the database failed: %1").arg(errorMessage),
|
||||
MessageWidget::Error,
|
||||
true,
|
||||
MessageWidget::LongAutoHideTimeout);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -1585,8 +1590,9 @@ bool DatabaseWidget::saveAs()
|
|||
// Ensure we don't recurse back into this function
|
||||
m_db->setReadOnly(false);
|
||||
m_db->setFilePath(newFilePath);
|
||||
m_saveAttempts = 0;
|
||||
|
||||
if (!save(-1)) {
|
||||
if (!save()) {
|
||||
// Failed to save, try again
|
||||
continue;
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue