diff --git a/autotests/abstractkeylistmodeltest.cpp b/autotests/abstractkeylistmodeltest.cpp index 7b44617b5..46b2ca8fa 100644 --- a/autotests/abstractkeylistmodeltest.cpp +++ b/autotests/abstractkeylistmodeltest.cpp @@ -1,207 +1,206 @@ /* autotests/abstractkeylistmodeltest.cpp This file is part of libkleopatra's test suite. SPDX-FileCopyrightText: 2021 g10 Code GmbH SPDX-FileContributor: Ingo Klöcker SPDX-License-Identifier: GPL-2.0-or-later */ #include "abstractkeylistmodeltest.h" #include "kleo/keygroup.h" #include "models/keylistmodel.h" #include #include #include using namespace Kleo; using namespace GpgME; namespace { Key createTestKey(const char *uid) { static int count = 0; count++; gpgme_key_t key; gpgme_key_from_uid(&key, uid); const QByteArray fingerprint = QByteArray::number(count, 16).rightJustified(40, '0'); key->fpr = strdup(fingerprint.constData()); return Key(key, false); } KeyGroup createGroup(const QString &name, const std::vector &keys = std::vector(), - KeyGroup::Source source = KeyGroup::UnknownSource, + KeyGroup::Source source = KeyGroup::ApplicationConfig, const QString &configName = QString()) { - static KeyGroup::Id nextGroupId = 0; - - KeyGroup g(nextGroupId, name, keys, source); - ++nextGroupId; - g.setConfigName(configName); + const KeyGroup::Id groupId = (source == KeyGroup::ApplicationConfig) ? + (configName.isEmpty() ? name : configName) : + name; + KeyGroup g(groupId, name, keys, source); return g; } } void AbstractKeyListModelTest::testCreation() { QScopedPointer model(createModel()); QCOMPARE( model->rowCount(), 0 ); } void AbstractKeyListModelTest::testSetKeys() { QScopedPointer model(createModel()); const std::vector keys = { createTestKey("test1@example.net") }; model->setKeys(keys); QCOMPARE( model->rowCount(), 1 ); QVERIFY( model->index(keys[0]).isValid() ); const std::vector otherKeys = { createTestKey("test2@example.net"), createTestKey("test3@example.net") }; model->setKeys(otherKeys); QCOMPARE( model->rowCount(), 2 ); QVERIFY( model->index(otherKeys[0]).isValid() ); QVERIFY( model->index(otherKeys[1]).isValid() ); QVERIFY( !model->index(keys[0]).isValid() ); } void AbstractKeyListModelTest::testSetGroups() { QScopedPointer model(createModel()); const std::vector groups = { createGroup("test1") }; model->setGroups(groups); QCOMPARE( model->rowCount(), 1 ); QVERIFY( model->index(groups[0]).isValid() ); const std::vector otherGroups = { createGroup("test2"), createGroup("test3") }; model->setGroups(otherGroups); QCOMPARE( model->rowCount(), 2 ); QVERIFY( model->index(otherGroups[0]).isValid() ); QVERIFY( model->index(otherGroups[1]).isValid() ); QVERIFY( !model->index(groups[0]).isValid() ); } void AbstractKeyListModelTest::testKeys() { QScopedPointer model(createModel()); const Key key = createTestKey("test@example.net"); const KeyGroup group = createGroup("test", {key}); model->setKeys({key}); model->setGroups({group}); QCOMPARE( model->rowCount(), 2 ); const QModelIndex keyIndex = model->index(key); QVERIFY( keyIndex.isValid() ); const QModelIndex groupIndex = model->index(group); QVERIFY( groupIndex.isValid() ); { const auto keys = model->keys({}); QCOMPARE( keys.size(), 0 ); } { const auto keys = model->keys({keyIndex}); QCOMPARE( keys.size(), 1 ); QCOMPARE( keys[0].userID(0).addrSpec(), UserID::addrSpecFromString("test@example.net") ); } { // duplicate keys are removed from result const auto keys = model->keys({keyIndex, keyIndex}); QCOMPARE( keys.size(), 1 ); QCOMPARE( keys[0].userID(0).addrSpec(), UserID::addrSpecFromString("test@example.net") ); } { // null keys are removed from result const auto keys = model->keys({groupIndex}); QCOMPARE( keys.size(), 0 ); } } void AbstractKeyListModelTest::testIndex() { QScopedPointer model(createModel()); const Key key = createTestKey("test@example.net"); const std::vector groups = { createGroup("test", {key}, KeyGroup::UnknownSource), createGroup("test", {key}, KeyGroup::GnuPGConfig), createGroup("test", {key}, KeyGroup::ApplicationConfig, "test"), createGroup("test", {key}, KeyGroup::ApplicationConfig, "otherConfigName") }; model->setKeys({key}); model->setGroups(groups); const QModelIndex keyIndex = model->index(0, 0); QVERIFY( keyIndex.isValid() ); QVERIFY( !model->key(keyIndex).isNull() ); const QModelIndex groupIndex = model->index(1, 0); QVERIFY( groupIndex.isValid() ); QVERIFY( !model->group(groupIndex).isNull() ); } void AbstractKeyListModelTest::testIndexForGroup() { QScopedPointer model(createModel()); const Key key = createTestKey("test@example.net"); const std::vector groups = { createGroup("test", {key}, KeyGroup::UnknownSource), createGroup("test", {key}, KeyGroup::GnuPGConfig), createGroup("test", {key}, KeyGroup::ApplicationConfig, "test"), createGroup("test", {key}, KeyGroup::ApplicationConfig, "otherConfigName") }; model->setKeys({key}); model->setGroups(groups); QSet rows; for (const KeyGroup &group : groups) { const QModelIndex groupIndex = model->index(group); QVERIFY( groupIndex.isValid() ); rows.insert(groupIndex.row()); } QCOMPARE(rows.size(), 4); } void AbstractKeyListModelTest::testClear() { QScopedPointer model(createModel()); model->setGroups({ createGroup("test") }); model->clear(AbstractKeyListModel::Keys); QCOMPARE( model->rowCount(), 1 ); model->clear(AbstractKeyListModel::Groups); QCOMPARE( model->rowCount(), 0 ); } diff --git a/src/kleo/keygroup.cpp b/src/kleo/keygroup.cpp index 3f56ef367..0d3b11c58 100644 --- a/src/kleo/keygroup.cpp +++ b/src/kleo/keygroup.cpp @@ -1,155 +1,140 @@ /* kleo/keygroup.cpp This file is part of libkleopatra, the KDE keymanagement library SPDX-FileCopyrightText: 2021 g10 Code GmbH SPDX-FileContributor: Ingo Klöcker SPDX-License-Identifier: GPL-2.0-or-later */ #include "keygroup.h" #include #include using namespace Kleo; using namespace GpgME; -static const KeyGroup::Id nullId = -1; - class KeyGroup::Private { public: explicit Private(Id id, const QString &name, const std::vector &keys, Source source); Id id; QString name; - QString configName; Keys keys; Source source; bool isImmutable = true; }; KeyGroup::Private::Private(Id id, const QString &name, const std::vector &keys, Source source) : id(id) , name(name) , keys(keys.cbegin(), keys.cend()) , source(source) { } KeyGroup::KeyGroup() - : KeyGroup(nullId, QString(), {}, UnknownSource) + : KeyGroup(QString(), QString(), {}, UnknownSource) { } KeyGroup::~KeyGroup() = default; KeyGroup::KeyGroup(Id id, const QString &name, const std::vector &keys, Source source) : d(new Private(id, name, keys, source)) { } KeyGroup::KeyGroup(const KeyGroup &other) : d(new Private(*other.d)) { } KeyGroup &KeyGroup::operator=(const KeyGroup &other) { *d = *other.d; return *this; } KeyGroup::KeyGroup(KeyGroup &&other) = default; KeyGroup &KeyGroup::operator=(KeyGroup &&other) = default; bool KeyGroup::isNull() const { - return !d || d->id == nullId; + return !d || d->id.isEmpty(); } KeyGroup::Id KeyGroup::id() const { - return d ? d->id : nullId; + return d ? d->id : QString(); } void KeyGroup::setName(const QString &name) { if (d) { d->name = name; } } QString KeyGroup::name() const { return d ? d->name : QString(); } void KeyGroup::setKeys(const KeyGroup::Keys &keys) { if (d) { d->keys = keys; } } void KeyGroup::setKeys(const std::vector &keys) { if (d) { d->keys = Keys(keys.cbegin(), keys.cend()); } } const KeyGroup::Keys &KeyGroup::keys() const { static const Keys empty; return d ? d->keys : empty; } KeyGroup::Source KeyGroup::source() const { return d ? d->source : UnknownSource; } -void KeyGroup::setConfigName(const QString &configName) -{ - if (d) { - d->configName = configName; - } -} - -QString KeyGroup::configName() const -{ - return d ? d->configName : QString(); -} - void KeyGroup::setIsImmutable(bool isImmutable) { if (d) { d->isImmutable = isImmutable; } } bool KeyGroup::isImmutable() const { return d ? d->isImmutable : true; } bool KeyGroup::insert(const GpgME::Key &key) { if (!d || key.isNull()) { return false; } return d->keys.insert(key).second; } bool KeyGroup::erase(const GpgME::Key &key) { if (!d || key.isNull()) { return false; } return d->keys.erase(key) > 0; } diff --git a/src/kleo/keygroup.h b/src/kleo/keygroup.h index 43259ee6b..4e409f120 100644 --- a/src/kleo/keygroup.h +++ b/src/kleo/keygroup.h @@ -1,84 +1,81 @@ /* kleo/keygroup.h This file is part of libkleopatra, the KDE keymanagement library SPDX-FileCopyrightText: 2021 g10 Code GmbH SPDX-FileContributor: Ingo Klöcker SPDX-License-Identifier: GPL-2.0-or-later */ #ifndef LIBKLEO_KEYGROUP_H #define LIBKLEO_KEYGROUP_H #include "kleo_export.h" #include #include #include #include class QString; namespace GpgME { class Key; } namespace Kleo { class KLEO_EXPORT KeyGroup { public: - typedef int32_t Id; + typedef QString Id; typedef std::set> Keys; enum Source { UnknownSource, ApplicationConfig, GnuPGConfig, Tags }; KeyGroup(); ~KeyGroup(); explicit KeyGroup(Id id, const QString &name, const std::vector &keys, Source source); KeyGroup(const KeyGroup &other); KeyGroup &operator=(const KeyGroup &other); KeyGroup(KeyGroup &&other); KeyGroup &operator=(KeyGroup &&other); bool isNull() const; Id id() const; Source source() const; void setName(const QString &name); QString name() const; void setKeys(const Keys &keys); void setKeys(const std::vector &keys); const Keys &keys() const; - void setConfigName(const QString &configName); - QString configName() const; - void setIsImmutable(bool isImmutable); bool isImmutable() const; bool insert(const GpgME::Key &key); bool erase(const GpgME::Key &key); private: class Private; std::unique_ptr d; }; } #endif // LIBKLEO_KEYGROUP_H diff --git a/src/models/keycache.cpp b/src/models/keycache.cpp index 74f8b490b..0d2a0c13c 100644 --- a/src/models/keycache.cpp +++ b/src/models/keycache.cpp @@ -1,1500 +1,1526 @@ /* -*- mode: c++; c-basic-offset:4 -*- models/keycache.cpp This file is part of Kleopatra, the KDE keymanager SPDX-FileCopyrightText: 2007, 2008 Klarälvdalens Datakonsult AB SPDX-FileCopyrightText: 2018 Intevation GmbH SPDX-FileCopyrightText: 2020 g10 Code GmbH SPDX-FileContributor: Ingo Klöcker SPDX-License-Identifier: GPL-2.0-or-later */ #include "keycache.h" #include "keycache_p.h" -#include "libkleo_debug.h" - #include "kleo/keygroup.h" #include "kleo/predicates.h" #include "kleo/stl_util.h" #include "kleo/dn.h" #include "utils/filesystemwatcher.h" #include #include #include #include #include #include #include #include #include #include #include #include //#include #include #include #include #include #include #include #include +#include "kleo/debug.h" +#include "libkleo_debug.h" + using namespace Kleo; using namespace GpgME; using namespace KMime::Types; static const unsigned int hours2ms = 1000 * 60 * 60; static const QString groupNamePrefix = QStringLiteral("Group-"); // // // KeyCache // // namespace { make_comparator_str(ByEMail, .first.c_str()); } class KeyCache::Private { friend class ::Kleo::KeyCache; KeyCache *const q; public: explicit Private(KeyCache *qq) : q(qq), m_refreshInterval(1), m_initalized(false), m_pgpOnly(true), m_remarks_enabled(false) { connect(&m_autoKeyListingTimer, &QTimer::timeout, q, [this]() { q->startKeyListing(); }); updateAutoKeyListingTimer(); } ~Private() { if (m_refreshJob) { m_refreshJob->cancel(); } } template < template