mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-22 20:51:23 -05:00
Allow specifing database backup paths. (#7035)
- Default backupFilePath is '{DB_FILENAME}.old.kdbx' to conform to existing standards - Implement backupPathPattern tests. - Show tooltip on how to format database backup location text field.
This commit is contained in:
parent
8d7e491810
commit
84ff6a13f9
@ -215,6 +215,10 @@
|
|||||||
<source>Monochrome</source>
|
<source>Monochrome</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Select backup storage directory</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>ApplicationSettingsWidgetGeneral</name>
|
<name>ApplicationSettingsWidgetGeneral</name>
|
||||||
@ -440,6 +444,22 @@
|
|||||||
<source>Directly write to database file (dangerous)</source>
|
<source>Directly write to database file (dangerous)</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Choose...</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Backup destination</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Specifies the database backup file location. Occurences of "{DB_FILENAME}" are replaced with the filename of the saved database without extension. {TIME:<format>} is replaced with the backup time, see https://doc.qt.io/qt-5/qdatetime.html#toString. <format> defaults to format string "dd_MM_yyyy_hh-mm-ss".</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>{DB_FILENAME}.old.kdbx</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>ApplicationSettingsWidgetSecurity</name>
|
<name>ApplicationSettingsWidgetSecurity</name>
|
||||||
|
@ -121,7 +121,7 @@ int Add::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<Q
|
|||||||
}
|
}
|
||||||
|
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!database->save(Database::Atomic, false, &errorMessage)) {
|
if (!database->save(Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Writing the database failed %1.").arg(errorMessage) << endl;
|
err << QObject::tr("Writing the database failed %1.").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -63,7 +63,7 @@ int AddGroup::executeWithDatabase(QSharedPointer<Database> database, QSharedPoin
|
|||||||
newGroup->setParent(parentGroup);
|
newGroup->setParent(parentGroup);
|
||||||
|
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!database->save(Database::Atomic, false, &errorMessage)) {
|
if (!database->save(Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Writing the database failed %1.").arg(errorMessage) << endl;
|
err << QObject::tr("Writing the database failed %1.").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -165,7 +165,7 @@ int Create::execute(const QStringList& arguments)
|
|||||||
}
|
}
|
||||||
|
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!db->saveAs(databaseFilename, Database::Atomic, false, &errorMessage)) {
|
if (!db->saveAs(databaseFilename, Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Failed to save the database: %1.").arg(errorMessage) << endl;
|
err << QObject::tr("Failed to save the database: %1.").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -126,7 +126,7 @@ int Edit::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<
|
|||||||
entry->endUpdate();
|
entry->endUpdate();
|
||||||
|
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!database->save(Database::Atomic, false, &errorMessage)) {
|
if (!database->save(Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Writing the database failed: %1").arg(errorMessage) << endl;
|
err << QObject::tr("Writing the database failed: %1").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -75,7 +75,7 @@ int Import::execute(const QStringList& arguments)
|
|||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!db->saveAs(dbPath, Database::Atomic, false, &errorMessage)) {
|
if (!db->saveAs(dbPath, Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Failed to save the database: %1.").arg(errorMessage) << endl;
|
err << QObject::tr("Failed to save the database: %1.").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -95,7 +95,7 @@ int Merge::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer
|
|||||||
|
|
||||||
if (!changeList.isEmpty() && !parser->isSet(Merge::DryRunOption)) {
|
if (!changeList.isEmpty() && !parser->isSet(Merge::DryRunOption)) {
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!database->save(Database::Atomic, false, &errorMessage)) {
|
if (!database->save(Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Unable to save database to file : %1").arg(errorMessage) << endl;
|
err << QObject::tr("Unable to save database to file : %1").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -65,7 +65,7 @@ int Move::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<
|
|||||||
entry->endUpdate();
|
entry->endUpdate();
|
||||||
|
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!database->save(Database::Atomic, false, &errorMessage)) {
|
if (!database->save(Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Writing the database failed %1.").arg(errorMessage) << endl;
|
err << QObject::tr("Writing the database failed %1.").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -53,7 +53,7 @@ int Remove::executeWithDatabase(QSharedPointer<Database> database, QSharedPointe
|
|||||||
};
|
};
|
||||||
|
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!database->save(Database::Atomic, false, &errorMessage)) {
|
if (!database->save(Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Unable to save database to file: %1").arg(errorMessage) << endl;
|
err << QObject::tr("Unable to save database to file: %1").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -63,7 +63,7 @@ int RemoveGroup::executeWithDatabase(QSharedPointer<Database> database, QSharedP
|
|||||||
};
|
};
|
||||||
|
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
if (!database->save(Database::Atomic, false, &errorMessage)) {
|
if (!database->save(Database::Atomic, QString(), &errorMessage)) {
|
||||||
err << QObject::tr("Unable to save database to file: %1").arg(errorMessage) << endl;
|
err << QObject::tr("Unable to save database to file: %1").arg(errorMessage) << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -61,6 +61,7 @@ static const QHash<Config::ConfigKey, ConfigDirective> configStrings = {
|
|||||||
{Config::AutoSaveOnExit,{QS("AutoSaveOnExit"), Roaming, true}},
|
{Config::AutoSaveOnExit,{QS("AutoSaveOnExit"), Roaming, true}},
|
||||||
{Config::AutoSaveNonDataChanges,{QS("AutoSaveNonDataChanges"), Roaming, true}},
|
{Config::AutoSaveNonDataChanges,{QS("AutoSaveNonDataChanges"), Roaming, true}},
|
||||||
{Config::BackupBeforeSave,{QS("BackupBeforeSave"), Roaming, false}},
|
{Config::BackupBeforeSave,{QS("BackupBeforeSave"), Roaming, false}},
|
||||||
|
{Config::BackupFilePathPattern,{QS("BackupFilePathPattern"), Roaming, QString("{DB_FILENAME}.old.kdbx")}},
|
||||||
{Config::UseAtomicSaves,{QS("UseAtomicSaves"), Roaming, true}},
|
{Config::UseAtomicSaves,{QS("UseAtomicSaves"), Roaming, true}},
|
||||||
{Config::UseDirectWriteSaves,{QS("UseDirectWriteSaves"), Local, false}},
|
{Config::UseDirectWriteSaves,{QS("UseDirectWriteSaves"), Local, false}},
|
||||||
{Config::SearchLimitGroup,{QS("SearchLimitGroup"), Roaming, false}},
|
{Config::SearchLimitGroup,{QS("SearchLimitGroup"), Roaming, false}},
|
||||||
@ -229,6 +230,11 @@ QVariant Config::get(ConfigKey key)
|
|||||||
return m_settings->value(cfg.name, defaultValue);
|
return m_settings->value(cfg.name, defaultValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QVariant Config::getDefault(Config::ConfigKey key)
|
||||||
|
{
|
||||||
|
return configStrings[key].defaultValue;
|
||||||
|
}
|
||||||
|
|
||||||
bool Config::hasAccessError()
|
bool Config::hasAccessError()
|
||||||
{
|
{
|
||||||
return m_settings->status() & QSettings::AccessError;
|
return m_settings->status() & QSettings::AccessError;
|
||||||
|
@ -43,6 +43,7 @@ public:
|
|||||||
AutoSaveOnExit,
|
AutoSaveOnExit,
|
||||||
AutoSaveNonDataChanges,
|
AutoSaveNonDataChanges,
|
||||||
BackupBeforeSave,
|
BackupBeforeSave,
|
||||||
|
BackupFilePathPattern,
|
||||||
UseAtomicSaves,
|
UseAtomicSaves,
|
||||||
UseDirectWriteSaves,
|
UseDirectWriteSaves,
|
||||||
SearchLimitGroup,
|
SearchLimitGroup,
|
||||||
@ -195,6 +196,7 @@ public:
|
|||||||
|
|
||||||
~Config() override;
|
~Config() override;
|
||||||
QVariant get(ConfigKey key);
|
QVariant get(ConfigKey key);
|
||||||
|
QVariant getDefault(ConfigKey key);
|
||||||
QString getFileName();
|
QString getFileName();
|
||||||
void set(ConfigKey key, const QVariant& value);
|
void set(ConfigKey key, const QVariant& value);
|
||||||
void remove(ConfigKey key);
|
void remove(ConfigKey key);
|
||||||
|
@ -178,10 +178,10 @@ bool Database::isSaving()
|
|||||||
*
|
*
|
||||||
* @param error error message in case of failure
|
* @param error error message in case of failure
|
||||||
* @param atomic Use atomic file transactions
|
* @param atomic Use atomic file transactions
|
||||||
* @param backup Backup the existing database file, if exists
|
* @param backupFilePath Absolute file path to write the backup file to. Pass an empty QString to disable backup.
|
||||||
* @return true on success
|
* @return true on success
|
||||||
*/
|
*/
|
||||||
bool Database::save(SaveAction action, bool backup, QString* error)
|
bool Database::save(SaveAction action, const QString& backupFilePath, QString* error)
|
||||||
{
|
{
|
||||||
Q_ASSERT(!m_data.filePath.isEmpty());
|
Q_ASSERT(!m_data.filePath.isEmpty());
|
||||||
if (m_data.filePath.isEmpty()) {
|
if (m_data.filePath.isEmpty()) {
|
||||||
@ -191,7 +191,7 @@ bool Database::save(SaveAction action, bool backup, QString* error)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return saveAs(m_data.filePath, action, backup, error);
|
return saveAs(m_data.filePath, action, backupFilePath, error);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -209,10 +209,11 @@ bool Database::save(SaveAction action, bool backup, QString* error)
|
|||||||
* @param filePath Absolute path of the file to save
|
* @param filePath Absolute path of the file to save
|
||||||
* @param error error message in case of failure
|
* @param error error message in case of failure
|
||||||
* @param atomic Use atomic file transactions
|
* @param atomic Use atomic file transactions
|
||||||
* @param backup Backup the existing database file, if exists
|
* @param backupFilePath Absolute path to the location where the backup should be stored. Passing an empty string
|
||||||
|
* disables backup.
|
||||||
* @return true on success
|
* @return true on success
|
||||||
*/
|
*/
|
||||||
bool Database::saveAs(const QString& filePath, SaveAction action, bool backup, QString* error)
|
bool Database::saveAs(const QString& filePath, SaveAction action, const QString& backupFilePath, QString* error)
|
||||||
{
|
{
|
||||||
// Disallow overlapping save operations
|
// Disallow overlapping save operations
|
||||||
if (isSaving()) {
|
if (isSaving()) {
|
||||||
@ -260,7 +261,7 @@ bool Database::saveAs(const QString& filePath, SaveAction action, bool backup, Q
|
|||||||
QFileInfo fileInfo(filePath);
|
QFileInfo fileInfo(filePath);
|
||||||
auto realFilePath = fileInfo.exists() ? fileInfo.canonicalFilePath() : fileInfo.absoluteFilePath();
|
auto realFilePath = fileInfo.exists() ? fileInfo.canonicalFilePath() : fileInfo.absoluteFilePath();
|
||||||
bool isNewFile = !QFile::exists(realFilePath);
|
bool isNewFile = !QFile::exists(realFilePath);
|
||||||
bool ok = AsyncTask::runAndWaitForFuture([&] { return performSave(realFilePath, action, backup, error); });
|
bool ok = AsyncTask::runAndWaitForFuture([&] { return performSave(realFilePath, action, backupFilePath, error); });
|
||||||
if (ok) {
|
if (ok) {
|
||||||
markAsClean();
|
markAsClean();
|
||||||
setFilePath(filePath);
|
setFilePath(filePath);
|
||||||
@ -276,10 +277,10 @@ bool Database::saveAs(const QString& filePath, SaveAction action, bool backup, Q
|
|||||||
return ok;
|
return ok;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::performSave(const QString& filePath, SaveAction action, bool backup, QString* error)
|
bool Database::performSave(const QString& filePath, SaveAction action, const QString& backupFilePath, QString* error)
|
||||||
{
|
{
|
||||||
if (backup) {
|
if (!backupFilePath.isNull()) {
|
||||||
backupDatabase(filePath);
|
backupDatabase(filePath, backupFilePath);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
@ -337,7 +338,7 @@ bool Database::performSave(const QString& filePath, SaveAction action, bool back
|
|||||||
tempFile.setFileTime(createTime, QFile::FileBirthTime);
|
tempFile.setFileTime(createTime, QFile::FileBirthTime);
|
||||||
#endif
|
#endif
|
||||||
return true;
|
return true;
|
||||||
} else if (!backup || !restoreDatabase(filePath)) {
|
} else if (backupFilePath.isEmpty() || !restoreDatabase(filePath, backupFilePath)) {
|
||||||
// Failed to copy new database in place, and
|
// Failed to copy new database in place, and
|
||||||
// failed to restore from backup or backups disabled
|
// failed to restore from backup or backups disabled
|
||||||
tempFile.setAutoRemove(false);
|
tempFile.setAutoRemove(false);
|
||||||
@ -485,23 +486,26 @@ void Database::releaseData()
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Remove the old backup and replace it with a new one
|
* Remove the old backup and replace it with a new one. Backup name is taken from destinationFilePath.
|
||||||
* backups are named <filename>.old.<extension>
|
* Non-existing parent directories will be created automatically.
|
||||||
*
|
*
|
||||||
* @param filePath Path to the file to backup
|
* @param filePath Path to the file to backup
|
||||||
|
* @param destinationFilePath Path to the backup destination file
|
||||||
* @return true on success
|
* @return true on success
|
||||||
*/
|
*/
|
||||||
bool Database::backupDatabase(const QString& filePath)
|
bool Database::backupDatabase(const QString& filePath, const QString& destinationFilePath)
|
||||||
{
|
{
|
||||||
static auto re = QRegularExpression("(\\.[^.]+)$");
|
// Ensure that the path to write to actually exists
|
||||||
|
auto parentDirectory = QFileInfo(destinationFilePath).absoluteDir();
|
||||||
auto match = re.match(filePath);
|
if (!parentDirectory.exists()) {
|
||||||
auto backupFilePath = filePath;
|
if (!QDir().mkpath(parentDirectory.absolutePath())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
auto perms = QFile::permissions(filePath);
|
auto perms = QFile::permissions(filePath);
|
||||||
backupFilePath = backupFilePath.replace(re, "") + ".old" + match.captured(1);
|
QFile::remove(destinationFilePath);
|
||||||
QFile::remove(backupFilePath);
|
bool res = QFile::copy(filePath, destinationFilePath);
|
||||||
bool res = QFile::copy(filePath, backupFilePath);
|
QFile::setPermissions(destinationFilePath, perms);
|
||||||
QFile::setPermissions(backupFilePath, perms);
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -513,17 +517,13 @@ bool Database::backupDatabase(const QString& filePath)
|
|||||||
* @param filePath Path to the file to restore
|
* @param filePath Path to the file to restore
|
||||||
* @return true on success
|
* @return true on success
|
||||||
*/
|
*/
|
||||||
bool Database::restoreDatabase(const QString& filePath)
|
bool Database::restoreDatabase(const QString& filePath, const QString& fromBackupFilePath)
|
||||||
{
|
{
|
||||||
static auto re = QRegularExpression("^(.*?)(\\.[^.]+)?$");
|
|
||||||
|
|
||||||
auto match = re.match(filePath);
|
|
||||||
auto perms = QFile::permissions(filePath);
|
auto perms = QFile::permissions(filePath);
|
||||||
auto backupFilePath = match.captured(1) + ".old" + match.captured(2);
|
|
||||||
// Only try to restore if the backup file actually exists
|
// Only try to restore if the backup file actually exists
|
||||||
if (QFile::exists(backupFilePath)) {
|
if (QFile::exists(fromBackupFilePath)) {
|
||||||
QFile::remove(filePath);
|
QFile::remove(filePath);
|
||||||
if (QFile::copy(backupFilePath, filePath)) {
|
if (QFile::copy(fromBackupFilePath, filePath)) {
|
||||||
return QFile::setPermissions(filePath, perms);
|
return QFile::setPermissions(filePath, perms);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -79,8 +79,11 @@ public:
|
|||||||
QSharedPointer<const CompositeKey> key,
|
QSharedPointer<const CompositeKey> key,
|
||||||
QString* error = nullptr,
|
QString* error = nullptr,
|
||||||
bool readOnly = false);
|
bool readOnly = false);
|
||||||
bool save(SaveAction action = Atomic, bool backup = false, QString* error = nullptr);
|
bool save(SaveAction action = Atomic, const QString& backupFilePath = QString(), QString* error = nullptr);
|
||||||
bool saveAs(const QString& filePath, SaveAction action = Atomic, bool backup = false, QString* error = nullptr);
|
bool saveAs(const QString& filePath,
|
||||||
|
SaveAction action = Atomic,
|
||||||
|
const QString& backupFilePath = QString(),
|
||||||
|
QString* error = nullptr);
|
||||||
bool extract(QByteArray&, QString* error = nullptr);
|
bool extract(QByteArray&, QString* error = nullptr);
|
||||||
bool import(const QString& xmlExportPath, QString* error = nullptr);
|
bool import(const QString& xmlExportPath, QString* error = nullptr);
|
||||||
|
|
||||||
@ -203,9 +206,9 @@ private:
|
|||||||
void createRecycleBin();
|
void createRecycleBin();
|
||||||
|
|
||||||
bool writeDatabase(QIODevice* device, QString* error = nullptr);
|
bool writeDatabase(QIODevice* device, QString* error = nullptr);
|
||||||
bool backupDatabase(const QString& filePath);
|
bool backupDatabase(const QString& filePath, const QString& destinationFilePath);
|
||||||
bool restoreDatabase(const QString& filePath);
|
bool restoreDatabase(const QString& filePath, const QString& fromBackupFilePath);
|
||||||
bool performSave(const QString& filePath, SaveAction flags, bool backup, QString* error);
|
bool performSave(const QString& filePath, SaveAction flags, const QString& backupFilePath, QString* error);
|
||||||
void startModifiedTimer();
|
void startModifiedTimer();
|
||||||
void stopModifiedTimer();
|
void stopModifiedTimer();
|
||||||
|
|
||||||
|
@ -22,7 +22,10 @@
|
|||||||
#include "config-keepassx.h"
|
#include "config-keepassx.h"
|
||||||
#include "git-info.h"
|
#include "git-info.h"
|
||||||
|
|
||||||
|
#include "core/Clock.h"
|
||||||
|
|
||||||
#include <QElapsedTimer>
|
#include <QElapsedTimer>
|
||||||
|
#include <QFileInfo>
|
||||||
#include <QImageReader>
|
#include <QImageReader>
|
||||||
#include <QLocale>
|
#include <QLocale>
|
||||||
#include <QMetaProperty>
|
#include <QMetaProperty>
|
||||||
@ -376,4 +379,37 @@ namespace Tools
|
|||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString substituteBackupFilePath(QString pattern, const QString& databasePath)
|
||||||
|
{
|
||||||
|
// Fail if substitution fails
|
||||||
|
if (databasePath.isEmpty()) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Replace backup pattern
|
||||||
|
QFileInfo dbFileInfo(databasePath);
|
||||||
|
QString baseName = dbFileInfo.completeBaseName();
|
||||||
|
|
||||||
|
pattern.replace(QString("{DB_FILENAME}"), baseName);
|
||||||
|
|
||||||
|
auto re = QRegularExpression(R"(\{TIME(?::([^\\]*))?\})");
|
||||||
|
auto match = re.match(pattern);
|
||||||
|
while (match.hasMatch()) {
|
||||||
|
// Extract time format specifier
|
||||||
|
auto formatSpecifier = QString("dd_MM_yyyy_hh-mm-ss");
|
||||||
|
if (!match.captured(1).isEmpty()) {
|
||||||
|
formatSpecifier = match.captured(1);
|
||||||
|
}
|
||||||
|
auto replacement = Clock::currentDateTime().toString(formatSpecifier);
|
||||||
|
pattern.replace(match.capturedStart(), match.capturedLength(), replacement);
|
||||||
|
match = re.match(pattern);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Replace escaped braces
|
||||||
|
pattern.replace("\\{", "{");
|
||||||
|
pattern.replace("\\}", "}");
|
||||||
|
|
||||||
|
return pattern;
|
||||||
|
}
|
||||||
} // namespace Tools
|
} // namespace Tools
|
||||||
|
@ -75,6 +75,8 @@ namespace Tools
|
|||||||
}
|
}
|
||||||
|
|
||||||
QVariantMap qo2qvm(const QObject* object, const QStringList& ignoredProperties = {"objectName"});
|
QVariantMap qo2qvm(const QObject* object, const QStringList& ignoredProperties = {"objectName"});
|
||||||
|
|
||||||
|
QString substituteBackupFilePath(QString pattern, const QString& databasePath);
|
||||||
} // namespace Tools
|
} // namespace Tools
|
||||||
|
|
||||||
#endif // KEEPASSX_TOOLS_H
|
#endif // KEEPASSX_TOOLS_H
|
||||||
|
@ -19,6 +19,8 @@
|
|||||||
#include "ApplicationSettingsWidget.h"
|
#include "ApplicationSettingsWidget.h"
|
||||||
#include "ui_ApplicationSettingsWidgetGeneral.h"
|
#include "ui_ApplicationSettingsWidgetGeneral.h"
|
||||||
#include "ui_ApplicationSettingsWidgetSecurity.h"
|
#include "ui_ApplicationSettingsWidgetSecurity.h"
|
||||||
|
#include <QDesktopServices>
|
||||||
|
#include <QDir>
|
||||||
|
|
||||||
#include "config-keepassx.h"
|
#include "config-keepassx.h"
|
||||||
|
|
||||||
@ -28,6 +30,7 @@
|
|||||||
#include "gui/MainWindow.h"
|
#include "gui/MainWindow.h"
|
||||||
#include "gui/osutils/OSUtils.h"
|
#include "gui/osutils/OSUtils.h"
|
||||||
|
|
||||||
|
#include "FileDialog.h"
|
||||||
#include "MessageBox.h"
|
#include "MessageBox.h"
|
||||||
#ifdef Q_OS_MACOS
|
#ifdef Q_OS_MACOS
|
||||||
#include "touchid/TouchID.h"
|
#include "touchid/TouchID.h"
|
||||||
@ -112,6 +115,12 @@ ApplicationSettingsWidget::ApplicationSettingsWidget(QWidget* parent)
|
|||||||
connect(m_generalUi->useAlternativeSaveCheckBox, SIGNAL(toggled(bool)),
|
connect(m_generalUi->useAlternativeSaveCheckBox, SIGNAL(toggled(bool)),
|
||||||
m_generalUi->alternativeSaveComboBox, SLOT(setEnabled(bool)));
|
m_generalUi->alternativeSaveComboBox, SLOT(setEnabled(bool)));
|
||||||
|
|
||||||
|
connect(m_generalUi->backupBeforeSaveCheckBox, SIGNAL(toggled(bool)),
|
||||||
|
m_generalUi->backupFilePath, SLOT(setEnabled(bool)));
|
||||||
|
connect(m_generalUi->backupBeforeSaveCheckBox, SIGNAL(toggled(bool)),
|
||||||
|
m_generalUi->backupFilePathPicker, SLOT(setEnabled(bool)));
|
||||||
|
connect(m_generalUi->backupFilePathPicker, SIGNAL(pressed()), SLOT(selectBackupDirectory()));
|
||||||
|
|
||||||
connect(m_secUi->clearClipboardCheckBox, SIGNAL(toggled(bool)),
|
connect(m_secUi->clearClipboardCheckBox, SIGNAL(toggled(bool)),
|
||||||
m_secUi->clearClipboardSpinBox, SLOT(setEnabled(bool)));
|
m_secUi->clearClipboardSpinBox, SLOT(setEnabled(bool)));
|
||||||
connect(m_secUi->clearSearchCheckBox, SIGNAL(toggled(bool)),
|
connect(m_secUi->clearSearchCheckBox, SIGNAL(toggled(bool)),
|
||||||
@ -188,6 +197,9 @@ void ApplicationSettingsWidget::loadSettings()
|
|||||||
m_generalUi->autoSaveOnExitCheckBox->setChecked(config()->get(Config::AutoSaveOnExit).toBool());
|
m_generalUi->autoSaveOnExitCheckBox->setChecked(config()->get(Config::AutoSaveOnExit).toBool());
|
||||||
m_generalUi->autoSaveNonDataChangesCheckBox->setChecked(config()->get(Config::AutoSaveNonDataChanges).toBool());
|
m_generalUi->autoSaveNonDataChangesCheckBox->setChecked(config()->get(Config::AutoSaveNonDataChanges).toBool());
|
||||||
m_generalUi->backupBeforeSaveCheckBox->setChecked(config()->get(Config::BackupBeforeSave).toBool());
|
m_generalUi->backupBeforeSaveCheckBox->setChecked(config()->get(Config::BackupBeforeSave).toBool());
|
||||||
|
|
||||||
|
m_generalUi->backupFilePath->setText(config()->get(Config::BackupFilePathPattern).toString());
|
||||||
|
|
||||||
m_generalUi->useAlternativeSaveCheckBox->setChecked(!config()->get(Config::UseAtomicSaves).toBool());
|
m_generalUi->useAlternativeSaveCheckBox->setChecked(!config()->get(Config::UseAtomicSaves).toBool());
|
||||||
m_generalUi->alternativeSaveComboBox->setCurrentIndex(config()->get(Config::UseDirectWriteSaves).toBool() ? 1 : 0);
|
m_generalUi->alternativeSaveComboBox->setCurrentIndex(config()->get(Config::UseDirectWriteSaves).toBool() ? 1 : 0);
|
||||||
m_generalUi->autoReloadOnChangeCheckBox->setChecked(config()->get(Config::AutoReloadOnChange).toBool());
|
m_generalUi->autoReloadOnChangeCheckBox->setChecked(config()->get(Config::AutoReloadOnChange).toBool());
|
||||||
@ -326,6 +338,9 @@ void ApplicationSettingsWidget::saveSettings()
|
|||||||
config()->set(Config::AutoSaveOnExit, m_generalUi->autoSaveOnExitCheckBox->isChecked());
|
config()->set(Config::AutoSaveOnExit, m_generalUi->autoSaveOnExitCheckBox->isChecked());
|
||||||
config()->set(Config::AutoSaveNonDataChanges, m_generalUi->autoSaveNonDataChangesCheckBox->isChecked());
|
config()->set(Config::AutoSaveNonDataChanges, m_generalUi->autoSaveNonDataChangesCheckBox->isChecked());
|
||||||
config()->set(Config::BackupBeforeSave, m_generalUi->backupBeforeSaveCheckBox->isChecked());
|
config()->set(Config::BackupBeforeSave, m_generalUi->backupBeforeSaveCheckBox->isChecked());
|
||||||
|
|
||||||
|
config()->set(Config::BackupFilePathPattern, m_generalUi->backupFilePath->text());
|
||||||
|
|
||||||
config()->set(Config::UseAtomicSaves, !m_generalUi->useAlternativeSaveCheckBox->isChecked());
|
config()->set(Config::UseAtomicSaves, !m_generalUi->useAlternativeSaveCheckBox->isChecked());
|
||||||
config()->set(Config::UseDirectWriteSaves, m_generalUi->alternativeSaveComboBox->currentIndex() == 1);
|
config()->set(Config::UseDirectWriteSaves, m_generalUi->alternativeSaveComboBox->currentIndex() == 1);
|
||||||
config()->set(Config::AutoReloadOnChange, m_generalUi->autoReloadOnChangeCheckBox->isChecked());
|
config()->set(Config::AutoReloadOnChange, m_generalUi->autoReloadOnChangeCheckBox->isChecked());
|
||||||
@ -504,3 +519,13 @@ void ApplicationSettingsWidget::checkUpdatesToggled(bool checked)
|
|||||||
{
|
{
|
||||||
m_generalUi->checkForUpdatesIncludeBetasCheckBox->setEnabled(checked);
|
m_generalUi->checkForUpdatesIncludeBetasCheckBox->setEnabled(checked);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ApplicationSettingsWidget::selectBackupDirectory()
|
||||||
|
{
|
||||||
|
auto backupDirectory =
|
||||||
|
FileDialog::instance()->getExistingDirectory(this, tr("Select backup storage directory"), QDir::homePath());
|
||||||
|
if (!backupDirectory.isEmpty()) {
|
||||||
|
m_generalUi->backupFilePath->setText(
|
||||||
|
QDir(backupDirectory).filePath(config()->getDefault(Config::BackupFilePathPattern).toString()));
|
||||||
|
}
|
||||||
|
}
|
@ -62,6 +62,7 @@ private slots:
|
|||||||
void systrayToggled(bool checked);
|
void systrayToggled(bool checked);
|
||||||
void rememberDatabasesToggled(bool checked);
|
void rememberDatabasesToggled(bool checked);
|
||||||
void checkUpdatesToggled(bool checked);
|
void checkUpdatesToggled(bool checked);
|
||||||
|
void selectBackupDirectory();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QWidget* const m_secWidget;
|
QWidget* const m_secWidget;
|
||||||
|
@ -58,8 +58,8 @@
|
|||||||
<rect>
|
<rect>
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>581</width>
|
<width>664</width>
|
||||||
<height>924</height>
|
<height>1215</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout_8">
|
<layout class="QVBoxLayout" name="verticalLayout_8">
|
||||||
@ -278,6 +278,52 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<layout class="QHBoxLayout" name="horizontalLayout_4">
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="label">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Minimum" vsizetype="Preferred">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string>Backup destination</string>
|
||||||
|
</property>
|
||||||
|
<property name="buddy">
|
||||||
|
<cstring>backupFilePath</cstring>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QLineEdit" name="backupFilePath">
|
||||||
|
<property name="enabled">
|
||||||
|
<bool>false</bool>
|
||||||
|
</property>
|
||||||
|
<property name="toolTip">
|
||||||
|
<string>Specifies the database backup file location. Occurences of "{DB_FILENAME}" are replaced with the filename of the saved database without extension. {TIME:<format>} is replaced with the backup time, see https://doc.qt.io/qt-5/qdatetime.html#toString. <format> defaults to format string "dd_MM_yyyy_hh-mm-ss".</string>
|
||||||
|
</property>
|
||||||
|
<property name="placeholderText">
|
||||||
|
<string>{DB_FILENAME}.old.kdbx</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="backupFilePathPicker">
|
||||||
|
<property name="enabled">
|
||||||
|
<bool>false</bool>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string>Choose...</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<layout class="QHBoxLayout" name="horizontalLayout_5"/>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QCheckBox" name="useAlternativeSaveCheckBox">
|
<widget class="QCheckBox" name="useAlternativeSaveCheckBox">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
#include <QProcess>
|
#include <QProcess>
|
||||||
#include <QSplitter>
|
#include <QSplitter>
|
||||||
#include <QTextEdit>
|
#include <QTextEdit>
|
||||||
|
#include <core/Tools.h>
|
||||||
|
|
||||||
#include "autotype/AutoType.h"
|
#include "autotype/AutoType.h"
|
||||||
#include "core/EntrySearcher.h"
|
#include "core/EntrySearcher.h"
|
||||||
@ -1879,11 +1880,31 @@ bool DatabaseWidget::performSave(QString& errorMessage, const QString& fileName)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString backupFilePath;
|
||||||
|
if (config()->get(Config::BackupBeforeSave).toBool()) {
|
||||||
|
backupFilePath = config()->get(Config::BackupFilePathPattern).toString();
|
||||||
|
// Fall back to default
|
||||||
|
if (backupFilePath.isEmpty()) {
|
||||||
|
backupFilePath = config()->getDefault(Config::BackupFilePathPattern).toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
QFileInfo dbFileInfo(m_db->filePath());
|
||||||
|
backupFilePath = Tools::substituteBackupFilePath(backupFilePath, dbFileInfo.canonicalFilePath());
|
||||||
|
if (!backupFilePath.isNull()) {
|
||||||
|
// Note that we cannot guarantee that backupFilePath is actually a valid filename. QT currently provides
|
||||||
|
// no function for this. Moreover, we don't check if backupFilePath is a file and not a directory.
|
||||||
|
// If this isn't the case, just let the backup fail.
|
||||||
|
if (QDir::isRelativePath(backupFilePath)) {
|
||||||
|
backupFilePath = QDir::cleanPath(dbFileInfo.absolutePath() + QDir::separator() + backupFilePath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool ok;
|
bool ok;
|
||||||
if (fileName.isEmpty()) {
|
if (fileName.isEmpty()) {
|
||||||
ok = m_db->save(saveAction, config()->get(Config::BackupBeforeSave).toBool(), &errorMessage);
|
ok = m_db->save(saveAction, backupFilePath, &errorMessage);
|
||||||
} else {
|
} else {
|
||||||
ok = m_db->saveAs(fileName, saveAction, config()->get(Config::BackupBeforeSave).toBool(), &errorMessage);
|
ok = m_db->saveAs(fileName, saveAction, backupFilePath, &errorMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return control
|
// Return control
|
||||||
|
@ -75,29 +75,26 @@ void TestDatabase::testSave()
|
|||||||
// Test safe saves
|
// Test safe saves
|
||||||
db->metadata()->setName("test");
|
db->metadata()->setName("test");
|
||||||
QVERIFY(db->isModified());
|
QVERIFY(db->isModified());
|
||||||
QVERIFY2(db->save(Database::Atomic, false, &error), error.toLatin1());
|
QVERIFY2(db->save(Database::Atomic, QString(), &error), error.toLatin1());
|
||||||
QVERIFY(!db->isModified());
|
QVERIFY(!db->isModified());
|
||||||
|
|
||||||
// Test temp-file saves
|
// Test temp-file saves
|
||||||
db->metadata()->setName("test2");
|
db->metadata()->setName("test2");
|
||||||
QVERIFY2(db->save(Database::TempFile, false, &error), error.toLatin1());
|
QVERIFY2(db->save(Database::TempFile, QString(), &error), error.toLatin1());
|
||||||
QVERIFY(!db->isModified());
|
QVERIFY(!db->isModified());
|
||||||
|
|
||||||
// Test direct-write saves
|
// Test direct-write saves
|
||||||
db->metadata()->setName("test3");
|
db->metadata()->setName("test3");
|
||||||
QVERIFY2(db->save(Database::DirectWrite, false, &error), error.toLatin1());
|
QVERIFY2(db->save(Database::DirectWrite, QString(), &error), error.toLatin1());
|
||||||
QVERIFY(!db->isModified());
|
QVERIFY(!db->isModified());
|
||||||
|
|
||||||
// Test save backups
|
// Test save backups
|
||||||
|
TemporaryFile backupFile;
|
||||||
|
auto backupFilePath = backupFile.fileName();
|
||||||
db->metadata()->setName("test4");
|
db->metadata()->setName("test4");
|
||||||
QVERIFY2(db->save(Database::Atomic, true, &error), error.toLatin1());
|
QVERIFY2(db->save(Database::Atomic, backupFilePath, &error), error.toLatin1());
|
||||||
QVERIFY(!db->isModified());
|
QVERIFY(!db->isModified());
|
||||||
|
|
||||||
// Confirm backup exists and then delete it
|
|
||||||
auto re = QRegularExpression("(\\.[^.]+)$");
|
|
||||||
auto match = re.match(tempFile.fileName());
|
|
||||||
auto backupFilePath = tempFile.fileName();
|
|
||||||
backupFilePath = backupFilePath.replace(re, "") + ".old" + match.captured(1);
|
|
||||||
QVERIFY(QFile::exists(backupFilePath));
|
QVERIFY(QFile::exists(backupFilePath));
|
||||||
QFile::remove(backupFilePath);
|
QFile::remove(backupFilePath);
|
||||||
QVERIFY(!QFile::exists(backupFilePath));
|
QVERIFY(!QFile::exists(backupFilePath));
|
||||||
@ -123,7 +120,7 @@ void TestDatabase::testSignals()
|
|||||||
QTRY_COMPARE(spyModified.count(), 1);
|
QTRY_COMPARE(spyModified.count(), 1);
|
||||||
|
|
||||||
QSignalSpy spySaved(db.data(), SIGNAL(databaseSaved()));
|
QSignalSpy spySaved(db.data(), SIGNAL(databaseSaved()));
|
||||||
QVERIFY(db->save(Database::Atomic, false, &error));
|
QVERIFY(db->save(Database::Atomic, QString(), &error));
|
||||||
QCOMPARE(spySaved.count(), 1);
|
QCOMPARE(spySaved.count(), 1);
|
||||||
|
|
||||||
// Short delay to allow file system settling to reduce test failures
|
// Short delay to allow file system settling to reduce test failures
|
||||||
|
@ -17,6 +17,8 @@
|
|||||||
|
|
||||||
#include "TestTools.h"
|
#include "TestTools.h"
|
||||||
|
|
||||||
|
#include "core/Clock.h"
|
||||||
|
|
||||||
#include <QTest>
|
#include <QTest>
|
||||||
|
|
||||||
QTEST_GUILESS_MAIN(TestTools)
|
QTEST_GUILESS_MAIN(TestTools)
|
||||||
@ -108,3 +110,55 @@ void TestTools::testValidUuid()
|
|||||||
QVERIFY(!Tools::isValidUuid(longUuid));
|
QVERIFY(!Tools::isValidUuid(longUuid));
|
||||||
QVERIFY(!Tools::isValidUuid(nonHexUuid));
|
QVERIFY(!Tools::isValidUuid(nonHexUuid));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestTools::testBackupFilePatternSubstitution_data()
|
||||||
|
{
|
||||||
|
QTest::addColumn<QString>("pattern");
|
||||||
|
QTest::addColumn<QString>("dbFilePath");
|
||||||
|
QTest::addColumn<QString>("expectedSubstitution");
|
||||||
|
|
||||||
|
static const auto DEFAULT_DB_FILE_NAME = QStringLiteral("KeePassXC");
|
||||||
|
static const auto DEFAULT_DB_FILE_PATH = QStringLiteral("/tmp/") + DEFAULT_DB_FILE_NAME + QStringLiteral(".kdbx");
|
||||||
|
static const auto NOW = Clock::currentDateTime();
|
||||||
|
auto DEFAULT_FORMATTED_TIME = NOW.toString("dd_MM_yyyy_hh-mm-ss");
|
||||||
|
|
||||||
|
QTest::newRow("Null pattern") << QString() << DEFAULT_DB_FILE_PATH << QString();
|
||||||
|
QTest::newRow("Empty pattern") << QString("") << DEFAULT_DB_FILE_PATH << QString("");
|
||||||
|
QTest::newRow("Null database path") << "valid_pattern" << QString() << QString();
|
||||||
|
QTest::newRow("Empty database path") << "valid_pattern" << QString("") << QString();
|
||||||
|
QTest::newRow("Unclosed/invalid pattern") << "{DB_FILENAME" << DEFAULT_DB_FILE_PATH << "{DB_FILENAME";
|
||||||
|
QTest::newRow("Unknown pattern") << "{NO_MATCH}" << DEFAULT_DB_FILE_PATH << "{NO_MATCH}";
|
||||||
|
QTest::newRow("Do not replace escaped patterns (filename)")
|
||||||
|
<< "\\{DB_FILENAME\\}" << DEFAULT_DB_FILE_PATH << "{DB_FILENAME}";
|
||||||
|
QTest::newRow("Do not replace escaped patterns (time)")
|
||||||
|
<< "\\{TIME:dd.MM.yyyy\\}" << DEFAULT_DB_FILE_PATH << "{TIME:dd.MM.yyyy}";
|
||||||
|
QTest::newRow("Multiple patterns should be replaced")
|
||||||
|
<< "{DB_FILENAME} {TIME} {DB_FILENAME}" << DEFAULT_DB_FILE_PATH
|
||||||
|
<< DEFAULT_DB_FILE_NAME + QStringLiteral(" ") + DEFAULT_FORMATTED_TIME + QStringLiteral(" ")
|
||||||
|
+ DEFAULT_DB_FILE_NAME;
|
||||||
|
QTest::newRow("Default time pattern") << "{TIME}" << DEFAULT_DB_FILE_PATH << DEFAULT_FORMATTED_TIME;
|
||||||
|
QTest::newRow("Default time pattern (empty formatter)")
|
||||||
|
<< "{TIME:}" << DEFAULT_DB_FILE_PATH << DEFAULT_FORMATTED_TIME;
|
||||||
|
QTest::newRow("Custom time pattern") << "{TIME:dd-ss}" << DEFAULT_DB_FILE_PATH << NOW.toString("dd-ss");
|
||||||
|
QTest::newRow("Invalid custom time pattern") << "{TIME:dd/-ss}" << DEFAULT_DB_FILE_PATH << NOW.toString("dd/-ss");
|
||||||
|
QTest::newRow("Recursive substitution") << "{TIME:'{TIME}'}" << DEFAULT_DB_FILE_PATH << DEFAULT_FORMATTED_TIME;
|
||||||
|
QTest::newRow("{DB_FILENAME} substitution")
|
||||||
|
<< "some {DB_FILENAME} thing" << DEFAULT_DB_FILE_PATH
|
||||||
|
<< QStringLiteral("some ") + DEFAULT_DB_FILE_NAME + QStringLiteral(" thing");
|
||||||
|
QTest::newRow("{DB_FILENAME} substitution with multiple extensions") << "some {DB_FILENAME} thing"
|
||||||
|
<< "/tmp/KeePassXC.kdbx.ext"
|
||||||
|
<< "some KeePassXC.kdbx thing";
|
||||||
|
// Not relevant right now, added test anyway
|
||||||
|
QTest::newRow("There should be no substitution loops") << "{DB_FILENAME}"
|
||||||
|
<< "{TIME:'{DB_FILENAME}'}.ext"
|
||||||
|
<< "{DB_FILENAME}";
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestTools::testBackupFilePatternSubstitution()
|
||||||
|
{
|
||||||
|
QFETCH(QString, pattern);
|
||||||
|
QFETCH(QString, dbFilePath);
|
||||||
|
QFETCH(QString, expectedSubstitution);
|
||||||
|
|
||||||
|
QCOMPARE(Tools::substituteBackupFilePath(pattern, dbFilePath), expectedSubstitution);
|
||||||
|
}
|
||||||
|
@ -29,6 +29,8 @@ private slots:
|
|||||||
void testIsBase64();
|
void testIsBase64();
|
||||||
void testEnvSubstitute();
|
void testEnvSubstitute();
|
||||||
void testValidUuid();
|
void testValidUuid();
|
||||||
|
void testBackupFilePatternSubstitution_data();
|
||||||
|
void testBackupFilePatternSubstitution();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // KEEPASSX_TESTTOOLS_H
|
#endif // KEEPASSX_TESTTOOLS_H
|
||||||
|
@ -82,22 +82,10 @@ int main(int argc, char* argv[])
|
|||||||
return QTest::qExec(&tc, argc, argv);
|
return QTest::qExec(&tc, argc, argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
static QString dbFileName = QStringLiteral(KEEPASSX_TEST_DATA_DIR).append("/NewDatabase.kdbx");
|
|
||||||
|
|
||||||
void TestGui::initTestCase()
|
void TestGui::initTestCase()
|
||||||
{
|
{
|
||||||
QVERIFY(Crypto::init());
|
QVERIFY(Crypto::init());
|
||||||
Config::createTempFileInstance();
|
Config::createTempFileInstance();
|
||||||
// Disable autosave so we can test the modified file indicator
|
|
||||||
config()->set(Config::AutoSaveAfterEveryChange, false);
|
|
||||||
config()->set(Config::AutoSaveOnExit, false);
|
|
||||||
// Enable the tray icon so we can test hiding/restoring the windowQByteArray
|
|
||||||
config()->set(Config::GUI_ShowTrayIcon, true);
|
|
||||||
// Disable advanced settings mode (activate within individual tests to test advanced settings)
|
|
||||||
config()->set(Config::GUI_AdvancedSettings, false);
|
|
||||||
// Disable the update check first time alert
|
|
||||||
config()->set(Config::UpdateCheckMessageShown, true);
|
|
||||||
|
|
||||||
Application::bootstrap();
|
Application::bootstrap();
|
||||||
|
|
||||||
m_mainWindow.reset(new MainWindow());
|
m_mainWindow.reset(new MainWindow());
|
||||||
@ -106,11 +94,22 @@ void TestGui::initTestCase()
|
|||||||
m_mainWindow->resize(1024, 768);
|
m_mainWindow->resize(1024, 768);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Every test starts with opening the temp database
|
// Every test starts with resetting config settings and opening the temp database
|
||||||
void TestGui::init()
|
void TestGui::init()
|
||||||
{
|
{
|
||||||
|
// Reset config to defaults
|
||||||
|
config()->resetToDefaults();
|
||||||
|
// Disable autosave so we can test the modified file indicator
|
||||||
|
config()->set(Config::AutoSaveAfterEveryChange, false);
|
||||||
|
config()->set(Config::AutoSaveOnExit, false);
|
||||||
|
// Enable the tray icon so we can test hiding/restoring the windowQByteArray
|
||||||
|
config()->set(Config::GUI_ShowTrayIcon, true);
|
||||||
|
// Disable the update check first time alert
|
||||||
|
config()->set(Config::UpdateCheckMessageShown, true);
|
||||||
|
|
||||||
// Copy the test database file to the temporary file
|
// Copy the test database file to the temporary file
|
||||||
QVERIFY(m_dbFile.copyFromFile(dbFileName));
|
auto origFilePath = QDir(KEEPASSX_TEST_DATA_DIR).absoluteFilePath("NewDatabase.kdbx");
|
||||||
|
QVERIFY(m_dbFile.copyFromFile(origFilePath));
|
||||||
|
|
||||||
m_dbFileName = QFileInfo(m_dbFile.fileName()).fileName();
|
m_dbFileName = QFileInfo(m_dbFile.fileName()).fileName();
|
||||||
m_dbFilePath = m_dbFile.fileName();
|
m_dbFilePath = m_dbFile.fileName();
|
||||||
@ -354,6 +353,8 @@ void TestGui::testAutoreloadDatabase()
|
|||||||
cleanup();
|
cleanup();
|
||||||
init();
|
init();
|
||||||
|
|
||||||
|
config()->set(Config::AutoReloadOnChange, false);
|
||||||
|
|
||||||
// Test rejecting new file in autoreload
|
// Test rejecting new file in autoreload
|
||||||
MessageBox::setNextAnswer(MessageBox::No);
|
MessageBox::setNextAnswer(MessageBox::No);
|
||||||
// Overwrite the current database with the temp data
|
// Overwrite the current database with the temp data
|
||||||
@ -1277,11 +1278,76 @@ void TestGui::testSave()
|
|||||||
QTRY_COMPARE(m_tabWidget->tabText(m_tabWidget->currentIndex()), QString("testSave*"));
|
QTRY_COMPARE(m_tabWidget->tabText(m_tabWidget->currentIndex()), QString("testSave*"));
|
||||||
|
|
||||||
triggerAction("actionDatabaseSave");
|
triggerAction("actionDatabaseSave");
|
||||||
QCOMPARE(m_tabWidget->tabName(m_tabWidget->currentIndex()), QString("testSave"));
|
QTRY_COMPARE(m_tabWidget->tabName(m_tabWidget->currentIndex()), QString("testSave"));
|
||||||
|
|
||||||
checkDatabase();
|
checkDatabase();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestGui::testSaveBackupPath_data()
|
||||||
|
{
|
||||||
|
QTest::addColumn<QString>("backupFilePathPattern");
|
||||||
|
QTest::addColumn<QString>("expectedBackupFile");
|
||||||
|
|
||||||
|
// Absolute paths should remain absolute
|
||||||
|
TemporaryFile tmpFile;
|
||||||
|
QVERIFY(tmpFile.open());
|
||||||
|
tmpFile.remove();
|
||||||
|
|
||||||
|
QTest::newRow("Absolute backup path") << tmpFile.fileName() << tmpFile.fileName();
|
||||||
|
// relative paths should be resolved to database parent directory
|
||||||
|
QTest::newRow("Relative backup path (implicit)") << "other_dir/test.old.kdbx"
|
||||||
|
<< "other_dir/test.old.kdbx";
|
||||||
|
QTest::newRow("Relative backup path (explicit)") << "./other_dir2/test2.old.kdbx"
|
||||||
|
<< "other_dir2/test2.old.kdbx";
|
||||||
|
|
||||||
|
QTest::newRow("Path with placeholders") << "{DB_FILENAME}.old.kdbx"
|
||||||
|
<< "KeePassXC.old.kdbx";
|
||||||
|
// empty path should be replaced with default pattern
|
||||||
|
QTest::newRow("Empty path") << QString("") << config()->getDefault(Config::BackupFilePathPattern).toString();
|
||||||
|
// {DB_FILENAME} should be replaced with database filename
|
||||||
|
QTest::newRow("") << "{DB_FILENAME}_.old.kdbx"
|
||||||
|
<< "{DB_FILENAME}_.old.kdbx";
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestGui::testSaveBackupPath()
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Tests that the backupFilePathPattern config entry is respected. We do not test patterns like {TIME} etc here
|
||||||
|
* as this is done in a separate test case. We do however check {DB_FILENAME} as this is a feature of the
|
||||||
|
* performBackup() function.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Get test data
|
||||||
|
QFETCH(QString, backupFilePathPattern);
|
||||||
|
QFETCH(QString, expectedBackupFile);
|
||||||
|
|
||||||
|
// Enable automatic backups
|
||||||
|
config()->set(Config::BackupBeforeSave, true);
|
||||||
|
config()->set(Config::BackupFilePathPattern, backupFilePathPattern);
|
||||||
|
|
||||||
|
// Replace placeholders and resolve relative paths. This cannot be done in the _data() function as the
|
||||||
|
// db path/filename is not known yet
|
||||||
|
auto dbFileInfo = QFileInfo(m_dbFilePath);
|
||||||
|
if (!QDir::isAbsolutePath(expectedBackupFile)) {
|
||||||
|
expectedBackupFile = QDir(dbFileInfo.absolutePath()).absoluteFilePath(expectedBackupFile);
|
||||||
|
}
|
||||||
|
expectedBackupFile.replace("{DB_FILENAME}", dbFileInfo.completeBaseName());
|
||||||
|
|
||||||
|
// Save a modified database
|
||||||
|
auto prevName = m_db->metadata()->name();
|
||||||
|
m_db->metadata()->setName("testBackupPathPattern");
|
||||||
|
|
||||||
|
QTRY_COMPARE(m_tabWidget->tabText(m_tabWidget->currentIndex()), QString("testBackupPathPattern*"));
|
||||||
|
triggerAction("actionDatabaseSave");
|
||||||
|
QTRY_COMPARE(m_tabWidget->tabText(m_tabWidget->currentIndex()), QString("testBackupPathPattern"));
|
||||||
|
|
||||||
|
// Test that the backup file has the previous database name
|
||||||
|
checkDatabase(expectedBackupFile, prevName);
|
||||||
|
|
||||||
|
// Clean up
|
||||||
|
QFile(expectedBackupFile).remove();
|
||||||
|
}
|
||||||
|
|
||||||
void TestGui::testDatabaseSettings()
|
void TestGui::testDatabaseSettings()
|
||||||
{
|
{
|
||||||
m_db->metadata()->setName("testDatabaseSettings");
|
m_db->metadata()->setName("testDatabaseSettings");
|
||||||
@ -1301,7 +1367,7 @@ void TestGui::testDatabaseSettings()
|
|||||||
QCOMPARE(m_db->kdf()->rounds(), 123456);
|
QCOMPARE(m_db->kdf()->rounds(), 123456);
|
||||||
|
|
||||||
triggerAction("actionDatabaseSave");
|
triggerAction("actionDatabaseSave");
|
||||||
QCOMPARE(m_tabWidget->tabText(m_tabWidget->currentIndex()), QString("testDatabaseSettings"));
|
QTRY_COMPARE(m_tabWidget->tabText(m_tabWidget->currentIndex()), QString("testDatabaseSettings"));
|
||||||
|
|
||||||
advancedToggle->setChecked(false);
|
advancedToggle->setChecked(false);
|
||||||
QApplication::processEvents();
|
QApplication::processEvents();
|
||||||
@ -1365,6 +1431,8 @@ void TestGui::testDragAndDropKdbxFiles()
|
|||||||
const int openedDatabasesCount = m_tabWidget->count();
|
const int openedDatabasesCount = m_tabWidget->count();
|
||||||
|
|
||||||
const QString badDatabaseFilePath(QString(KEEPASSX_TEST_DATA_DIR).append("/NotDatabase.notkdbx"));
|
const QString badDatabaseFilePath(QString(KEEPASSX_TEST_DATA_DIR).append("/NotDatabase.notkdbx"));
|
||||||
|
const QString goodDatabaseFilePath(QString(KEEPASSX_TEST_DATA_DIR).append("/NewDatabase.kdbx"));
|
||||||
|
|
||||||
QMimeData badMimeData;
|
QMimeData badMimeData;
|
||||||
badMimeData.setUrls({QUrl::fromLocalFile(badDatabaseFilePath)});
|
badMimeData.setUrls({QUrl::fromLocalFile(badDatabaseFilePath)});
|
||||||
QDragEnterEvent badDragEvent(QPoint(1, 1), Qt::LinkAction, &badMimeData, Qt::LeftButton, Qt::NoModifier);
|
QDragEnterEvent badDragEvent(QPoint(1, 1), Qt::LinkAction, &badMimeData, Qt::LeftButton, Qt::NoModifier);
|
||||||
@ -1378,7 +1446,7 @@ void TestGui::testDragAndDropKdbxFiles()
|
|||||||
QCOMPARE(m_tabWidget->count(), openedDatabasesCount);
|
QCOMPARE(m_tabWidget->count(), openedDatabasesCount);
|
||||||
|
|
||||||
QMimeData goodMimeData;
|
QMimeData goodMimeData;
|
||||||
goodMimeData.setUrls({QUrl::fromLocalFile(dbFileName)});
|
goodMimeData.setUrls({QUrl::fromLocalFile(goodDatabaseFilePath)});
|
||||||
QDragEnterEvent goodDragEvent(QPoint(1, 1), Qt::LinkAction, &goodMimeData, Qt::LeftButton, Qt::NoModifier);
|
QDragEnterEvent goodDragEvent(QPoint(1, 1), Qt::LinkAction, &goodMimeData, Qt::LeftButton, Qt::NoModifier);
|
||||||
qApp->notify(m_mainWindow.data(), &goodDragEvent);
|
qApp->notify(m_mainWindow.data(), &goodDragEvent);
|
||||||
QCOMPARE(goodDragEvent.isAccepted(), true);
|
QCOMPARE(goodDragEvent.isAccepted(), true);
|
||||||
@ -1700,17 +1768,18 @@ void TestGui::addCannedEntries()
|
|||||||
QTest::mouseClick(editEntryWidgetButtonBox->button(QDialogButtonBox::Ok), Qt::LeftButton);
|
QTest::mouseClick(editEntryWidgetButtonBox->button(QDialogButtonBox::Ok), Qt::LeftButton);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestGui::checkDatabase(QString dbFileName)
|
void TestGui::checkDatabase(const QString& filePath, const QString& expectedDbName)
|
||||||
{
|
{
|
||||||
if (dbFileName.isEmpty()) {
|
|
||||||
dbFileName = m_dbFilePath;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto key = QSharedPointer<CompositeKey>::create();
|
auto key = QSharedPointer<CompositeKey>::create();
|
||||||
key->addKey(QSharedPointer<PasswordKey>::create("a"));
|
key->addKey(QSharedPointer<PasswordKey>::create("a"));
|
||||||
auto dbSaved = QSharedPointer<Database>::create();
|
auto dbSaved = QSharedPointer<Database>::create();
|
||||||
QVERIFY(dbSaved->open(dbFileName, key, nullptr, false));
|
QVERIFY(dbSaved->open(filePath, key, nullptr, false));
|
||||||
QCOMPARE(dbSaved->metadata()->name(), m_db->metadata()->name());
|
QCOMPARE(dbSaved->metadata()->name(), expectedDbName);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestGui::checkDatabase(const QString& filePath)
|
||||||
|
{
|
||||||
|
checkDatabase(filePath.isEmpty() ? m_dbFilePath : filePath, m_db->metadata()->name());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestGui::triggerAction(const QString& name)
|
void TestGui::triggerAction(const QString& name)
|
||||||
|
@ -57,6 +57,8 @@ private slots:
|
|||||||
void testSaveAs();
|
void testSaveAs();
|
||||||
void testSaveBackup();
|
void testSaveBackup();
|
||||||
void testSave();
|
void testSave();
|
||||||
|
void testSaveBackupPath();
|
||||||
|
void testSaveBackupPath_data();
|
||||||
void testDatabaseSettings();
|
void testDatabaseSettings();
|
||||||
void testKeePass1Import();
|
void testKeePass1Import();
|
||||||
void testDatabaseLocking();
|
void testDatabaseLocking();
|
||||||
@ -67,7 +69,8 @@ private slots:
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
void addCannedEntries();
|
void addCannedEntries();
|
||||||
void checkDatabase(QString dbFileName = "");
|
void checkDatabase(const QString& filePath, const QString& expectedDbName);
|
||||||
|
void checkDatabase(const QString& filePath = {});
|
||||||
void triggerAction(const QString& name);
|
void triggerAction(const QString& name);
|
||||||
void dragAndDropGroup(const QModelIndex& sourceIndex,
|
void dragAndDropGroup(const QModelIndex& sourceIndex,
|
||||||
const QModelIndex& targetIndex,
|
const QModelIndex& targetIndex,
|
||||||
|
Loading…
Reference in New Issue
Block a user