mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-04-03 20:47:37 +03:00
Support remote database access using external tools (#7222)
* Provide remote database sync capability Allow arbitrary commands to be defined and executed for syncing databases with remote services. This includes sftp, scp, rsync, etc. Remote commands are stored per-database and sync operations are manually triggered by the user from the Database -> Remote Sync menu. --------- Co-authored-by: Stefan Forstenlechner <t-h-e@users.noreply.github.com> Co-authored-by: Jonathan White <support@dmapps.us>
This commit is contained in:
parent
ad8a00d56b
commit
1ca607792d
38 changed files with 1780 additions and 21 deletions
BIN
tests/data/SyncDatabase.kdbx
Normal file
BIN
tests/data/SyncDatabase.kdbx
Normal file
Binary file not shown.
BIN
tests/data/SyncDatabaseDifferentPassword.kdbx
Normal file
BIN
tests/data/SyncDatabaseDifferentPassword.kdbx
Normal file
Binary file not shown.
|
@ -15,7 +15,7 @@
|
|||
|
||||
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/..)
|
||||
|
||||
add_unit_test(NAME testgui SOURCES TestGui.cpp ../util/TemporaryFile.cpp LIBS ${TEST_LIBRARIES})
|
||||
add_unit_test(NAME testgui SOURCES TestGui.cpp ../util/TemporaryFile.cpp ../mock/MockRemoteProcess.cpp LIBS ${TEST_LIBRARIES})
|
||||
add_unit_test(NAME testguipixmaps SOURCES TestGuiPixmaps.cpp LIBS ${TEST_LIBRARIES})
|
||||
|
||||
if(WITH_XC_BROWSER)
|
||||
|
|
|
@ -21,6 +21,9 @@
|
|||
|
||||
#include <QCheckBox>
|
||||
#include <QClipboard>
|
||||
#include <QListWidget>
|
||||
#include <QMenu>
|
||||
#include <QMenuBar>
|
||||
#include <QMimeData>
|
||||
#include <QPlainTextEdit>
|
||||
#include <QPushButton>
|
||||
|
@ -56,9 +59,11 @@
|
|||
#include "gui/group/EditGroupWidget.h"
|
||||
#include "gui/group/GroupModel.h"
|
||||
#include "gui/group/GroupView.h"
|
||||
#include "gui/remote/RemoteHandler.h"
|
||||
#include "gui/tag/TagsEdit.h"
|
||||
#include "gui/wizard/NewDatabaseWizard.h"
|
||||
#include "keys/FileKey.h"
|
||||
#include "mock/MockRemoteProcess.h"
|
||||
|
||||
#define TEST_MODAL_NO_WAIT(TEST_CODE) \
|
||||
bool dialogFinished = false; \
|
||||
|
@ -370,6 +375,107 @@ void TestGui::testMergeDatabase()
|
|||
QCOMPARE(m_db->rootGroup()->findChildByName("General")->entries().size(), 1);
|
||||
}
|
||||
|
||||
void TestGui::prepareAndTriggerRemoteSync(const QString& sourceToSync)
|
||||
{
|
||||
auto* menuRemoteSync = m_mainWindow->findChild<QMenu*>("menuRemoteSync");
|
||||
QSignalSpy remoteAboutToShow(menuRemoteSync, &QMenu::aboutToShow);
|
||||
QApplication::processEvents();
|
||||
|
||||
// create remote settings in settings dialog
|
||||
triggerAction("actionDatabaseSettings");
|
||||
auto* dbSettingsDialog = m_dbWidget->findChild<QWidget*>("databaseSettingsDialog");
|
||||
auto* dbSettingsCategoryList = dbSettingsDialog->findChild<CategoryListWidget*>("categoryList");
|
||||
auto* dbSettingsStackedWidget = dbSettingsDialog->findChild<QStackedWidget*>("stackedWidget");
|
||||
dbSettingsCategoryList->setCurrentCategory(2); // go into remote category
|
||||
auto name = "testCommand";
|
||||
auto* nameEdit = dbSettingsStackedWidget->findChild<QLineEdit*>("nameLineEdit");
|
||||
auto* downloadCommandEdit = dbSettingsStackedWidget->findChild<QLineEdit*>("downloadCommand");
|
||||
QVERIFY(downloadCommandEdit != nullptr);
|
||||
downloadCommandEdit->setText(sourceToSync);
|
||||
nameEdit->setText(name);
|
||||
auto* saveSettingsButton = dbSettingsStackedWidget->findChild<QPushButton*>("saveSettingsButton");
|
||||
QVERIFY(saveSettingsButton != nullptr);
|
||||
QTest::mouseClick(saveSettingsButton, Qt::LeftButton);
|
||||
|
||||
// find and click dialog OK button
|
||||
auto buttons = dbSettingsDialog->findChild<QDialogButtonBox*>()->findChildren<QPushButton*>();
|
||||
for (QPushButton* b : buttons) {
|
||||
if (b->text() == "OK") {
|
||||
QTest::mouseClick(b, Qt::LeftButton);
|
||||
break;
|
||||
}
|
||||
}
|
||||
QTRY_COMPARE(m_dbWidget->getRemoteParams().size(), 1);
|
||||
|
||||
// trigger aboutToShow to create remote actions
|
||||
menuRemoteSync->popup(QPoint(0, 0));
|
||||
QApplication::processEvents();
|
||||
QTRY_COMPARE(remoteAboutToShow.count(), 1);
|
||||
// close the opened menu
|
||||
QTest::keyClick(menuRemoteSync, Qt::Key::Key_Escape);
|
||||
|
||||
// trigger remote sync action
|
||||
for (auto* remoteAction : menuRemoteSync->actions()) {
|
||||
if (remoteAction->text() == name) {
|
||||
remoteAction->trigger();
|
||||
break;
|
||||
}
|
||||
}
|
||||
QApplication::processEvents();
|
||||
}
|
||||
|
||||
void TestGui::testRemoteSyncDatabaseSameKey()
|
||||
{
|
||||
QString sourceToSync = "sftp user@server:Database.kdbx";
|
||||
RemoteHandler::setRemoteProcessFunc([sourceToSync](QObject* parent) {
|
||||
return QScopedPointer<RemoteProcess>(
|
||||
new MockRemoteProcess(parent, QString(KEEPASSX_TEST_DATA_DIR).append("/SyncDatabase.kdbx")));
|
||||
});
|
||||
QSignalSpy dbSyncSpy(m_dbWidget.data(), &DatabaseWidget::databaseSyncCompleted);
|
||||
prepareAndTriggerRemoteSync(sourceToSync);
|
||||
QTRY_COMPARE(dbSyncSpy.count(), 1);
|
||||
|
||||
m_db = m_tabWidget->currentDatabaseWidget()->database();
|
||||
|
||||
// there are seven child groups of the root group
|
||||
QCOMPARE(m_db->rootGroup()->children().size(), 7);
|
||||
// the merged group should contain an entry
|
||||
QCOMPARE(m_db->rootGroup()->children().at(6)->entries().size(), 1);
|
||||
// the General group contains one entry merged from the other db
|
||||
QCOMPARE(m_db->rootGroup()->findChildByName("General")->entries().size(), 1);
|
||||
}
|
||||
|
||||
void TestGui::testRemoteSyncDatabaseRequiresPassword()
|
||||
{
|
||||
QString sourceToSync = "sftp user@server:Database.kdbx";
|
||||
RemoteHandler::setRemoteProcessFunc([sourceToSync](QObject* parent) {
|
||||
return QScopedPointer<RemoteProcess>(new MockRemoteProcess(
|
||||
parent, QString(KEEPASSX_TEST_DATA_DIR).append("/SyncDatabaseDifferentPassword.kdbx")));
|
||||
});
|
||||
QSignalSpy dbSyncSpy(m_dbWidget.data(), &DatabaseWidget::databaseSyncCompleted);
|
||||
prepareAndTriggerRemoteSync(sourceToSync);
|
||||
|
||||
// need to process more events as opening with the same key did not work and more events have been fired
|
||||
QApplication::processEvents(QEventLoop::WaitForMoreEvents);
|
||||
|
||||
QTRY_COMPARE(QApplication::focusWidget()->objectName(), QString("passwordEdit"));
|
||||
auto* editPasswordSync = QApplication::focusWidget();
|
||||
QVERIFY(editPasswordSync->isVisible());
|
||||
|
||||
QTest::keyClicks(editPasswordSync, "b");
|
||||
QTest::keyClick(editPasswordSync, Qt::Key_Enter);
|
||||
|
||||
QTRY_COMPARE(dbSyncSpy.count(), 1);
|
||||
m_db = m_tabWidget->currentDatabaseWidget()->database();
|
||||
|
||||
// there are seven child groups of the root group
|
||||
QCOMPARE(m_db->rootGroup()->children().size(), 7);
|
||||
// the merged group should contain an entry
|
||||
QCOMPARE(m_db->rootGroup()->children().at(6)->entries().size(), 1);
|
||||
// the General group contains one entry merged from the other db
|
||||
QCOMPARE(m_db->rootGroup()->findChildByName("General")->entries().size(), 1);
|
||||
}
|
||||
|
||||
void TestGui::testAutoreloadDatabase()
|
||||
{
|
||||
config()->set(Config::AutoReloadOnChange, false);
|
||||
|
|
|
@ -40,6 +40,8 @@ private slots:
|
|||
void testSettingsDefaultTabOrder();
|
||||
void testCreateDatabase();
|
||||
void testMergeDatabase();
|
||||
void testRemoteSyncDatabaseSameKey();
|
||||
void testRemoteSyncDatabaseRequiresPassword();
|
||||
void testAutoreloadDatabase();
|
||||
void testTabs();
|
||||
void testEditEntry();
|
||||
|
@ -85,6 +87,7 @@ private:
|
|||
Qt::KeyboardModifiers stateKey = 0);
|
||||
void checkSaveDatabase();
|
||||
void checkStatusBarText(const QString& textFragment);
|
||||
void prepareAndTriggerRemoteSync(const QString& sourceToSync);
|
||||
|
||||
QScopedPointer<MainWindow> m_mainWindow;
|
||||
QPointer<QLabel> m_statusBarLabel;
|
||||
|
|
56
tests/mock/MockRemoteProcess.cpp
Normal file
56
tests/mock/MockRemoteProcess.cpp
Normal file
|
@ -0,0 +1,56 @@
|
|||
/*
|
||||
* Copyright (C) 2023 KeePassXC Team <team@keepassxc.org>
|
||||
*
|
||||
* 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
|
||||
* the Free Software Foundation, either version 2 or (at your option)
|
||||
* version 3 of the License.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <QFile>
|
||||
|
||||
#include "MockRemoteProcess.h"
|
||||
|
||||
MockRemoteProcess::MockRemoteProcess(QObject* parent, const QString& dbPath)
|
||||
: RemoteProcess(parent)
|
||||
, m_dbPath(dbPath)
|
||||
{
|
||||
}
|
||||
|
||||
void MockRemoteProcess::start(const QString&)
|
||||
{
|
||||
QFile ::copy(m_dbPath, m_tempFileLocation);
|
||||
}
|
||||
|
||||
qint64 MockRemoteProcess::write(const QString& data)
|
||||
{
|
||||
return data.length();
|
||||
}
|
||||
|
||||
bool MockRemoteProcess::waitForBytesWritten()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
void MockRemoteProcess::closeWriteChannel()
|
||||
{
|
||||
// nothing to do
|
||||
}
|
||||
|
||||
bool MockRemoteProcess::waitForFinished(int)
|
||||
{
|
||||
return true; // no need to wait
|
||||
}
|
||||
|
||||
int MockRemoteProcess::exitCode() const
|
||||
{
|
||||
return 0; // always return success
|
||||
}
|
41
tests/mock/MockRemoteProcess.h
Normal file
41
tests/mock/MockRemoteProcess.h
Normal file
|
@ -0,0 +1,41 @@
|
|||
/*
|
||||
* Copyright (C) 2023 KeePassXC Team <team@keepassxc.org>
|
||||
*
|
||||
* 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
|
||||
* the Free Software Foundation, either version 2 or (at your option)
|
||||
* version 3 of the License.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef KEEPASSXC_MOCKREMOTEPROCESS_H
|
||||
#define KEEPASSXC_MOCKREMOTEPROCESS_H
|
||||
|
||||
#include "gui/remote/RemoteProcess.h"
|
||||
|
||||
class MockRemoteProcess : public RemoteProcess
|
||||
{
|
||||
public:
|
||||
explicit MockRemoteProcess(QObject* parent, const QString& dbPath);
|
||||
~MockRemoteProcess() override = default;
|
||||
|
||||
void start(const QString& program) override;
|
||||
qint64 write(const QString& data) override;
|
||||
bool waitForBytesWritten() override;
|
||||
void closeWriteChannel() override;
|
||||
bool waitForFinished(int msecs) override;
|
||||
[[nodiscard]] int exitCode() const override;
|
||||
|
||||
private:
|
||||
QByteArray m_data;
|
||||
QString m_dbPath;
|
||||
};
|
||||
|
||||
#endif // KEEPASSXC_MOCKREMOTEPROCESS_H
|
Loading…
Add table
Add a link
Reference in a new issue