Merge branch 'release/2.5.2' into develop

This commit is contained in:
Jonathan White 2019-12-15 00:11:02 -05:00
commit ed0b76813d
22 changed files with 297 additions and 226 deletions

View File

@ -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);

View File

@ -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>

View File

@ -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;
} }

View File

@ -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();

View File

@ -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();
} }

View File

@ -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;

View File

@ -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;

View File

@ -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()

View File

@ -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

View File

@ -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

View File

@ -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:

View File

@ -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);
} }

View File

@ -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);

View File

@ -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));

View File

@ -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>

View File

@ -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;

View File

@ -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();

View File

@ -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"));
} }
} }

View File

@ -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();

View File

@ -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)

View File

@ -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;
}

View File

@ -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;
}; };