mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-27 14:57:09 -05:00
Merge branch 'release/2.5.2' into develop
This commit is contained in:
commit
ed0b76813d
@ -68,6 +68,10 @@ BrowserOptionDialog::BrowserOptionDialog(QWidget* parent)
|
|||||||
connect(m_ui->useCustomProxy, SIGNAL(toggled(bool)), m_ui->customProxyLocationBrowseButton, SLOT(setEnabled(bool)));
|
connect(m_ui->useCustomProxy, SIGNAL(toggled(bool)), m_ui->customProxyLocationBrowseButton, SLOT(setEnabled(bool)));
|
||||||
connect(m_ui->customProxyLocationBrowseButton, SIGNAL(clicked()), this, SLOT(showProxyLocationFileDialog()));
|
connect(m_ui->customProxyLocationBrowseButton, SIGNAL(clicked()), this, SLOT(showProxyLocationFileDialog()));
|
||||||
|
|
||||||
|
#ifndef Q_OS_LINUX
|
||||||
|
m_ui->snapWarningLabel->setVisible(false);
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef Q_OS_WIN
|
#ifdef Q_OS_WIN
|
||||||
// Brave uses Chrome's registry settings
|
// Brave uses Chrome's registry settings
|
||||||
m_ui->braveSupport->setHidden(true);
|
m_ui->braveSupport->setHidden(true);
|
||||||
|
@ -60,7 +60,7 @@
|
|||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QLabel" name="label">
|
<widget class="QLabel" name="snapWarningLabel">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Browsers installed as snaps are currently not supported.</string>
|
<string>Browsers installed as snaps are currently not supported.</string>
|
||||||
</property>
|
</property>
|
||||||
|
@ -380,7 +380,7 @@ QJsonArray BrowserService::findMatchingEntries(const QString& id,
|
|||||||
// Check entries for authorization
|
// Check entries for authorization
|
||||||
QList<Entry*> pwEntriesToConfirm;
|
QList<Entry*> pwEntriesToConfirm;
|
||||||
QList<Entry*> pwEntries;
|
QList<Entry*> pwEntries;
|
||||||
for (auto* entry : searchEntries(url, keyList)) {
|
for (auto* entry : searchEntries(url, submitUrl, keyList)) {
|
||||||
if (entry->customData()->contains(BrowserService::OPTION_HIDE_ENTRY)
|
if (entry->customData()->contains(BrowserService::OPTION_HIDE_ENTRY)
|
||||||
&& entry->customData()->value(BrowserService::OPTION_HIDE_ENTRY) == "true") {
|
&& entry->customData()->value(BrowserService::OPTION_HIDE_ENTRY) == "true") {
|
||||||
continue;
|
continue;
|
||||||
@ -583,7 +583,7 @@ BrowserService::ReturnValue BrowserService::updateEntry(const QString& id,
|
|||||||
}
|
}
|
||||||
|
|
||||||
QList<Entry*>
|
QList<Entry*>
|
||||||
BrowserService::searchEntries(const QSharedPointer<Database>& db, const QString& hostname, const QString& url)
|
BrowserService::searchEntries(const QSharedPointer<Database>& db, const QString& url, const QString& submitUrl)
|
||||||
{
|
{
|
||||||
QList<Entry*> entries;
|
QList<Entry*> entries;
|
||||||
auto* rootGroup = db->rootGroup();
|
auto* rootGroup = db->rootGroup();
|
||||||
@ -601,19 +601,17 @@ BrowserService::searchEntries(const QSharedPointer<Database>& db, const QString&
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto domain = baseDomain(hostname);
|
|
||||||
|
|
||||||
// Search for additional URL's starting with KP2A_URL
|
// Search for additional URL's starting with KP2A_URL
|
||||||
if (entry->attributes()->keys().contains(ADDITIONAL_URL)) {
|
if (entry->attributes()->keys().contains(ADDITIONAL_URL)) {
|
||||||
for (const auto& key : entry->attributes()->keys()) {
|
for (const auto& key : entry->attributes()->keys()) {
|
||||||
if (key.startsWith(ADDITIONAL_URL) && handleURL(entry->attributes()->value(key), domain, url)) {
|
if (key.startsWith(ADDITIONAL_URL) && handleURL(entry->attributes()->value(key), url, submitUrl)) {
|
||||||
entries.append(entry);
|
entries.append(entry);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!handleURL(entry->url(), domain, url)) {
|
if (!handleURL(entry->url(), url, submitUrl)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -624,7 +622,7 @@ BrowserService::searchEntries(const QSharedPointer<Database>& db, const QString&
|
|||||||
return entries;
|
return entries;
|
||||||
}
|
}
|
||||||
|
|
||||||
QList<Entry*> BrowserService::searchEntries(const QString& url, const StringPairList& keyList)
|
QList<Entry*> BrowserService::searchEntries(const QString& url, const QString& submitUrl, const StringPairList& keyList)
|
||||||
{
|
{
|
||||||
// Check if database is connected with KeePassXC-Browser
|
// Check if database is connected with KeePassXC-Browser
|
||||||
auto databaseConnected = [&](const QSharedPointer<Database>& db) {
|
auto databaseConnected = [&](const QSharedPointer<Database>& db) {
|
||||||
@ -661,7 +659,7 @@ QList<Entry*> BrowserService::searchEntries(const QString& url, const StringPair
|
|||||||
QList<Entry*> entries;
|
QList<Entry*> entries;
|
||||||
do {
|
do {
|
||||||
for (const auto& db : databases) {
|
for (const auto& db : databases) {
|
||||||
entries << searchEntries(db, hostname, url);
|
entries << searchEntries(db, url, submitUrl);
|
||||||
}
|
}
|
||||||
} while (entries.isEmpty() && removeFirstDomain(hostname));
|
} while (entries.isEmpty() && removeFirstDomain(hostname));
|
||||||
|
|
||||||
@ -999,7 +997,7 @@ bool BrowserService::removeFirstDomain(QString& hostname)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool BrowserService::handleURL(const QString& entryUrl, const QString& hostname, const QString& url)
|
bool BrowserService::handleURL(const QString& entryUrl, const QString& url, const QString& submitUrl)
|
||||||
{
|
{
|
||||||
if (entryUrl.isEmpty()) {
|
if (entryUrl.isEmpty()) {
|
||||||
return false;
|
return false;
|
||||||
@ -1016,32 +1014,36 @@ bool BrowserService::handleURL(const QString& entryUrl, const QString& hostname,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Make a direct compare if a local file is used
|
||||||
|
if (url.contains("file://")) {
|
||||||
|
return entryUrl == submitUrl;
|
||||||
|
}
|
||||||
|
|
||||||
// URL host validation fails
|
// URL host validation fails
|
||||||
if (browserSettings()->matchUrlScheme() && entryQUrl.host().isEmpty()) {
|
if (entryQUrl.host().isEmpty()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Match port, if used
|
// Match port, if used
|
||||||
QUrl qUrl(url);
|
QUrl siteQUrl(url);
|
||||||
if (entryQUrl.port() > 0 && entryQUrl.port() != qUrl.port()) {
|
if (entryQUrl.port() > 0 && entryQUrl.port() != siteQUrl.port()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Match scheme
|
// Match scheme
|
||||||
if (browserSettings()->matchUrlScheme() && !entryQUrl.scheme().isEmpty()
|
if (browserSettings()->matchUrlScheme() && !entryQUrl.scheme().isEmpty()
|
||||||
&& entryQUrl.scheme().compare(qUrl.scheme()) != 0) {
|
&& entryQUrl.scheme().compare(siteQUrl.scheme()) != 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for illegal characters
|
// Check for illegal characters
|
||||||
QRegularExpression re("[<>\\^`{|}]");
|
QRegularExpression re("[<>\\^`{|}]");
|
||||||
auto match = re.match(entryUrl);
|
if (re.match(entryUrl).hasMatch()) {
|
||||||
if (match.hasMatch()) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Filter to match hostname in URL field
|
// Filter to match hostname in URL field
|
||||||
if (entryQUrl.host().endsWith(hostname)) {
|
if (siteQUrl.host().endsWith(entryQUrl.host())) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,8 +63,8 @@ public:
|
|||||||
const QString& group,
|
const QString& group,
|
||||||
const QString& groupUuid,
|
const QString& groupUuid,
|
||||||
const QSharedPointer<Database>& selectedDb = {});
|
const QSharedPointer<Database>& selectedDb = {});
|
||||||
QList<Entry*> searchEntries(const QSharedPointer<Database>& db, const QString& hostname, const QString& url);
|
QList<Entry*> searchEntries(const QSharedPointer<Database>& db, const QString& url, const QString& submitUrl);
|
||||||
QList<Entry*> searchEntries(const QString& url, const StringPairList& keyList);
|
QList<Entry*> searchEntries(const QString& url, const QString& submitUrl, const StringPairList& keyList);
|
||||||
void convertAttributesToCustomData(const QSharedPointer<Database>& currentDb = {});
|
void convertAttributesToCustomData(const QSharedPointer<Database>& currentDb = {});
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -130,7 +130,7 @@ private:
|
|||||||
sortPriority(const Entry* entry, const QString& host, const QString& submitUrl, const QString& baseSubmitUrl) const;
|
sortPriority(const Entry* entry, const QString& host, const QString& submitUrl, const QString& baseSubmitUrl) const;
|
||||||
bool schemeFound(const QString& url);
|
bool schemeFound(const QString& url);
|
||||||
bool removeFirstDomain(QString& hostname);
|
bool removeFirstDomain(QString& hostname);
|
||||||
bool handleURL(const QString& entryUrl, const QString& hostname, const QString& url);
|
bool handleURL(const QString& entryUrl, const QString& url, const QString& submitUrl);
|
||||||
QString baseDomain(const QString& hostname) const;
|
QString baseDomain(const QString& hostname) const;
|
||||||
QSharedPointer<Database> getDatabase();
|
QSharedPointer<Database> getDatabase();
|
||||||
QSharedPointer<Database> selectedDatabase();
|
QSharedPointer<Database> selectedDatabase();
|
||||||
|
@ -106,7 +106,11 @@ namespace Bootstrap
|
|||||||
{
|
{
|
||||||
// start minimized if configured
|
// start minimized if configured
|
||||||
if (config()->get("GUI/MinimizeOnStartup").toBool()) {
|
if (config()->get("GUI/MinimizeOnStartup").toBool()) {
|
||||||
|
#ifdef Q_OS_WIN
|
||||||
mainWindow.showMinimized();
|
mainWindow.showMinimized();
|
||||||
|
#else
|
||||||
|
mainWindow.hideWindow();
|
||||||
|
#endif
|
||||||
} else {
|
} else {
|
||||||
mainWindow.bringToFront();
|
mainWindow.bringToFront();
|
||||||
}
|
}
|
||||||
|
@ -42,7 +42,6 @@ Database::Database()
|
|||||||
: m_metadata(new Metadata(this))
|
: m_metadata(new Metadata(this))
|
||||||
, m_data()
|
, m_data()
|
||||||
, m_rootGroup(nullptr)
|
, m_rootGroup(nullptr)
|
||||||
, m_timer(new QTimer(this))
|
|
||||||
, m_fileWatcher(new FileWatcher(this))
|
, m_fileWatcher(new FileWatcher(this))
|
||||||
, m_emitModified(false)
|
, m_emitModified(false)
|
||||||
, m_uuid(QUuid::createUuid())
|
, m_uuid(QUuid::createUuid())
|
||||||
@ -50,12 +49,12 @@ Database::Database()
|
|||||||
setRootGroup(new Group());
|
setRootGroup(new Group());
|
||||||
rootGroup()->setUuid(QUuid::createUuid());
|
rootGroup()->setUuid(QUuid::createUuid());
|
||||||
rootGroup()->setName(tr("Root", "Root group name"));
|
rootGroup()->setName(tr("Root", "Root group name"));
|
||||||
m_timer->setSingleShot(true);
|
m_modifiedTimer.setSingleShot(true);
|
||||||
|
|
||||||
s_uuidMap.insert(m_uuid, this);
|
s_uuidMap.insert(m_uuid, this);
|
||||||
|
|
||||||
connect(m_metadata, SIGNAL(metadataModified()), SLOT(markAsModified()));
|
connect(m_metadata, SIGNAL(metadataModified()), SLOT(markAsModified()));
|
||||||
connect(m_timer, SIGNAL(timeout()), SIGNAL(databaseModified()));
|
connect(&m_modifiedTimer, SIGNAL(timeout()), SIGNAL(databaseModified()));
|
||||||
connect(this, SIGNAL(databaseOpened()), SLOT(updateCommonUsernames()));
|
connect(this, SIGNAL(databaseOpened()), SLOT(updateCommonUsernames()));
|
||||||
connect(this, SIGNAL(databaseSaved()), SLOT(updateCommonUsernames()));
|
connect(this, SIGNAL(databaseSaved()), SLOT(updateCommonUsernames()));
|
||||||
connect(m_fileWatcher, SIGNAL(fileChanged()), SIGNAL(databaseFileChanged()));
|
connect(m_fileWatcher, SIGNAL(fileChanged()), SIGNAL(databaseFileChanged()));
|
||||||
@ -229,6 +228,7 @@ bool Database::saveAs(const QString& filePath, QString* error, bool atomic, bool
|
|||||||
auto& canonicalFilePath = QFileInfo::exists(filePath) ? QFileInfo(filePath).canonicalFilePath() : filePath;
|
auto& canonicalFilePath = QFileInfo::exists(filePath) ? QFileInfo(filePath).canonicalFilePath() : filePath;
|
||||||
bool ok = performSave(canonicalFilePath, error, atomic, backup);
|
bool ok = performSave(canonicalFilePath, error, atomic, backup);
|
||||||
if (ok) {
|
if (ok) {
|
||||||
|
markAsClean();
|
||||||
setFilePath(filePath);
|
setFilePath(filePath);
|
||||||
m_fileWatcher->start(canonicalFilePath, 30, 1);
|
m_fileWatcher->start(canonicalFilePath, 30, 1);
|
||||||
} else {
|
} else {
|
||||||
@ -343,7 +343,6 @@ bool Database::writeDatabase(QIODevice* device, QString* error)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
markAsClean();
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -832,7 +831,7 @@ void Database::emptyRecycleBin()
|
|||||||
void Database::setEmitModified(bool value)
|
void Database::setEmitModified(bool value)
|
||||||
{
|
{
|
||||||
if (m_emitModified && !value) {
|
if (m_emitModified && !value) {
|
||||||
m_timer->stop();
|
m_modifiedTimer.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
m_emitModified = value;
|
m_emitModified = value;
|
||||||
@ -846,8 +845,9 @@ bool Database::isModified() const
|
|||||||
void Database::markAsModified()
|
void Database::markAsModified()
|
||||||
{
|
{
|
||||||
m_modified = true;
|
m_modified = true;
|
||||||
if (m_emitModified) {
|
if (m_emitModified && !m_modifiedTimer.isActive()) {
|
||||||
startModifiedTimer();
|
// Small time delay prevents numerous consecutive saves due to repeated signals
|
||||||
|
m_modifiedTimer.start(150);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -855,6 +855,7 @@ void Database::markAsClean()
|
|||||||
{
|
{
|
||||||
bool emitSignal = m_modified;
|
bool emitSignal = m_modified;
|
||||||
m_modified = false;
|
m_modified = false;
|
||||||
|
m_modifiedTimer.stop();
|
||||||
if (emitSignal) {
|
if (emitSignal) {
|
||||||
emit databaseSaved();
|
emit databaseSaved();
|
||||||
}
|
}
|
||||||
@ -869,18 +870,6 @@ Database* Database::databaseByUuid(const QUuid& uuid)
|
|||||||
return s_uuidMap.value(uuid, nullptr);
|
return s_uuidMap.value(uuid, nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::startModifiedTimer()
|
|
||||||
{
|
|
||||||
if (!m_emitModified) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (m_timer->isActive()) {
|
|
||||||
m_timer->stop();
|
|
||||||
}
|
|
||||||
m_timer->start(150);
|
|
||||||
}
|
|
||||||
|
|
||||||
QSharedPointer<const CompositeKey> Database::key() const
|
QSharedPointer<const CompositeKey> Database::key() const
|
||||||
{
|
{
|
||||||
return m_data.key;
|
return m_data.key;
|
||||||
|
@ -21,9 +21,9 @@
|
|||||||
|
|
||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
#include <QHash>
|
#include <QHash>
|
||||||
#include <QObject>
|
|
||||||
#include <QPointer>
|
#include <QPointer>
|
||||||
#include <QScopedPointer>
|
#include <QScopedPointer>
|
||||||
|
#include <QTimer>
|
||||||
|
|
||||||
#include "config-keepassx.h"
|
#include "config-keepassx.h"
|
||||||
#include "crypto/kdf/AesKdf.h"
|
#include "crypto/kdf/AesKdf.h"
|
||||||
@ -37,7 +37,6 @@ enum class EntryReferenceType;
|
|||||||
class FileWatcher;
|
class FileWatcher;
|
||||||
class Group;
|
class Group;
|
||||||
class Metadata;
|
class Metadata;
|
||||||
class QTimer;
|
|
||||||
class QIODevice;
|
class QIODevice;
|
||||||
|
|
||||||
struct DeletedObject
|
struct DeletedObject
|
||||||
@ -155,9 +154,6 @@ signals:
|
|||||||
void databaseDiscarded();
|
void databaseDiscarded();
|
||||||
void databaseFileChanged();
|
void databaseFileChanged();
|
||||||
|
|
||||||
private slots:
|
|
||||||
void startModifiedTimer();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct DatabaseData
|
struct DatabaseData
|
||||||
{
|
{
|
||||||
@ -211,7 +207,7 @@ private:
|
|||||||
DatabaseData m_data;
|
DatabaseData m_data;
|
||||||
QPointer<Group> m_rootGroup;
|
QPointer<Group> m_rootGroup;
|
||||||
QList<DeletedObject> m_deletedObjects;
|
QList<DeletedObject> m_deletedObjects;
|
||||||
QPointer<QTimer> m_timer;
|
QTimer m_modifiedTimer;
|
||||||
QPointer<FileWatcher> m_fileWatcher;
|
QPointer<FileWatcher> m_fileWatcher;
|
||||||
bool m_initialized = false;
|
bool m_initialized = false;
|
||||||
bool m_modified = false;
|
bool m_modified = false;
|
||||||
|
@ -35,11 +35,10 @@ namespace
|
|||||||
|
|
||||||
FileWatcher::FileWatcher(QObject* parent)
|
FileWatcher::FileWatcher(QObject* parent)
|
||||||
: QObject(parent)
|
: QObject(parent)
|
||||||
, m_ignoreFileChange(false)
|
|
||||||
{
|
{
|
||||||
connect(&m_fileWatcher, SIGNAL(fileChanged(QString)), SLOT(onWatchedFileChanged()));
|
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, SIGNAL(timeout()), SIGNAL(fileChanged()));
|
||||||
connect(&m_fileChecksumTimer, SIGNAL(timeout()), SLOT(checkFileChecksum()));
|
|
||||||
m_fileChangeDelayTimer.setSingleShot(true);
|
m_fileChangeDelayTimer.setSingleShot(true);
|
||||||
m_fileIgnoreDelayTimer.setSingleShot(true);
|
m_fileIgnoreDelayTimer.setSingleShot(true);
|
||||||
}
|
}
|
||||||
@ -101,17 +100,6 @@ void FileWatcher::resume()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FileWatcher::onWatchedFileChanged()
|
|
||||||
{
|
|
||||||
// Don't notify if we are ignoring events or already started a notification chain
|
|
||||||
if (shouldIgnoreChanges()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
m_fileChecksum = calculateChecksum();
|
|
||||||
m_fileChangeDelayTimer.start(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool FileWatcher::shouldIgnoreChanges()
|
bool FileWatcher::shouldIgnoreChanges()
|
||||||
{
|
{
|
||||||
return m_filePath.isEmpty() || m_ignoreFileChange || m_fileIgnoreDelayTimer.isActive()
|
return m_filePath.isEmpty() || m_ignoreFileChange || m_fileIgnoreDelayTimer.isActive()
|
||||||
@ -123,15 +111,23 @@ bool FileWatcher::hasSameFileChecksum()
|
|||||||
return calculateChecksum() == m_fileChecksum;
|
return calculateChecksum() == m_fileChecksum;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FileWatcher::checkFileChecksum()
|
void FileWatcher::checkFileChanged()
|
||||||
{
|
{
|
||||||
if (shouldIgnoreChanges()) {
|
if (shouldIgnoreChanges()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!hasSameFileChecksum()) {
|
// Prevent reentrance
|
||||||
onWatchedFileChanged();
|
m_ignoreFileChange = true;
|
||||||
|
|
||||||
|
// Only trigger the change notice if there is a checksum mismatch
|
||||||
|
auto checksum = calculateChecksum();
|
||||||
|
if (checksum != m_fileChecksum) {
|
||||||
|
m_fileChecksum = checksum;
|
||||||
|
m_fileChangeDelayTimer.start(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
m_ignoreFileChange = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
QByteArray FileWatcher::calculateChecksum()
|
QByteArray FileWatcher::calculateChecksum()
|
||||||
|
@ -43,8 +43,7 @@ public slots:
|
|||||||
void resume();
|
void resume();
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void onWatchedFileChanged();
|
void checkFileChanged();
|
||||||
void checkFileChecksum();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QByteArray calculateChecksum();
|
QByteArray calculateChecksum();
|
||||||
@ -56,8 +55,8 @@ private:
|
|||||||
QTimer m_fileChangeDelayTimer;
|
QTimer m_fileChangeDelayTimer;
|
||||||
QTimer m_fileIgnoreDelayTimer;
|
QTimer m_fileIgnoreDelayTimer;
|
||||||
QTimer m_fileChecksumTimer;
|
QTimer m_fileChecksumTimer;
|
||||||
int m_fileChecksumSizeBytes;
|
int m_fileChecksumSizeBytes = -1;
|
||||||
bool m_ignoreFileChange;
|
bool m_ignoreFileChange = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
class BulkFileWatcher : public QObject
|
class BulkFileWatcher : public QObject
|
||||||
|
@ -328,7 +328,11 @@ namespace FdoSecrets
|
|||||||
|
|
||||||
// when creation finishes in backend, we will already have item
|
// when creation finishes in backend, we will already have item
|
||||||
item = m_entryToItem.value(entry, nullptr);
|
item = m_entryToItem.value(entry, nullptr);
|
||||||
Q_ASSERT(item);
|
|
||||||
|
if (!item) {
|
||||||
|
// may happen if entry somehow ends up in recycle bin
|
||||||
|
return DBusReturn<>::Error(QStringLiteral(DBUS_ERROR_SECRET_NO_SUCH_OBJECT));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = item->setProperties(properties);
|
ret = item->setProperties(properties);
|
||||||
@ -439,7 +443,7 @@ namespace FdoSecrets
|
|||||||
|
|
||||||
auto newUuid = FdoSecrets::settings()->exposedGroup(m_backend->database());
|
auto newUuid = FdoSecrets::settings()->exposedGroup(m_backend->database());
|
||||||
auto newGroup = m_backend->database()->rootGroup()->findGroupByUuid(newUuid);
|
auto newGroup = m_backend->database()->rootGroup()->findGroupByUuid(newUuid);
|
||||||
if (!newGroup) {
|
if (!newGroup || inRecycleBin(newGroup)) {
|
||||||
// no exposed group, delete self
|
// no exposed group, delete self
|
||||||
doDelete();
|
doDelete();
|
||||||
return;
|
return;
|
||||||
@ -451,10 +455,11 @@ namespace FdoSecrets
|
|||||||
m_exposedGroup = newGroup;
|
m_exposedGroup = newGroup;
|
||||||
|
|
||||||
// Attach signal to update exposed group settings if the group was removed.
|
// Attach signal to update exposed group settings if the group was removed.
|
||||||
|
//
|
||||||
// The lifetime of the connection is bound to the database object, because
|
// The lifetime of the connection is bound to the database object, because
|
||||||
// in Database::~Database, groups are also deleted, but we don't want to
|
// in Database::~Database, groups are also deleted as children, but we don't
|
||||||
// trigger this.
|
// want to trigger this.
|
||||||
// This rely on the fact that QObject disconnects signals BEFORE deleting
|
// This works because the fact that QObject disconnects signals BEFORE deleting
|
||||||
// children.
|
// children.
|
||||||
QPointer<Database> db = m_backend->database().data();
|
QPointer<Database> db = m_backend->database().data();
|
||||||
connect(m_exposedGroup.data(), &Group::groupAboutToRemove, db, [db](Group* toBeRemoved) {
|
connect(m_exposedGroup.data(), &Group::groupAboutToRemove, db, [db](Group* toBeRemoved) {
|
||||||
@ -468,6 +473,13 @@ namespace FdoSecrets
|
|||||||
FdoSecrets::settings()->setExposedGroup(db, {});
|
FdoSecrets::settings()->setExposedGroup(db, {});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
// Another possibility is the group being moved to recycle bin.
|
||||||
|
connect(m_exposedGroup.data(), &Group::groupModified, this, [this]() {
|
||||||
|
if (inRecycleBin(m_exposedGroup->parentGroup())) {
|
||||||
|
// reset the exposed group to none
|
||||||
|
FdoSecrets::settings()->setExposedGroup(m_backend->database().data(), {});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// Monitor exposed group settings
|
// Monitor exposed group settings
|
||||||
connect(m_backend->database()->metadata()->customData(), &CustomData::customDataModified, this, [this]() {
|
connect(m_backend->database()->metadata()->customData(), &CustomData::customDataModified, this, [this]() {
|
||||||
@ -646,17 +658,21 @@ namespace FdoSecrets
|
|||||||
{
|
{
|
||||||
Q_ASSERT(m_backend);
|
Q_ASSERT(m_backend);
|
||||||
|
|
||||||
if (!m_backend->database()->metadata()->recycleBin()) {
|
if (!group) {
|
||||||
|
// just to be safe
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!m_backend->database()->metadata()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (group) {
|
auto recycleBin = m_backend->database()->metadata()->recycleBin();
|
||||||
if (group->uuid() == m_backend->database()->metadata()->recycleBin()->uuid()) {
|
if (!recycleBin) {
|
||||||
return true;
|
return false;
|
||||||
}
|
|
||||||
group = group->parentGroup();
|
|
||||||
}
|
}
|
||||||
return false;
|
|
||||||
|
return group->uuid() == recycleBin->uuid() || group->isRecycled();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Collection::inRecycleBin(Entry* entry) const
|
bool Collection::inRecycleBin(Entry* entry) const
|
||||||
|
@ -74,6 +74,11 @@ namespace FdoSecrets
|
|||||||
public:
|
public:
|
||||||
DBusReturn<void> setProperties(const QVariantMap& properties);
|
DBusReturn<void> setProperties(const QVariantMap& properties);
|
||||||
|
|
||||||
|
bool isValid() const
|
||||||
|
{
|
||||||
|
return backend();
|
||||||
|
}
|
||||||
|
|
||||||
DBusReturn<void> removeAlias(QString alias);
|
DBusReturn<void> removeAlias(QString alias);
|
||||||
DBusReturn<void> addAlias(QString alias);
|
DBusReturn<void> addAlias(QString alias);
|
||||||
const QSet<QString> aliases() const;
|
const QSet<QString> aliases() const;
|
||||||
@ -106,6 +111,7 @@ namespace FdoSecrets
|
|||||||
private slots:
|
private slots:
|
||||||
void onDatabaseLockChanged();
|
void onDatabaseLockChanged();
|
||||||
void onDatabaseExposedGroupChanged();
|
void onDatabaseExposedGroupChanged();
|
||||||
|
// force reload info from backend, potentially delete self
|
||||||
void reloadBackend();
|
void reloadBackend();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -93,7 +93,24 @@ namespace FdoSecrets
|
|||||||
|
|
||||||
void Service::onDatabaseTabOpened(DatabaseWidget* dbWidget, bool emitSignal)
|
void Service::onDatabaseTabOpened(DatabaseWidget* dbWidget, bool emitSignal)
|
||||||
{
|
{
|
||||||
|
// The Collection will monitor the database's exposed group.
|
||||||
|
// When the Collection finds that no exposed group, it will delete itself.
|
||||||
|
// Thus the service also needs to monitor it and recreate the collection if the user changes
|
||||||
|
// from no exposed to exposed something.
|
||||||
|
if (!dbWidget->isLocked()) {
|
||||||
|
monitorDatabaseExposedGroup(dbWidget);
|
||||||
|
}
|
||||||
|
connect(dbWidget, &DatabaseWidget::databaseUnlocked, this, [this, dbWidget]() {
|
||||||
|
monitorDatabaseExposedGroup(dbWidget);
|
||||||
|
});
|
||||||
|
|
||||||
auto coll = new Collection(this, dbWidget);
|
auto coll = new Collection(this, dbWidget);
|
||||||
|
// Creation may fail if the database is not exposed.
|
||||||
|
// This is okay, because we monitor the expose settings above
|
||||||
|
if (!coll->isValid()) {
|
||||||
|
coll->deleteLater();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
m_collections << coll;
|
m_collections << coll;
|
||||||
m_dbToCollection[dbWidget] = coll;
|
m_dbToCollection[dbWidget] = coll;
|
||||||
@ -127,15 +144,6 @@ namespace FdoSecrets
|
|||||||
emit collectionDeleted(coll);
|
emit collectionDeleted(coll);
|
||||||
});
|
});
|
||||||
|
|
||||||
// a special case: the database changed from no expose to expose something.
|
|
||||||
// in this case, there is no collection out there monitoring it, so create a new collection
|
|
||||||
if (!dbWidget->isLocked()) {
|
|
||||||
monitorDatabaseExposedGroup(dbWidget);
|
|
||||||
}
|
|
||||||
connect(dbWidget, &DatabaseWidget::databaseUnlocked, this, [this, dbWidget]() {
|
|
||||||
monitorDatabaseExposedGroup(dbWidget);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (emitSignal) {
|
if (emitSignal) {
|
||||||
emit collectionCreated(coll);
|
emit collectionCreated(coll);
|
||||||
}
|
}
|
||||||
|
@ -85,7 +85,7 @@ protected:
|
|||||||
// can not call mapFromSource, which internally calls filterAcceptsRow
|
// can not call mapFromSource, which internally calls filterAcceptsRow
|
||||||
auto group = groupFromSourceIndex(source_idx);
|
auto group = groupFromSourceIndex(source_idx);
|
||||||
|
|
||||||
return group->uuid() != recycleBin->uuid();
|
return group && !group->isRecycled() && group->uuid() != recycleBin->uuid();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -118,8 +118,13 @@ void DatabaseSettingsWidgetFdoSecrets::loadSettings(QSharedPointer<Database> db)
|
|||||||
m_model.reset(new GroupModelNoRecycle(m_db.data()));
|
m_model.reset(new GroupModelNoRecycle(m_db.data()));
|
||||||
m_ui->selectGroup->setModel(m_model.data());
|
m_ui->selectGroup->setModel(m_model.data());
|
||||||
|
|
||||||
|
Group* recycleBin = nullptr;
|
||||||
|
if (m_db->metadata() && m_db->metadata()->recycleBin()) {
|
||||||
|
recycleBin = m_db->metadata()->recycleBin();
|
||||||
|
}
|
||||||
|
|
||||||
auto group = m_db->rootGroup()->findGroupByUuid(FdoSecrets::settings()->exposedGroup(m_db));
|
auto group = m_db->rootGroup()->findGroupByUuid(FdoSecrets::settings()->exposedGroup(m_db));
|
||||||
if (!group) {
|
if (!group || group->isRecycled() || (recycleBin && group->uuid() == recycleBin->uuid())) {
|
||||||
m_ui->radioDonotExpose->setChecked(true);
|
m_ui->radioDonotExpose->setChecked(true);
|
||||||
} else {
|
} else {
|
||||||
auto idx = m_model->indexFromGroup(group);
|
auto idx = m_model->indexFromGroup(group);
|
||||||
|
@ -204,9 +204,14 @@ void DatabaseOpenWidget::openDatabase()
|
|||||||
|
|
||||||
m_db.reset(new Database());
|
m_db.reset(new Database());
|
||||||
QString error;
|
QString error;
|
||||||
|
|
||||||
QApplication::setOverrideCursor(QCursor(Qt::WaitCursor));
|
QApplication::setOverrideCursor(QCursor(Qt::WaitCursor));
|
||||||
|
m_ui->passwordFormFrame->setEnabled(false);
|
||||||
|
QCoreApplication::processEvents();
|
||||||
bool ok = m_db->open(m_filename, masterKey, &error, false);
|
bool ok = m_db->open(m_filename, masterKey, &error, false);
|
||||||
QApplication::restoreOverrideCursor();
|
QApplication::restoreOverrideCursor();
|
||||||
|
m_ui->passwordFormFrame->setEnabled(true);
|
||||||
|
|
||||||
if (!ok) {
|
if (!ok) {
|
||||||
if (m_ui->editPassword->text().isEmpty() && !m_retryUnlockWithEmptyPassword) {
|
if (m_ui->editPassword->text().isEmpty() && !m_retryUnlockWithEmptyPassword) {
|
||||||
QScopedPointer<QMessageBox> msgBox(new QMessageBox(this));
|
QScopedPointer<QMessageBox> msgBox(new QMessageBox(this));
|
||||||
|
@ -132,7 +132,7 @@
|
|||||||
<number>15</number>
|
<number>15</number>
|
||||||
</property>
|
</property>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QFrame" name="verticalFrame">
|
<widget class="QFrame" name="passwordFormFrame">
|
||||||
<property name="minimumSize">
|
<property name="minimumSize">
|
||||||
<size>
|
<size>
|
||||||
<width>400</width>
|
<width>400</width>
|
||||||
|
@ -275,6 +275,11 @@ bool DatabaseWidget::isEntryEditActive() const
|
|||||||
return currentWidget() == m_editEntryWidget;
|
return currentWidget() == m_editEntryWidget;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool DatabaseWidget::isGroupEditActive() const
|
||||||
|
{
|
||||||
|
return currentWidget() == m_editGroupWidget;
|
||||||
|
}
|
||||||
|
|
||||||
bool DatabaseWidget::isEditWidgetModified() const
|
bool DatabaseWidget::isEditWidgetModified() const
|
||||||
{
|
{
|
||||||
if (currentWidget() == m_editEntryWidget) {
|
if (currentWidget() == m_editEntryWidget) {
|
||||||
@ -387,6 +392,8 @@ void DatabaseWidget::createEntry()
|
|||||||
|
|
||||||
void DatabaseWidget::replaceDatabase(QSharedPointer<Database> db)
|
void DatabaseWidget::replaceDatabase(QSharedPointer<Database> db)
|
||||||
{
|
{
|
||||||
|
Q_ASSERT(!isEntryEditActive() && !isGroupEditActive());
|
||||||
|
|
||||||
// Save off new parent UUID which will be valid when creating a new entry
|
// Save off new parent UUID which will be valid when creating a new entry
|
||||||
QUuid newParentUuid;
|
QUuid newParentUuid;
|
||||||
if (m_newParent) {
|
if (m_newParent) {
|
||||||
@ -1370,7 +1377,7 @@ bool DatabaseWidget::lock()
|
|||||||
if (m_db->isModified()) {
|
if (m_db->isModified()) {
|
||||||
bool saved = false;
|
bool saved = false;
|
||||||
// Attempt to save on exit, but don't block locking if it fails
|
// Attempt to save on exit, but don't block locking if it fails
|
||||||
if (config()->get("AutoSaveOnExit").toBool()) {
|
if (config()->get("AutoSaveOnExit").toBool() || config()->get("AutoSaveAfterEveryChange").toBool()) {
|
||||||
saved = save();
|
saved = save();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1421,7 +1428,8 @@ bool DatabaseWidget::lock()
|
|||||||
|
|
||||||
void DatabaseWidget::reloadDatabaseFile()
|
void DatabaseWidget::reloadDatabaseFile()
|
||||||
{
|
{
|
||||||
if (!m_db || isLocked()) {
|
// Ignore reload if we are locked or currently editing an entry or group
|
||||||
|
if (!m_db || isLocked() || isEntryEditActive() || isGroupEditActive()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1441,6 +1449,11 @@ void DatabaseWidget::reloadDatabaseFile()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Lock out interactions
|
||||||
|
m_entryView->setDisabled(true);
|
||||||
|
m_groupView->setDisabled(true);
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
QString error;
|
QString error;
|
||||||
auto db = QSharedPointer<Database>::create(m_db->filePath());
|
auto db = QSharedPointer<Database>::create(m_db->filePath());
|
||||||
if (db->open(database()->key(), &error)) {
|
if (db->open(database()->key(), &error)) {
|
||||||
@ -1480,6 +1493,10 @@ void DatabaseWidget::reloadDatabaseFile()
|
|||||||
// Mark db as modified since existing data may differ from file or file was deleted
|
// Mark db as modified since existing data may differ from file or file was deleted
|
||||||
m_db->markAsModified();
|
m_db->markAsModified();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Return control
|
||||||
|
m_entryView->setDisabled(false);
|
||||||
|
m_groupView->setDisabled(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
int DatabaseWidget::numberOfSelectedEntries() const
|
int DatabaseWidget::numberOfSelectedEntries() const
|
||||||
@ -1620,11 +1637,20 @@ bool DatabaseWidget::save()
|
|||||||
m_blockAutoSave = true;
|
m_blockAutoSave = true;
|
||||||
++m_saveAttempts;
|
++m_saveAttempts;
|
||||||
|
|
||||||
// TODO: Make this async, but lock out the database widget to prevent re-entrance
|
// TODO: Make this async
|
||||||
|
// Lock out interactions
|
||||||
|
m_entryView->setDisabled(true);
|
||||||
|
m_groupView->setDisabled(true);
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
bool useAtomicSaves = config()->get("UseAtomicSaves", true).toBool();
|
bool useAtomicSaves = config()->get("UseAtomicSaves", true).toBool();
|
||||||
QString errorMessage;
|
QString errorMessage;
|
||||||
bool ok = m_db->save(&errorMessage, useAtomicSaves, config()->get("BackupBeforeSave").toBool());
|
bool ok = m_db->save(&errorMessage, useAtomicSaves, config()->get("BackupBeforeSave").toBool());
|
||||||
|
|
||||||
|
// Return control
|
||||||
|
m_entryView->setDisabled(false);
|
||||||
|
m_groupView->setDisabled(false);
|
||||||
|
|
||||||
if (ok) {
|
if (ok) {
|
||||||
m_saveAttempts = 0;
|
m_saveAttempts = 0;
|
||||||
m_blockAutoSave = false;
|
m_blockAutoSave = false;
|
||||||
|
@ -81,6 +81,7 @@ public:
|
|||||||
bool isLocked() const;
|
bool isLocked() const;
|
||||||
bool isSearchActive() const;
|
bool isSearchActive() const;
|
||||||
bool isEntryEditActive() const;
|
bool isEntryEditActive() const;
|
||||||
|
bool isGroupEditActive() const;
|
||||||
|
|
||||||
QString getCurrentSearch();
|
QString getCurrentSearch();
|
||||||
void refreshSearch();
|
void refreshSearch();
|
||||||
|
@ -201,7 +201,7 @@ void EditEntryWidget::setupAdvanced()
|
|||||||
connect(m_advancedUi->editAttributeButton, SIGNAL(clicked()), SLOT(editCurrentAttribute()));
|
connect(m_advancedUi->editAttributeButton, SIGNAL(clicked()), SLOT(editCurrentAttribute()));
|
||||||
connect(m_advancedUi->removeAttributeButton, SIGNAL(clicked()), SLOT(removeCurrentAttribute()));
|
connect(m_advancedUi->removeAttributeButton, SIGNAL(clicked()), SLOT(removeCurrentAttribute()));
|
||||||
connect(m_advancedUi->protectAttributeButton, SIGNAL(toggled(bool)), SLOT(protectCurrentAttribute(bool)));
|
connect(m_advancedUi->protectAttributeButton, SIGNAL(toggled(bool)), SLOT(protectCurrentAttribute(bool)));
|
||||||
connect(m_advancedUi->revealAttributeButton, SIGNAL(clicked(bool)), SLOT(revealCurrentAttribute()));
|
connect(m_advancedUi->revealAttributeButton, SIGNAL(clicked(bool)), SLOT(toggleCurrentAttributeVisibility()));
|
||||||
connect(m_advancedUi->attributesView->selectionModel(),
|
connect(m_advancedUi->attributesView->selectionModel(),
|
||||||
SIGNAL(currentChanged(QModelIndex,QModelIndex)),
|
SIGNAL(currentChanged(QModelIndex,QModelIndex)),
|
||||||
SLOT(updateCurrentAttribute()));
|
SLOT(updateCurrentAttribute()));
|
||||||
@ -1297,6 +1297,7 @@ void EditEntryWidget::displayAttribute(QModelIndex index, bool showProtected)
|
|||||||
// Block signals to prevent modified being set
|
// Block signals to prevent modified being set
|
||||||
m_advancedUi->protectAttributeButton->blockSignals(true);
|
m_advancedUi->protectAttributeButton->blockSignals(true);
|
||||||
m_advancedUi->attributesEdit->blockSignals(true);
|
m_advancedUi->attributesEdit->blockSignals(true);
|
||||||
|
m_advancedUi->revealAttributeButton->setText(tr("Reveal"));
|
||||||
|
|
||||||
if (index.isValid()) {
|
if (index.isValid()) {
|
||||||
QString key = m_attributesModel->keyByIndex(index);
|
QString key = m_attributesModel->keyByIndex(index);
|
||||||
@ -1348,7 +1349,7 @@ void EditEntryWidget::protectCurrentAttribute(bool state)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditEntryWidget::revealCurrentAttribute()
|
void EditEntryWidget::toggleCurrentAttributeVisibility()
|
||||||
{
|
{
|
||||||
if (!m_advancedUi->attributesEdit->isEnabled()) {
|
if (!m_advancedUi->attributesEdit->isEnabled()) {
|
||||||
QModelIndex index = m_advancedUi->attributesView->currentIndex();
|
QModelIndex index = m_advancedUi->attributesView->currentIndex();
|
||||||
@ -1359,6 +1360,10 @@ void EditEntryWidget::revealCurrentAttribute()
|
|||||||
m_advancedUi->attributesEdit->setEnabled(true);
|
m_advancedUi->attributesEdit->setEnabled(true);
|
||||||
m_advancedUi->attributesEdit->blockSignals(oldBlockSignals);
|
m_advancedUi->attributesEdit->blockSignals(oldBlockSignals);
|
||||||
}
|
}
|
||||||
|
m_advancedUi->revealAttributeButton->setText(tr("Hide"));
|
||||||
|
} else {
|
||||||
|
protectCurrentAttribute(true);
|
||||||
|
m_advancedUi->revealAttributeButton->setText(tr("Reveal"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -93,7 +93,7 @@ private slots:
|
|||||||
void removeCurrentAttribute();
|
void removeCurrentAttribute();
|
||||||
void updateCurrentAttribute();
|
void updateCurrentAttribute();
|
||||||
void protectCurrentAttribute(bool state);
|
void protectCurrentAttribute(bool state);
|
||||||
void revealCurrentAttribute();
|
void toggleCurrentAttributeVisibility();
|
||||||
void updateAutoTypeEnabled();
|
void updateAutoTypeEnabled();
|
||||||
void openAutotypeHelp();
|
void openAutotypeHelp();
|
||||||
void insertAutoTypeAssoc();
|
void insertAutoTypeAssoc();
|
||||||
|
@ -105,8 +105,10 @@ void ElidedLabel::updateElidedText()
|
|||||||
const QFontMetrics metrix(font());
|
const QFontMetrics metrix(font());
|
||||||
displayText = metrix.elidedText(m_rawText, m_elideMode, width() - 2);
|
displayText = metrix.elidedText(m_rawText, m_elideMode, width() - 2);
|
||||||
}
|
}
|
||||||
setText(m_url.isEmpty() ? displayText : htmlLinkTemplate.arg(m_url, displayText));
|
|
||||||
setOpenExternalLinks(!m_url.isEmpty());
|
bool hasUrl = !m_url.isEmpty();
|
||||||
|
setText(hasUrl ? htmlLinkTemplate.arg(m_url.toHtmlEscaped(), displayText) : displayText);
|
||||||
|
setOpenExternalLinks(!hasUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ElidedLabel::resizeEvent(QResizeEvent* event)
|
void ElidedLabel::resizeEvent(QResizeEvent* event)
|
||||||
|
@ -179,29 +179,22 @@ void TestBrowser::testSearchEntries()
|
|||||||
auto db = QSharedPointer<Database>::create();
|
auto db = QSharedPointer<Database>::create();
|
||||||
auto* root = db->rootGroup();
|
auto* root = db->rootGroup();
|
||||||
|
|
||||||
QList<QString> urls;
|
QStringList urls = {"https://github.com/login_page",
|
||||||
urls.push_back("https://github.com/login_page");
|
"https://github.com/login",
|
||||||
urls.push_back("https://github.com/login");
|
"https://github.com/",
|
||||||
urls.push_back("https://github.com/");
|
"github.com/login",
|
||||||
urls.push_back("github.com/login");
|
"http://github.com",
|
||||||
urls.push_back("http://github.com");
|
"http://github.com/login",
|
||||||
urls.push_back("http://github.com/login");
|
"github.com",
|
||||||
urls.push_back("github.com");
|
"github.com/login",
|
||||||
urls.push_back("github.com/login");
|
"https://github", // Invalid URL
|
||||||
urls.push_back("https://github"); // Invalid URL
|
"github.com"};
|
||||||
urls.push_back("github.com");
|
|
||||||
|
|
||||||
for (int i = 0; i < urls.length(); ++i) {
|
createEntries(urls, root);
|
||||||
auto entry = new Entry();
|
|
||||||
entry->setGroup(root);
|
|
||||||
entry->beginUpdate();
|
|
||||||
entry->setUrl(urls[i]);
|
|
||||||
entry->setUsername(QString("User %1").arg(i));
|
|
||||||
entry->endUpdate();
|
|
||||||
}
|
|
||||||
|
|
||||||
browserSettings()->setMatchUrlScheme(false);
|
browserSettings()->setMatchUrlScheme(false);
|
||||||
auto result = m_browserService->searchEntries(db, "github.com", "https://github.com"); // db, hostname, url
|
auto result =
|
||||||
|
m_browserService->searchEntries(db, "https://github.com", "https://github.com/session"); // db, url, submitUrl
|
||||||
|
|
||||||
QCOMPARE(result.length(), 9);
|
QCOMPARE(result.length(), 9);
|
||||||
QCOMPARE(result[0]->url(), QString("https://github.com/login_page"));
|
QCOMPARE(result[0]->url(), QString("https://github.com/login_page"));
|
||||||
@ -213,7 +206,7 @@ void TestBrowser::testSearchEntries()
|
|||||||
|
|
||||||
// With matching there should be only 3 results + 4 without a scheme
|
// With matching there should be only 3 results + 4 without a scheme
|
||||||
browserSettings()->setMatchUrlScheme(true);
|
browserSettings()->setMatchUrlScheme(true);
|
||||||
result = m_browserService->searchEntries(db, "github.com", "https://github.com"); // db, hostname, url
|
result = m_browserService->searchEntries(db, "https://github.com", "https://github.com/session");
|
||||||
QCOMPARE(result.length(), 7);
|
QCOMPARE(result.length(), 7);
|
||||||
QCOMPARE(result[0]->url(), QString("https://github.com/login_page"));
|
QCOMPARE(result[0]->url(), QString("https://github.com/login_page"));
|
||||||
QCOMPARE(result[1]->url(), QString("https://github.com/login"));
|
QCOMPARE(result[1]->url(), QString("https://github.com/login"));
|
||||||
@ -226,22 +219,13 @@ void TestBrowser::testSearchEntriesWithPort()
|
|||||||
auto db = QSharedPointer<Database>::create();
|
auto db = QSharedPointer<Database>::create();
|
||||||
auto* root = db->rootGroup();
|
auto* root = db->rootGroup();
|
||||||
|
|
||||||
QList<QString> urls;
|
QStringList urls = {"http://127.0.0.1:443", "http://127.0.0.1:80"};
|
||||||
urls.push_back("http://127.0.0.1:443");
|
|
||||||
urls.push_back("http://127.0.0.1:80");
|
|
||||||
|
|
||||||
for (int i = 0; i < urls.length(); ++i) {
|
createEntries(urls, root);
|
||||||
auto entry = new Entry();
|
|
||||||
entry->setGroup(root);
|
|
||||||
entry->beginUpdate();
|
|
||||||
entry->setUrl(urls[i]);
|
|
||||||
entry->setUsername(QString("User %1").arg(i));
|
|
||||||
entry->endUpdate();
|
|
||||||
}
|
|
||||||
|
|
||||||
auto result = m_browserService->searchEntries(db, "127.0.0.1", "http://127.0.0.1:443"); // db, hostname, url
|
auto result = m_browserService->searchEntries(db, "http://127.0.0.1:443", "http://127.0.0.1");
|
||||||
QCOMPARE(result.length(), 1);
|
QCOMPARE(result.length(), 1);
|
||||||
QCOMPARE(result[0]->url(), urls[0]);
|
QCOMPARE(result[0]->url(), QString("http://127.0.0.1:443"));
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestBrowser::testSearchEntriesWithAdditionalURLs()
|
void TestBrowser::testSearchEntriesWithAdditionalURLs()
|
||||||
@ -249,70 +233,55 @@ void TestBrowser::testSearchEntriesWithAdditionalURLs()
|
|||||||
auto db = QSharedPointer<Database>::create();
|
auto db = QSharedPointer<Database>::create();
|
||||||
auto* root = db->rootGroup();
|
auto* root = db->rootGroup();
|
||||||
|
|
||||||
QList<Entry*> entries;
|
QStringList urls = {"https://github.com/", "https://www.example.com", "http://domain.com"};
|
||||||
QList<QString> urls;
|
|
||||||
urls.push_back("https://github.com/");
|
|
||||||
urls.push_back("https://www.example.com");
|
|
||||||
urls.push_back("http://domain.com");
|
|
||||||
|
|
||||||
for (int i = 0; i < urls.length(); ++i) {
|
auto entries = createEntries(urls, root);
|
||||||
auto entry = new Entry();
|
|
||||||
entry->setGroup(root);
|
|
||||||
entry->beginUpdate();
|
|
||||||
entry->setUrl(urls[i]);
|
|
||||||
entry->setUsername(QString("User %1").arg(i));
|
|
||||||
entry->endUpdate();
|
|
||||||
entries.push_back(entry);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add an additional URL to the first entry
|
// Add an additional URL to the first entry
|
||||||
entries.first()->attributes()->set(BrowserService::ADDITIONAL_URL, "https://keepassxc.org");
|
entries.first()->attributes()->set(BrowserService::ADDITIONAL_URL, "https://keepassxc.org");
|
||||||
|
|
||||||
auto result = m_browserService->searchEntries(db, "github.com", "https://github.com"); // db, hostname, url
|
auto result = m_browserService->searchEntries(db, "https://github.com", "https://github.com/session");
|
||||||
QCOMPARE(result.length(), 1);
|
QCOMPARE(result.length(), 1);
|
||||||
QCOMPARE(result[0]->url(), urls[0]);
|
QCOMPARE(result[0]->url(), QString("https://github.com/"));
|
||||||
|
|
||||||
// Search the additional URL. It should return the same entry
|
// Search the additional URL. It should return the same entry
|
||||||
auto additionalResult = m_browserService->searchEntries(db, "keepassxc.org", "https://keepassxc.org");
|
auto additionalResult = m_browserService->searchEntries(db, "https://keepassxc.org", "https://keepassxc.org");
|
||||||
QCOMPARE(additionalResult.length(), 1);
|
QCOMPARE(additionalResult.length(), 1);
|
||||||
QCOMPARE(additionalResult[0]->url(), urls[0]);
|
QCOMPARE(additionalResult[0]->url(), QString("https://github.com/"));
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestBrowser::testInvalidEntries()
|
void TestBrowser::testInvalidEntries()
|
||||||
{
|
{
|
||||||
auto db = QSharedPointer<Database>::create();
|
auto db = QSharedPointer<Database>::create();
|
||||||
auto* root = db->rootGroup();
|
auto* root = db->rootGroup();
|
||||||
|
const QString url("https://github.com");
|
||||||
|
const QString submitUrl("https://github.com/session");
|
||||||
|
|
||||||
QList<QString> urls;
|
QStringList urls = {
|
||||||
urls.push_back("https://github.com/login");
|
"https://github.com/login",
|
||||||
urls.push_back("https:///github.com/"); // Extra '/'
|
"https:///github.com/", // Extra '/'
|
||||||
urls.push_back("http://github.com/**//*");
|
"http://github.com/**//*",
|
||||||
urls.push_back("http://*.github.com/login");
|
"http://*.github.com/login",
|
||||||
urls.push_back("//github.com"); // fromUserInput() corrects this one.
|
"//github.com", // fromUserInput() corrects this one.
|
||||||
urls.push_back("github.com/{}<>");
|
"github.com/{}<>",
|
||||||
|
"http:/example.com",
|
||||||
|
};
|
||||||
|
|
||||||
for (int i = 0; i < urls.length(); ++i) {
|
createEntries(urls, root);
|
||||||
auto entry = new Entry();
|
|
||||||
entry->setGroup(root);
|
|
||||||
entry->beginUpdate();
|
|
||||||
entry->setUrl(urls[i]);
|
|
||||||
entry->setUsername(QString("User %1").arg(i));
|
|
||||||
entry->endUpdate();
|
|
||||||
}
|
|
||||||
|
|
||||||
browserSettings()->setMatchUrlScheme(true);
|
browserSettings()->setMatchUrlScheme(true);
|
||||||
auto result = m_browserService->searchEntries(db, "github.com", "https://github.com"); // db, hostname, url
|
auto result = m_browserService->searchEntries(db, "https://github.com", "https://github.com/session");
|
||||||
QCOMPARE(result.length(), 2);
|
QCOMPARE(result.length(), 2);
|
||||||
QCOMPARE(result[0]->url(), QString("https://github.com/login"));
|
QCOMPARE(result[0]->url(), QString("https://github.com/login"));
|
||||||
QCOMPARE(result[1]->url(), QString("//github.com"));
|
QCOMPARE(result[1]->url(), QString("//github.com"));
|
||||||
|
|
||||||
// Test the URL's directly
|
// Test the URL's directly
|
||||||
QCOMPARE(m_browserService->handleURL(urls[0], "github.com", "https://github.com"), true);
|
QCOMPARE(m_browserService->handleURL(urls[0], url, submitUrl), true);
|
||||||
QCOMPARE(m_browserService->handleURL(urls[1], "github.com", "https://github.com"), false);
|
QCOMPARE(m_browserService->handleURL(urls[1], url, submitUrl), false);
|
||||||
QCOMPARE(m_browserService->handleURL(urls[2], "github.com", "https://github.com"), false);
|
QCOMPARE(m_browserService->handleURL(urls[2], url, submitUrl), false);
|
||||||
QCOMPARE(m_browserService->handleURL(urls[3], "github.com", "https://github.com"), false);
|
QCOMPARE(m_browserService->handleURL(urls[3], url, submitUrl), false);
|
||||||
QCOMPARE(m_browserService->handleURL(urls[4], "github.com", "https://github.com"), true);
|
QCOMPARE(m_browserService->handleURL(urls[4], url, submitUrl), true);
|
||||||
QCOMPARE(m_browserService->handleURL(urls[5], "github.com", "https://github.com"), false);
|
QCOMPARE(m_browserService->handleURL(urls[5], url, submitUrl), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestBrowser::testSubdomainsAndPaths()
|
void TestBrowser::testSubdomainsAndPaths()
|
||||||
@ -320,44 +289,74 @@ void TestBrowser::testSubdomainsAndPaths()
|
|||||||
auto db = QSharedPointer<Database>::create();
|
auto db = QSharedPointer<Database>::create();
|
||||||
auto* root = db->rootGroup();
|
auto* root = db->rootGroup();
|
||||||
|
|
||||||
QList<QString> urls;
|
QStringList urls = {
|
||||||
urls.push_back("https://www.github.com/login/page.xml");
|
"https://www.github.com/login/page.xml",
|
||||||
urls.push_back("https://login.github.com/");
|
"https://login.github.com/",
|
||||||
urls.push_back("https://github.com");
|
"https://github.com",
|
||||||
urls.push_back("http://www.github.com");
|
"http://www.github.com",
|
||||||
urls.push_back("http://login.github.com/pathtonowhere");
|
"http://login.github.com/pathtonowhere",
|
||||||
urls.push_back(".github.com"); // Invalid URL
|
".github.com", // Invalid URL
|
||||||
urls.push_back("www.github.com/");
|
"www.github.com/",
|
||||||
urls.push_back("https://github"); // Invalid URL
|
"https://github" // Invalid URL
|
||||||
|
};
|
||||||
|
|
||||||
for (int i = 0; i < urls.length(); ++i) {
|
createEntries(urls, root);
|
||||||
auto entry = new Entry();
|
|
||||||
entry->setGroup(root);
|
|
||||||
entry->beginUpdate();
|
|
||||||
entry->setUrl(urls[i]);
|
|
||||||
entry->setUsername(QString("User %1").arg(i));
|
|
||||||
entry->endUpdate();
|
|
||||||
}
|
|
||||||
|
|
||||||
browserSettings()->setMatchUrlScheme(false);
|
browserSettings()->setMatchUrlScheme(false);
|
||||||
auto result = m_browserService->searchEntries(db, "github.com", "https://github.com"); // db, hostname, url
|
auto result = m_browserService->searchEntries(db, "https://github.com", "https://github.com/session");
|
||||||
|
QCOMPARE(result.length(), 1);
|
||||||
|
QCOMPARE(result[0]->url(), QString("https://github.com"));
|
||||||
|
|
||||||
QCOMPARE(result.length(), 6);
|
// With www subdomain
|
||||||
QCOMPARE(result[0]->url(), urls[0]);
|
result = m_browserService->searchEntries(db, "https://www.github.com", "https://www.github.com/session");
|
||||||
QCOMPARE(result[1]->url(), urls[1]);
|
|
||||||
QCOMPARE(result[2]->url(), urls[2]);
|
|
||||||
QCOMPARE(result[3]->url(), urls[3]);
|
|
||||||
QCOMPARE(result[4]->url(), urls[4]);
|
|
||||||
QCOMPARE(result[5]->url(), urls[6]);
|
|
||||||
|
|
||||||
// With matching there should be only 3 results
|
|
||||||
browserSettings()->setMatchUrlScheme(true);
|
|
||||||
result = m_browserService->searchEntries(db, "github.com", "https://github.com"); // db, hostname, url
|
|
||||||
QCOMPARE(result.length(), 4);
|
QCOMPARE(result.length(), 4);
|
||||||
QCOMPARE(result[0]->url(), urls[0]);
|
QCOMPARE(result[0]->url(), QString("https://www.github.com/login/page.xml"));
|
||||||
QCOMPARE(result[1]->url(), urls[1]);
|
QCOMPARE(result[1]->url(), QString("https://github.com")); // Accepts any subdomain
|
||||||
QCOMPARE(result[2]->url(), urls[2]);
|
QCOMPARE(result[2]->url(), QString("http://www.github.com"));
|
||||||
QCOMPARE(result[3]->url(), urls[6]);
|
QCOMPARE(result[3]->url(), QString("www.github.com/"));
|
||||||
|
|
||||||
|
// With scheme matching there should be only 1 result
|
||||||
|
browserSettings()->setMatchUrlScheme(true);
|
||||||
|
result = m_browserService->searchEntries(db, "https://github.com", "https://github.com/session");
|
||||||
|
QCOMPARE(result.length(), 1);
|
||||||
|
QCOMPARE(result[0]->url(), QString("https://github.com"));
|
||||||
|
|
||||||
|
// Test site with subdomain in the site URL
|
||||||
|
QStringList entryURLs = {
|
||||||
|
"https://accounts.example.com",
|
||||||
|
"https://accounts.example.com/path",
|
||||||
|
"https://subdomain.example.com/",
|
||||||
|
"https://another.accounts.example.com/",
|
||||||
|
"https://another.subdomain.example.com/",
|
||||||
|
"https://example.com/",
|
||||||
|
"https://example" // Invalid URL
|
||||||
|
};
|
||||||
|
|
||||||
|
createEntries(entryURLs, root);
|
||||||
|
|
||||||
|
result = m_browserService->searchEntries(db, "https://accounts.example.com", "https://accounts.example.com");
|
||||||
|
QCOMPARE(result.length(), 3);
|
||||||
|
QCOMPARE(result[0]->url(), QString("https://accounts.example.com"));
|
||||||
|
QCOMPARE(result[1]->url(), QString("https://accounts.example.com/path"));
|
||||||
|
QCOMPARE(result[2]->url(), QString("https://example.com/")); // Accepts any subdomain
|
||||||
|
|
||||||
|
result = m_browserService->searchEntries(
|
||||||
|
db, "https://another.accounts.example.com", "https://another.accounts.example.com");
|
||||||
|
QCOMPARE(result.length(), 4);
|
||||||
|
QCOMPARE(result[0]->url(),
|
||||||
|
QString("https://accounts.example.com")); // Accepts any subdomain under accounts.example.com
|
||||||
|
QCOMPARE(result[1]->url(), QString("https://accounts.example.com/path"));
|
||||||
|
QCOMPARE(result[2]->url(), QString("https://another.accounts.example.com/"));
|
||||||
|
QCOMPARE(result[3]->url(), QString("https://example.com/")); // Accepts one or more subdomains
|
||||||
|
|
||||||
|
// Test local files. It should be a direct match.
|
||||||
|
QStringList localFiles = {"file:///Users/testUser/tests/test.html"};
|
||||||
|
|
||||||
|
createEntries(localFiles, root);
|
||||||
|
|
||||||
|
// With local files, url is always set to the file scheme + ://. Submit URL holds the actual URL.
|
||||||
|
result = m_browserService->searchEntries(db, "file://", "file:///Users/testUser/tests/test.html");
|
||||||
|
QCOMPARE(result.length(), 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestBrowser::testSortEntries()
|
void TestBrowser::testSortEntries()
|
||||||
@ -365,28 +364,18 @@ void TestBrowser::testSortEntries()
|
|||||||
auto db = QSharedPointer<Database>::create();
|
auto db = QSharedPointer<Database>::create();
|
||||||
auto* root = db->rootGroup();
|
auto* root = db->rootGroup();
|
||||||
|
|
||||||
QList<QString> urls;
|
QStringList urls = {"https://github.com/login_page",
|
||||||
urls.push_back("https://github.com/login_page");
|
"https://github.com/login",
|
||||||
urls.push_back("https://github.com/login");
|
"https://github.com/",
|
||||||
urls.push_back("https://github.com/");
|
"github.com/login",
|
||||||
urls.push_back("github.com/login");
|
"http://github.com",
|
||||||
urls.push_back("http://github.com");
|
"http://github.com/login",
|
||||||
urls.push_back("http://github.com/login");
|
"github.com",
|
||||||
urls.push_back("github.com");
|
"github.com/login",
|
||||||
urls.push_back("github.com/login");
|
"https://github", // Invalid URL
|
||||||
urls.push_back("https://github"); // Invalid URL
|
"github.com"};
|
||||||
urls.push_back("github.com");
|
|
||||||
|
|
||||||
QList<Entry*> entries;
|
auto entries = createEntries(urls, root);
|
||||||
for (int i = 0; i < urls.length(); ++i) {
|
|
||||||
auto entry = new Entry();
|
|
||||||
entry->setGroup(root);
|
|
||||||
entry->beginUpdate();
|
|
||||||
entry->setUrl(urls[i]);
|
|
||||||
entry->setUsername(QString("User %1").arg(i));
|
|
||||||
entry->endUpdate();
|
|
||||||
entries.push_back(entry);
|
|
||||||
}
|
|
||||||
|
|
||||||
browserSettings()->setBestMatchOnly(false);
|
browserSettings()->setBestMatchOnly(false);
|
||||||
auto result =
|
auto result =
|
||||||
@ -457,3 +446,19 @@ void TestBrowser::testGetDatabaseGroups()
|
|||||||
auto lastChild = lastChildren.at(0);
|
auto lastChild = lastChildren.at(0);
|
||||||
QCOMPARE(lastChild.toObject()["name"].toString(), QString("group2_1_1"));
|
QCOMPARE(lastChild.toObject()["name"].toString(), QString("group2_1_1"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QList<Entry*> TestBrowser::createEntries(QStringList& urls, Group* root) const
|
||||||
|
{
|
||||||
|
QList<Entry*> entries;
|
||||||
|
for (int i = 0; i < urls.length(); ++i) {
|
||||||
|
auto entry = new Entry();
|
||||||
|
entry->setGroup(root);
|
||||||
|
entry->beginUpdate();
|
||||||
|
entry->setUrl(urls[i]);
|
||||||
|
entry->setUsername(QString("User %1").arg(i));
|
||||||
|
entry->endUpdate();
|
||||||
|
entries.push_back(entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
return entries;
|
||||||
|
}
|
@ -49,6 +49,8 @@ private slots:
|
|||||||
void testGetDatabaseGroups();
|
void testGetDatabaseGroups();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
QList<Entry*> createEntries(QStringList& urls, Group* root) const;
|
||||||
|
|
||||||
QScopedPointer<BrowserAction> m_browserAction;
|
QScopedPointer<BrowserAction> m_browserAction;
|
||||||
QScopedPointer<BrowserService> m_browserService;
|
QScopedPointer<BrowserService> m_browserService;
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user