Libkleo

keyselectiondialog.cpp
1/* -*- c++ -*-
2 keyselectiondialog.cpp
3
4 This file is part of libkleopatra, the KDE keymanagement library
5 SPDX-FileCopyrightText: 2004 Klarälvdalens Datakonsult AB
6
7 Based on kpgpui.cpp
8 SPDX-FileCopyrightText: 2001, 2002 the KPGP authors
9 See file libkdenetwork/AUTHORS.kpgp for details
10
11 SPDX-License-Identifier: GPL-2.0-or-later
12*/
13
14#include <config-libkleo.h>
15
16#include "keyselectiondialog.h"
17
18#include "keylistview.h"
19#include "progressdialog.h"
20
21#include <libkleo/compat.h>
22#include <libkleo/compliance.h>
23#include <libkleo/formatting.h>
24
25#include <kleo_ui_debug.h>
26
27#include <KConfig>
28#include <KConfigGroup>
29#include <KLocalizedString>
30#include <KMessageBox>
31#include <KSharedConfig>
32
33#include <QGpgME/KeyListJob>
34
35#include <QApplication>
36#include <QCheckBox>
37#include <QDialogButtonBox>
38#include <QFrame>
39#include <QHBoxLayout>
40#include <QLabel>
41#include <QLineEdit>
42#include <QMenu>
43#include <QProcess>
44#include <QPushButton>
45#include <QRegularExpression>
46#include <QScrollBar>
47#include <QTimer>
48#include <QVBoxLayout>
49
50#include <gpgme++/key.h>
51#include <gpgme++/keylistresult.h>
52
53#include <algorithm>
54#include <iterator>
55#include <string.h>
56
57using namespace Kleo;
58
59static bool checkKeyUsage(const GpgME::Key &key, unsigned int keyUsage, QString *statusString = nullptr)
60{
61 auto setStatusString = [statusString](const QString &status) {
62 if (statusString) {
63 *statusString = status;
64 }
65 };
66
67 if (keyUsage & KeySelectionDialog::ValidKeys) {
68 if (key.isInvalid()) {
69 if (key.keyListMode() & GpgME::Validate) {
70 qCDebug(KLEO_UI_LOG) << "key is invalid";
71 setStatusString(i18n("The key is not valid."));
72 return false;
73 } else {
74 qCDebug(KLEO_UI_LOG) << "key is invalid - ignoring";
75 }
76 }
77 if (key.isExpired()) {
78 qCDebug(KLEO_UI_LOG) << "key is expired";
79 setStatusString(i18n("The key is expired."));
80 return false;
81 } else if (key.isRevoked()) {
82 qCDebug(KLEO_UI_LOG) << "key is revoked";
83 setStatusString(i18n("The key is revoked."));
84 return false;
85 } else if (key.isDisabled()) {
86 qCDebug(KLEO_UI_LOG) << "key is disabled";
87 setStatusString(i18n("The key is disabled."));
88 return false;
89 }
90 }
91
92 if (keyUsage & KeySelectionDialog::EncryptionKeys && !Kleo::keyHasEncrypt(key)) {
93 qCDebug(KLEO_UI_LOG) << "key can't encrypt";
94 setStatusString(i18n("The key is not designated for encryption."));
95 return false;
96 }
97 if (keyUsage & KeySelectionDialog::SigningKeys && !Kleo::keyHasSign(key)) {
98 qCDebug(KLEO_UI_LOG) << "key can't sign";
99 setStatusString(i18n("The key is not designated for signing."));
100 return false;
101 }
102 if (keyUsage & KeySelectionDialog::CertificationKeys && !Kleo::keyHasCertify(key)) {
103 qCDebug(KLEO_UI_LOG) << "key can't certify";
104 setStatusString(i18n("The key is not designated for certifying."));
105 return false;
106 }
107 if (keyUsage & KeySelectionDialog::AuthenticationKeys && !Kleo::keyHasAuthenticate(key)) {
108 qCDebug(KLEO_UI_LOG) << "key can't authenticate";
109 setStatusString(i18n("The key is not designated for authentication."));
110 return false;
111 }
112
113 if (keyUsage & KeySelectionDialog::SecretKeys && !(keyUsage & KeySelectionDialog::PublicKeys) && !key.hasSecret()) {
114 qCDebug(KLEO_UI_LOG) << "key isn't secret";
115 setStatusString(i18n("The key is not secret."));
116 return false;
117 }
118
119 if (keyUsage & KeySelectionDialog::TrustedKeys && key.protocol() == GpgME::OpenPGP &&
120 // only check this for secret keys for now.
121 // Seems validity isn't checked for secret keylistings...
122 !key.hasSecret()) {
123 std::vector<GpgME::UserID> uids = key.userIDs();
124 for (std::vector<GpgME::UserID>::const_iterator it = uids.begin(); it != uids.end(); ++it) {
125 if (!it->isRevoked() && it->validity() >= GpgME::UserID::Marginal) {
126 setStatusString(i18n("The key can be used."));
127 return true;
128 }
129 }
130 qCDebug(KLEO_UI_LOG) << "key has no UIDs with validity >= Marginal";
131 setStatusString(i18n("The key is not trusted enough."));
132 return false;
133 }
134 // X.509 keys are always trusted, else they won't be the keybox.
135 // PENDING(marc) check that this ^ is correct
136
137 setStatusString(i18n("The key can be used."));
138 return true;
139}
140
141static bool checkKeyUsage(const std::vector<GpgME::Key> &keys, unsigned int keyUsage)
142{
143 for (auto it = keys.begin(); it != keys.end(); ++it) {
144 if (!checkKeyUsage(*it, keyUsage)) {
145 return false;
146 }
147 }
148 return true;
149}
150
151namespace
152{
153
154class ColumnStrategy : public KeyListView::ColumnStrategy
155{
156public:
157 ColumnStrategy(unsigned int keyUsage);
158
159 QString title(int col) const override;
160 int width(int col, const QFontMetrics &fm) const override;
161
162 QString text(const GpgME::Key &key, int col) const override;
163 QString accessibleText(const GpgME::Key &key, int column) const override;
164 QString toolTip(const GpgME::Key &key, int col) const override;
165 QIcon icon(const GpgME::Key &key, int col) const override;
166
167private:
168 const QIcon mKeyGoodPix, mKeyBadPix, mKeyUnknownPix, mKeyValidPix;
169 const unsigned int mKeyUsage;
170};
171
172ColumnStrategy::ColumnStrategy(unsigned int keyUsage)
173 : KeyListView::ColumnStrategy()
174 , mKeyGoodPix(QStringLiteral(":/libkleopatra/key_ok"))
175 , mKeyBadPix(QStringLiteral(":/libkleopatra/key_bad"))
176 , mKeyUnknownPix(QStringLiteral(":/libkleopatra/key_unknown"))
177 , mKeyValidPix(QStringLiteral(":/libkleopatra/key"))
178 , mKeyUsage(keyUsage)
179{
180 if (keyUsage == 0) {
181 qCWarning(KLEO_UI_LOG) << "KeySelectionDialog: keyUsage == 0. You want to use AllKeys instead.";
182 }
183}
184
185QString ColumnStrategy::title(int col) const
186{
187 switch (col) {
188 case 0:
189 return i18n("Key ID");
190 case 1:
191 return i18n("User ID");
192 default:
193 return QString();
194 }
195}
196
197int ColumnStrategy::width(int col, const QFontMetrics &fm) const
198{
199 if (col == 0) {
200 static const char hexchars[] = "0123456789ABCDEF";
201 int maxWidth = 0;
202 for (unsigned int i = 0; i < 16; ++i) {
203 maxWidth = qMax(fm.boundingRect(QLatin1Char(hexchars[i])).width(), maxWidth);
204 }
205 return 8 * maxWidth + 2 * 16 /* KIconLoader::SizeSmall */;
206 }
207 return KeyListView::ColumnStrategy::width(col, fm);
208}
209
210QString ColumnStrategy::text(const GpgME::Key &key, int col) const
211{
212 switch (col) {
213 case 0: {
214 if (key.keyID()) {
215 return Formatting::prettyID(key.keyID());
216 } else {
217 return xi18n("<placeholder>unknown</placeholder>");
218 }
219 }
220 case 1: {
221 const char *uid = key.userID(0).id();
222 if (key.protocol() == GpgME::OpenPGP) {
223 return uid && *uid ? QString::fromUtf8(uid) : QString();
224 } else { // CMS
225 return Formatting::prettyDN(uid);
226 }
227 }
228 default:
229 return QString();
230 }
231}
232
233QString ColumnStrategy::accessibleText(const GpgME::Key &key, int col) const
234{
235 switch (col) {
236 case 0: {
237 if (key.keyID()) {
238 return Formatting::accessibleHexID(key.keyID());
239 }
240 [[fallthrough]];
241 }
242 default:
243 return {};
244 }
245}
246
247QString ColumnStrategy::toolTip(const GpgME::Key &key, int) const
248{
249 const char *uid = key.userID(0).id();
250 const char *fpr = key.primaryFingerprint();
251 const char *issuer = key.issuerName();
252 const GpgME::Subkey subkey = key.subkey(0);
253 const QString expiry = Formatting::expirationDateString(subkey);
254 const QString creation = Formatting::creationDateString(subkey);
255 QString keyStatusString;
256 if (!checkKeyUsage(key, mKeyUsage, &keyStatusString)) {
257 // Show the status in bold if there is a problem
258 keyStatusString = QLatin1StringView("<b>") % keyStatusString % QLatin1StringView("</b>");
259 }
260
261 QString html = QStringLiteral("<qt><p style=\"style='white-space:pre'\">");
262 if (key.protocol() == GpgME::OpenPGP) {
263 html += i18n("OpenPGP key for <b>%1</b>", uid ? QString::fromUtf8(uid) : i18n("unknown"));
264 } else {
265 html += i18n("S/MIME key for <b>%1</b>", uid ? Formatting::prettyDN(uid) : i18n("unknown"));
266 }
267 html += QStringLiteral("</p><table>");
268
269 const auto addRow = [&html](const QString &name, const QString &value) {
270 html += QStringLiteral("<tr><td align=\"right\"><b>%1: </b></td><td>%2</td></tr>").arg(name, value);
271 };
272 addRow(i18n("Valid from"), creation);
273 addRow(i18n("Valid until"), expiry);
274 addRow(i18nc("Key fingerprint", "Fingerprint"), fpr ? QString::fromLatin1(fpr) : i18n("unknown"));
275 if (key.protocol() != GpgME::OpenPGP) {
276 addRow(i18nc("Key issuer", "Issuer"), issuer ? Formatting::prettyDN(issuer) : i18n("unknown"));
277 }
278 addRow(i18nc("Key status", "Status"), keyStatusString);
279 if (DeVSCompliance::isActive()) {
280 addRow(i18nc("Compliance of key", "Compliance"), DeVSCompliance::name(key.isDeVs()));
281 }
282 html += QStringLiteral("</table></qt>");
283
284 return html;
285}
286
287QIcon ColumnStrategy::icon(const GpgME::Key &key, int col) const
288{
289 if (col != 0) {
290 return QIcon();
291 }
292 // this key did not undergo a validating keylisting yet:
293 if (!(key.keyListMode() & GpgME::Validate)) {
294 return mKeyUnknownPix;
295 }
296
297 if (!checkKeyUsage(key, mKeyUsage)) {
298 return mKeyBadPix;
299 }
300
301 if (key.protocol() == GpgME::CMS) {
302 return mKeyGoodPix;
303 }
304
305 switch (key.userID(0).validity()) {
306 default:
307 case GpgME::UserID::Unknown:
308 case GpgME::UserID::Undefined:
309 return mKeyUnknownPix;
310 case GpgME::UserID::Never:
311 return mKeyValidPix;
312 case GpgME::UserID::Marginal:
313 case GpgME::UserID::Full:
314 case GpgME::UserID::Ultimate: {
315 if (DeVSCompliance::isActive() && !key.isDeVs()) {
316 return mKeyValidPix;
317 }
318 return mKeyGoodPix;
319 }
320 }
321}
322
323}
324
325static const int sCheckSelectionDelay = 250;
326
327KeySelectionDialog::KeySelectionDialog(QWidget *parent, Options options)
328 : QDialog(parent)
329 , mOpenPGPBackend(QGpgME::openpgp())
330 , mSMIMEBackend(QGpgME::smime())
331 , mKeyUsage(AllKeys)
332{
333 qCDebug(KLEO_UI_LOG) << "mTruncated:" << mTruncated << "mSavedOffsetY:" << mSavedOffsetY;
334 setUpUI(options, QString());
335}
336
337KeySelectionDialog::KeySelectionDialog(const QString &title,
338 const QString &text,
339 const std::vector<GpgME::Key> &selectedKeys,
340 unsigned int keyUsage,
341 bool extendedSelection,
342 bool rememberChoice,
343 QWidget *parent,
344 bool modal)
345 : QDialog(parent)
346 , mSelectedKeys(selectedKeys)
347 , mKeyUsage(keyUsage)
348{
349 setWindowTitle(title);
350 setModal(modal);
351 init(rememberChoice, extendedSelection, text, QString());
352}
353
354KeySelectionDialog::KeySelectionDialog(const QString &title,
355 const QString &text,
356 const QString &initialQuery,
357 const std::vector<GpgME::Key> &selectedKeys,
358 unsigned int keyUsage,
359 bool extendedSelection,
360 bool rememberChoice,
361 QWidget *parent,
362 bool modal)
363 : QDialog(parent)
364 , mSelectedKeys(selectedKeys)
365 , mKeyUsage(keyUsage)
366 , mSearchText(initialQuery)
367 , mInitialQuery(initialQuery)
368{
369 setWindowTitle(title);
370 setModal(modal);
371 init(rememberChoice, extendedSelection, text, initialQuery);
372}
373
374KeySelectionDialog::KeySelectionDialog(const QString &title,
375 const QString &text,
376 const QString &initialQuery,
377 unsigned int keyUsage,
378 bool extendedSelection,
379 bool rememberChoice,
380 QWidget *parent,
381 bool modal)
382 : QDialog(parent)
383 , mKeyUsage(keyUsage)
384 , mSearchText(initialQuery)
385 , mInitialQuery(initialQuery)
386{
387 setWindowTitle(title);
388 setModal(modal);
389 init(rememberChoice, extendedSelection, text, initialQuery);
390}
391
392void KeySelectionDialog::setUpUI(Options options, const QString &initialQuery)
393{
394 auto mainLayout = new QVBoxLayout(this);
395 QDialogButtonBox *buttonBox = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel, this);
396 mOkButton = buttonBox->button(QDialogButtonBox::Ok);
397 mOkButton->setDefault(true);
398 mOkButton->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_Return));
399
400 mCheckSelectionTimer = new QTimer(this);
401 mStartSearchTimer = new QTimer(this);
402
403 QFrame *page = new QFrame(this);
404 mainLayout->addWidget(page);
405 mainLayout->addWidget(buttonBox);
406
407 mTopLayout = new QVBoxLayout(page);
408 mTopLayout->setContentsMargins(0, 0, 0, 0);
409
410 mTextLabel = new QLabel(page);
411 mTextLabel->setWordWrap(true);
412
413 // Setting the size policy is necessary as a workaround for https://issues.kolab.org/issue4429
414 // and http://bugreports.qt.nokia.com/browse/QTBUG-8740
416 connect(mTextLabel, &QLabel::linkActivated, this, &KeySelectionDialog::slotStartCertificateManager);
417 mTopLayout->addWidget(mTextLabel);
418 mTextLabel->hide();
419
420 QPushButton *const searchExternalPB = new QPushButton(i18nc("@action:button", "Search for &External Certificates"), page);
421 mTopLayout->addWidget(searchExternalPB, 0, Qt::AlignLeft);
422 connect(searchExternalPB, &QAbstractButton::clicked, this, &KeySelectionDialog::slotStartSearchForExternalCertificates);
423 if (initialQuery.isEmpty()) {
424 searchExternalPB->hide();
425 }
426
427 auto hlay = new QHBoxLayout();
428 mTopLayout->addLayout(hlay);
429
430 auto le = new QLineEdit(page);
431 le->setClearButtonEnabled(true);
432 le->setText(initialQuery);
433
434 QLabel *lbSearchFor = new QLabel(i18nc("@label:textbox", "&Search for:"), page);
435 lbSearchFor->setBuddy(le);
436
437 hlay->addWidget(lbSearchFor);
438 hlay->addWidget(le, 1);
439 le->setFocus();
440
441 connect(le, &QLineEdit::textChanged, this, [this](const QString &s) {
442 slotSearch(s);
443 });
444 connect(mStartSearchTimer, &QTimer::timeout, this, &KeySelectionDialog::slotFilter);
445
446 mKeyListView = new KeyListView(new ColumnStrategy(mKeyUsage), nullptr, page);
447 mKeyListView->setObjectName(QLatin1StringView("mKeyListView"));
448 mKeyListView->header()->stretchLastSection();
449 mKeyListView->setRootIsDecorated(true);
450 mKeyListView->setSortingEnabled(true);
451 mKeyListView->header()->setSortIndicatorShown(true);
452 mKeyListView->header()->setSortIndicator(1, Qt::AscendingOrder); // sort by User ID
453 if (options & ExtendedSelection) {
454 mKeyListView->setSelectionMode(QAbstractItemView::ExtendedSelection);
455 }
456 mTopLayout->addWidget(mKeyListView, 10);
457
458 if (options & RememberChoice) {
459 mRememberCB = new QCheckBox(i18nc("@option:check", "&Remember choice"), page);
460 mTopLayout->addWidget(mRememberCB);
461 mRememberCB->setWhatsThis(
462 i18n("<qt><p>If you check this box your choice will "
463 "be stored and you will not be asked again."
464 "</p></qt>"));
465 }
466
467 connect(mCheckSelectionTimer, &QTimer::timeout, this, [this]() {
468 slotCheckSelection();
469 });
470 connectSignals();
471
472 connect(mKeyListView, &KeyListView::doubleClicked, this, &KeySelectionDialog::slotTryOk);
473 connect(mKeyListView, &KeyListView::contextMenu, this, &KeySelectionDialog::slotRMB);
474
475 if (options & RereadKeys) {
476 QPushButton *button = new QPushButton(i18nc("@action:button", "&Reread Keys"));
477 buttonBox->addButton(button, QDialogButtonBox::ActionRole);
478 connect(button, &QPushButton::clicked, this, &KeySelectionDialog::slotRereadKeys);
479 }
480 if (options & ExternalCertificateManager) {
481 QPushButton *button = new QPushButton(i18nc("@action:button", "&Start Certificate Manager"));
482 buttonBox->addButton(button, QDialogButtonBox::ActionRole);
483 connect(button, &QPushButton::clicked, this, [this]() {
484 slotStartCertificateManager();
485 });
486 }
487 connect(mOkButton, &QPushButton::clicked, this, &KeySelectionDialog::slotOk);
488 connect(buttonBox->button(QDialogButtonBox::Cancel), &QPushButton::clicked, this, &KeySelectionDialog::slotCancel);
489
490 mTopLayout->activate();
491
492 if (qApp) {
493 QSize dialogSize(sizeHint());
494 KConfigGroup dialogConfig(KSharedConfig::openStateConfig(), QStringLiteral("Key Selection Dialog"));
495 dialogSize = dialogConfig.readEntry("Dialog size", dialogSize);
496 const QByteArray headerState = dialogConfig.readEntry("header", QByteArray());
497 if (!headerState.isEmpty()) {
498 mKeyListView->header()->restoreState(headerState);
499 }
500 resize(dialogSize);
501 }
502}
503
504void KeySelectionDialog::init(bool rememberChoice, bool extendedSelection, const QString &text, const QString &initialQuery)
505{
506 Options options = {RereadKeys, ExternalCertificateManager};
507 options.setFlag(ExtendedSelection, extendedSelection);
508 options.setFlag(RememberChoice, rememberChoice);
509
510 setUpUI(options, initialQuery);
511 setText(text);
512
513 if (mKeyUsage & OpenPGPKeys) {
514 mOpenPGPBackend = QGpgME::openpgp();
515 }
516 if (mKeyUsage & SMIMEKeys) {
517 mSMIMEBackend = QGpgME::smime();
518 }
519
520 slotRereadKeys();
521}
522
523KeySelectionDialog::~KeySelectionDialog()
524{
525 disconnectSignals();
526 KConfigGroup dialogConfig(KSharedConfig::openStateConfig(), QStringLiteral("Key Selection Dialog"));
527 dialogConfig.writeEntry("Dialog size", size());
528 dialogConfig.writeEntry("header", mKeyListView->header()->saveState());
529 dialogConfig.sync();
530}
531
532void KeySelectionDialog::setText(const QString &text)
533{
534 mTextLabel->setText(text);
535 mTextLabel->setVisible(!text.isEmpty());
536}
537
538void KeySelectionDialog::setKeys(const std::vector<GpgME::Key> &keys)
539{
540 for (const GpgME::Key &key : keys) {
541 mKeyListView->slotAddKey(key);
542 }
543}
544
545void KeySelectionDialog::connectSignals()
546{
547 if (mKeyListView->isMultiSelection()) {
548 connect(mKeyListView, &QTreeWidget::itemSelectionChanged, this, &KeySelectionDialog::slotSelectionChanged);
549 } else {
550 connect(mKeyListView,
551 qOverload<KeyListViewItem *>(&KeyListView::selectionChanged),
552 this,
553 qOverload<KeyListViewItem *>(&KeySelectionDialog::slotCheckSelection));
554 }
555}
556
557void KeySelectionDialog::disconnectSignals()
558{
559 if (mKeyListView->isMultiSelection()) {
560 disconnect(mKeyListView, &QTreeWidget::itemSelectionChanged, this, &KeySelectionDialog::slotSelectionChanged);
561 } else {
562 disconnect(mKeyListView,
563 qOverload<KeyListViewItem *>(&KeyListView::selectionChanged),
564 this,
565 qOverload<KeyListViewItem *>(&KeySelectionDialog::slotCheckSelection));
566 }
567}
568
569const GpgME::Key &KeySelectionDialog::selectedKey() const
570{
571 static const GpgME::Key null = GpgME::Key::null;
572 if (mKeyListView->isMultiSelection() || !mKeyListView->selectedItem()) {
573 return null;
574 }
575 return mKeyListView->selectedItem()->key();
576}
577
578QString KeySelectionDialog::fingerprint() const
579{
580 return QLatin1StringView(selectedKey().primaryFingerprint());
581}
582
583QStringList KeySelectionDialog::fingerprints() const
584{
585 QStringList result;
586 for (auto it = mSelectedKeys.begin(); it != mSelectedKeys.end(); ++it) {
587 if (const char *fpr = it->primaryFingerprint()) {
588 result.push_back(QLatin1StringView(fpr));
589 }
590 }
591 return result;
592}
593
594QStringList KeySelectionDialog::pgpKeyFingerprints() const
595{
596 QStringList result;
597 for (auto it = mSelectedKeys.begin(); it != mSelectedKeys.end(); ++it) {
598 if (it->protocol() == GpgME::OpenPGP) {
599 if (const char *fpr = it->primaryFingerprint()) {
600 result.push_back(QLatin1StringView(fpr));
601 }
602 }
603 }
604 return result;
605}
606
607QStringList KeySelectionDialog::smimeFingerprints() const
608{
609 QStringList result;
610 for (auto it = mSelectedKeys.begin(); it != mSelectedKeys.end(); ++it) {
611 if (it->protocol() == GpgME::CMS) {
612 if (const char *fpr = it->primaryFingerprint()) {
613 result.push_back(QLatin1StringView(fpr));
614 }
615 }
616 }
617 return result;
618}
619
620void KeySelectionDialog::slotRereadKeys()
621{
622 mKeyListView->clear();
623 mListJobCount = 0;
624 mTruncated = 0;
625 mSavedOffsetY = mKeyListView->verticalScrollBar()->value();
626
627 disconnectSignals();
628 mKeyListView->setEnabled(false);
629
630 // FIXME: save current selection
631 if (mOpenPGPBackend) {
632 startKeyListJobForBackend(mOpenPGPBackend, std::vector<GpgME::Key>(), false /*non-validating*/);
633 }
634 if (mSMIMEBackend) {
635 startKeyListJobForBackend(mSMIMEBackend, std::vector<GpgME::Key>(), false /*non-validating*/);
636 }
637
638 if (mListJobCount == 0) {
639 mKeyListView->setEnabled(true);
641 i18n("No backends found for listing keys. "
642 "Check your installation."),
643 i18nc("@title:window", "Key Listing Failed"));
644 connectSignals();
645 }
646}
647
648void KeySelectionDialog::slotStartCertificateManager(const QString &query)
649{
650 QStringList args;
651
652 if (!query.isEmpty()) {
653 args << QStringLiteral("--search") << query;
654 }
655 const QString exec = QStandardPaths::findExecutable(QStringLiteral("kleopatra"));
656 if (exec.isEmpty()) {
657 qCWarning(KLEO_UI_LOG) << "Could not find kleopatra executable in PATH";
659 i18n("Could not start certificate manager; "
660 "please check your installation."),
661 i18nc("@title:window", "Certificate Manager Error"));
662 } else {
663 QProcess::startDetached(QStringLiteral("kleopatra"), args);
664 qCDebug(KLEO_UI_LOG) << "\nslotStartCertManager(): certificate manager started.";
665 }
666}
667
668#ifndef __KLEO_UI_SHOW_KEY_LIST_ERROR_H__
669#define __KLEO_UI_SHOW_KEY_LIST_ERROR_H__
670static void showKeyListError(QWidget *parent, const GpgME::Error &err)
671{
672 Q_ASSERT(err);
673 const QString msg = i18n(
674 "<qt><p>An error occurred while fetching "
675 "the keys from the backend:</p>"
676 "<p><b>%1</b></p></qt>",
677 Formatting::errorAsString(err));
678
679 KMessageBox::error(parent, msg, i18nc("@title:window", "Key Listing Failed"));
680}
681#endif // __KLEO_UI_SHOW_KEY_LIST_ERROR_H__
682
683namespace
684{
685struct ExtractFingerprint {
686 QString operator()(const GpgME::Key &key)
687 {
688 return QLatin1StringView(key.primaryFingerprint());
689 }
690};
691}
692
693void KeySelectionDialog::startKeyListJobForBackend(const QGpgME::Protocol *backend, const std::vector<GpgME::Key> &keys, bool validate)
694{
695 Q_ASSERT(backend);
696 QGpgME::KeyListJob *job = backend->keyListJob(false, false, validate); // local, w/o sigs, validation as given
697 if (!job) {
698 return;
699 }
700
701 connect(job, &QGpgME::KeyListJob::result, this, &KeySelectionDialog::slotKeyListResult);
702 if (validate) {
703 connect(job, &QGpgME::KeyListJob::nextKey, mKeyListView, &KeyListView::slotRefreshKey);
704 } else {
705 connect(job, &QGpgME::KeyListJob::nextKey, mKeyListView, &KeyListView::slotAddKey);
706 }
707
708 QStringList fprs;
709 std::transform(keys.begin(), keys.end(), std::back_inserter(fprs), ExtractFingerprint());
710 const GpgME::Error err = job->start(fprs, mKeyUsage & SecretKeys && !(mKeyUsage & PublicKeys));
711
712 if (err) {
713 return showKeyListError(this, err);
714 }
715
716#ifndef LIBKLEO_NO_PROGRESSDIALOG
717 // FIXME: create a MultiProgressDialog:
718 (void)new ProgressDialog(job, validate ? i18n("Checking selected keys...") : i18n("Fetching keys..."), this);
719#endif
720 ++mListJobCount;
721}
722
723static void selectKeys(KeyListView *klv, const std::vector<GpgME::Key> &selectedKeys)
724{
725 klv->clearSelection();
726 if (selectedKeys.empty()) {
727 return;
728 }
729 for (auto it = selectedKeys.begin(); it != selectedKeys.end(); ++it) {
730 if (KeyListViewItem *item = klv->itemByFingerprint(it->primaryFingerprint())) {
731 item->setSelected(true);
732 }
733 }
734}
735
736void KeySelectionDialog::slotKeyListResult(const GpgME::KeyListResult &res)
737{
738 if (res.error()) {
739 showKeyListError(this, res.error());
740 } else if (res.isTruncated()) {
741 ++mTruncated;
742 }
743
744 if (--mListJobCount > 0) {
745 return; // not yet finished...
746 }
747
748 if (mTruncated > 0) {
750 i18np("<qt>One backend returned truncated output.<p>"
751 "Not all available keys are shown</p></qt>",
752 "<qt>%1 backends returned truncated output.<p>"
753 "Not all available keys are shown</p></qt>",
754 mTruncated),
755 i18n("Key List Result"));
756 }
757
758 mKeyListView->flushKeys();
759
760 mKeyListView->setEnabled(true);
761 mListJobCount = mTruncated = 0;
762 mKeysToCheck.clear();
763
764 selectKeys(mKeyListView, mSelectedKeys);
765
766 slotFilter();
767
768 connectSignals();
769
770 slotSelectionChanged();
771
772 // restore the saved position of the contents
773 mKeyListView->verticalScrollBar()->setValue(mSavedOffsetY);
774 mSavedOffsetY = 0;
775}
776
777void KeySelectionDialog::slotSelectionChanged()
778{
779 qCDebug(KLEO_UI_LOG) << "KeySelectionDialog::slotSelectionChanged()";
780
781 // (re)start the check selection timer. Checking the selection is delayed
782 // because else drag-selection doesn't work very good (checking key trust
783 // is slow).
784 mCheckSelectionTimer->start(sCheckSelectionDelay);
785}
786
787namespace
788{
789struct AlreadyChecked {
790 bool operator()(const GpgME::Key &key) const
791 {
792 return key.keyListMode() & GpgME::Validate;
793 }
794};
795}
796
797void KeySelectionDialog::slotCheckSelection(KeyListViewItem *item)
798{
799 qCDebug(KLEO_UI_LOG) << "KeySelectionDialog::slotCheckSelection()";
800
801 mCheckSelectionTimer->stop();
802
803 mSelectedKeys.clear();
804
805 if (!mKeyListView->isMultiSelection()) {
806 if (item) {
807 mSelectedKeys.push_back(item->key());
808 }
809 }
810
811 for (KeyListViewItem *it = mKeyListView->firstChild(); it; it = it->nextSibling()) {
812 if (it->isSelected()) {
813 mSelectedKeys.push_back(it->key());
814 }
815 }
816
817 mKeysToCheck.clear();
818 std::remove_copy_if(mSelectedKeys.begin(), mSelectedKeys.end(), std::back_inserter(mKeysToCheck), AlreadyChecked());
819 if (mKeysToCheck.empty()) {
820 mOkButton->setEnabled(!mSelectedKeys.empty() && checkKeyUsage(mSelectedKeys, mKeyUsage));
821 return;
822 }
823
824 // performed all fast checks - now for validating key listing:
825 startValidatingKeyListing();
826}
827
828void KeySelectionDialog::startValidatingKeyListing()
829{
830 if (mKeysToCheck.empty()) {
831 return;
832 }
833
834 mListJobCount = 0;
835 mTruncated = 0;
836 mSavedOffsetY = mKeyListView->verticalScrollBar()->value();
837
838 disconnectSignals();
839 mKeyListView->setEnabled(false);
840
841 std::vector<GpgME::Key> smime;
842 std::vector<GpgME::Key> openpgp;
843 for (std::vector<GpgME::Key>::const_iterator it = mKeysToCheck.begin(); it != mKeysToCheck.end(); ++it) {
844 if (it->protocol() == GpgME::OpenPGP) {
845 openpgp.push_back(*it);
846 } else {
847 smime.push_back(*it);
848 }
849 }
850
851 if (!openpgp.empty()) {
852 Q_ASSERT(mOpenPGPBackend);
853 startKeyListJobForBackend(mOpenPGPBackend, openpgp, true /*validate*/);
854 }
855 if (!smime.empty()) {
856 Q_ASSERT(mSMIMEBackend);
857 startKeyListJobForBackend(mSMIMEBackend, smime, true /*validate*/);
858 }
859
860 Q_ASSERT(mListJobCount > 0);
861}
862
863bool KeySelectionDialog::rememberSelection() const
864{
865 return mRememberCB && mRememberCB->isChecked();
866}
867
868void KeySelectionDialog::slotRMB(KeyListViewItem *item, const QPoint &p)
869{
870 if (!item) {
871 return;
872 }
873
874 mCurrentContextMenuItem = item;
875
876 QMenu menu;
877 menu.addAction(i18n("Recheck Key"), this, &KeySelectionDialog::slotRecheckKey);
878 menu.exec(p);
879}
880
881void KeySelectionDialog::slotRecheckKey()
882{
883 if (!mCurrentContextMenuItem || mCurrentContextMenuItem->key().isNull()) {
884 return;
885 }
886
887 mKeysToCheck.clear();
888 mKeysToCheck.push_back(mCurrentContextMenuItem->key());
889}
890
891void KeySelectionDialog::slotTryOk()
892{
893 if (!mSelectedKeys.empty() && checkKeyUsage(mSelectedKeys, mKeyUsage)) {
894 slotOk();
895 }
896}
897
898void KeySelectionDialog::slotOk()
899{
900 if (mCheckSelectionTimer->isActive()) {
901 slotCheckSelection();
902 }
903#if 0 // Laurent I don't understand why we returns here.
904 // button could be disabled again after checking the selected key1
905 if (!mSelectedKeys.empty() && checkKeyUsage(mSelectedKeys, mKeyUsage)) {
906 return;
907 }
908#endif
909 mStartSearchTimer->stop();
910 accept();
911}
912
913void KeySelectionDialog::slotCancel()
914{
915 mCheckSelectionTimer->stop();
916 mStartSearchTimer->stop();
917 reject();
918}
919
920void KeySelectionDialog::slotSearch(const QString &text)
921{
922 mSearchText = text.trimmed().toUpper();
923 slotSearch();
924}
925
926void KeySelectionDialog::slotSearch()
927{
928 mStartSearchTimer->setSingleShot(true);
929 mStartSearchTimer->start(sCheckSelectionDelay);
930}
931
932void KeySelectionDialog::slotFilter()
933{
934 if (mSearchText.isEmpty()) {
935 showAllItems();
936 return;
937 }
938
939 // OK, so we need to filter:
940 QRegularExpression keyIdRegExp(QRegularExpression::anchoredPattern(QLatin1StringView("(?:0x)?[A-F0-9]{1,16}")), QRegularExpression::CaseInsensitiveOption);
941 if (keyIdRegExp.match(mSearchText).hasMatch()) {
942 if (mSearchText.startsWith(QLatin1StringView("0X"))) {
943 // search for keyID only:
944 filterByKeyID(mSearchText.mid(2));
945 } else {
946 // search for UID and keyID:
947 filterByKeyIDOrUID(mSearchText);
948 }
949 } else {
950 // search in UID:
951 filterByUID(mSearchText);
952 }
953}
954
955void KeySelectionDialog::filterByKeyID(const QString &keyID)
956{
957 Q_ASSERT(keyID.length() <= 16);
958 Q_ASSERT(!keyID.isEmpty()); // regexp in slotFilter should prevent these
959 if (keyID.isEmpty()) {
960 showAllItems();
961 } else {
962 for (KeyListViewItem *item = mKeyListView->firstChild(); item; item = item->nextSibling()) {
963 item->setHidden(!item->text(0).remove(u' ').toUpper().startsWith(keyID));
964 }
965 }
966}
967
968static bool anyUIDMatches(const KeyListViewItem *item, const QRegularExpression &rx)
969{
970 if (!item) {
971 return false;
972 }
973
974 const std::vector<GpgME::UserID> uids = item->key().userIDs();
975 for (auto it = uids.begin(); it != uids.end(); ++it) {
976 if (it->id() && rx.match(QString::fromUtf8(it->id())).hasMatch()) {
977 return true;
978 }
979 }
980 return false;
981}
982
983void KeySelectionDialog::filterByKeyIDOrUID(const QString &str)
984{
985 Q_ASSERT(!str.isEmpty());
986
987 // match beginnings of words:
988 QRegularExpression rx(QLatin1StringView("\\b") + QRegularExpression::escape(str), QRegularExpression::CaseInsensitiveOption);
989
990 for (KeyListViewItem *item = mKeyListView->firstChild(); item; item = item->nextSibling()) {
991 item->setHidden(!item->text(0).remove(u' ').toUpper().startsWith(str) && !anyUIDMatches(item, rx));
992 }
993}
994
995void KeySelectionDialog::filterByUID(const QString &str)
996{
997 Q_ASSERT(!str.isEmpty());
998
999 // match beginnings of words:
1000 QRegularExpression rx(QLatin1StringView("\\b") + QRegularExpression::escape(str), QRegularExpression::CaseInsensitiveOption);
1001
1002 for (KeyListViewItem *item = mKeyListView->firstChild(); item; item = item->nextSibling()) {
1003 item->setHidden(!anyUIDMatches(item, rx));
1004 }
1005}
1006
1007void KeySelectionDialog::showAllItems()
1008{
1009 for (KeyListViewItem *item = mKeyListView->firstChild(); item; item = item->nextSibling()) {
1010 item->setHidden(false);
1011 }
1012}
1013
1014#include "moc_keyselectiondialog.cpp"
static KSharedConfig::Ptr openStateConfig(const QString &fileName=QString())
Q_SCRIPTABLE CaptureState status()
QString i18np(const char *singular, const char *plural, const TYPE &arg...)
QString i18nc(const char *context, const char *text, const TYPE &arg...)
QString xi18n(const char *text, const TYPE &arg...)
QString i18n(const char *text, const TYPE &arg...)
std::optional< QSqlQuery > query(const QString &queryStatement)
void information(QWidget *parent, const QString &text, const QString &title=QString(), const QString &dontShowAgainName=QString(), Options options=Notify)
void error(QWidget *parent, const QString &text, const QString &title, const KGuiItem &buttonOk, Options options=Notify)
QCA_EXPORT void init()
void clicked(bool checked)
bool isEmpty() const const
virtual void accept()
virtual int exec()
virtual void reject()
int result() const const
virtual QSize sizeHint() const const override
QPushButton * addButton(StandardButton button)
QPushButton * button(StandardButton which) const const
QRect boundingRect(QChar ch) const const
void linkActivated(const QString &link)
void setBuddy(QWidget *buddy)
void textChanged(const QString &text)
QAction * addAction(const QIcon &icon, const QString &text, Functor functor, const QKeySequence &shortcut)
QAction * exec()
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
bool disconnect(const QMetaObject::Connection &connection)
bool startDetached(const QString &program, const QStringList &arguments, const QString &workingDirectory, qint64 *pid)
void setDefault(bool)
QRegularExpressionMatch match(QStringView subjectView, qsizetype offset, MatchType matchType, MatchOptions matchOptions) const const
QString anchoredPattern(QStringView expression)
QString escape(QStringView str)
bool hasMatch() const const
QString findExecutable(const QString &executableName, const QStringList &paths)
QString arg(Args &&... args) const const
QString fromLatin1(QByteArrayView str)
QString fromUtf8(QByteArrayView str)
bool isEmpty() const const
qsizetype length() const const
QString & remove(QChar ch, Qt::CaseSensitivity cs)
bool startsWith(QChar c, Qt::CaseSensitivity cs) const const
QString toUpper() const const
QString trimmed() const const
AlignLeft
Key_Return
AscendingOrder
QTestData & addRow(const char *format,...)
void timeout()
void itemSelectionChanged()
void setHidden(bool hide)
QString text(int column) const const
void hide()
void resize(const QSize &)
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 12:04:00 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.