2011-11-13 08:55:20 -05:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2011 Felix Geyer <debfx@fobos.de>
|
2017-06-09 17:40:36 -04:00
|
|
|
* Copyright (C) 2017 KeePassXC Team <team@keepassxc.org>
|
2011-11-13 08:55:20 -05:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 2 or (at your option)
|
|
|
|
* version 3 of the License.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
#include "DatabaseOpenWidget.h"
|
|
|
|
#include "ui_DatabaseOpenWidget.h"
|
2011-11-13 08:55:20 -05:00
|
|
|
|
2021-07-11 22:10:29 -04:00
|
|
|
#include "config-keepassx.h"
|
2011-12-25 19:21:29 -05:00
|
|
|
#include "gui/FileDialog.h"
|
2020-10-05 20:41:00 -04:00
|
|
|
#include "gui/Icons.h"
|
2018-03-31 16:01:30 -04:00
|
|
|
#include "gui/MainWindow.h"
|
2013-10-08 16:09:20 -04:00
|
|
|
#include "gui/MessageBox.h"
|
2021-04-22 23:07:49 -04:00
|
|
|
#include "keys/ChallengeResponseKey.h"
|
2011-12-24 13:19:52 -05:00
|
|
|
#include "keys/FileKey.h"
|
2017-02-20 19:06:32 -05:00
|
|
|
|
2021-07-11 22:10:29 -04:00
|
|
|
#ifdef Q_OS_MACOS
|
|
|
|
#include "touchid/TouchID.h"
|
|
|
|
#endif
|
2022-02-21 20:40:01 -05:00
|
|
|
#ifdef Q_CC_MSVC
|
|
|
|
#include "winhello/WindowsHello.h"
|
|
|
|
#endif
|
2014-05-26 03:49:28 -04:00
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
#include <QCheckBox>
|
2022-06-12 16:35:42 -04:00
|
|
|
#include <QCloseEvent>
|
2019-06-22 12:00:31 -04:00
|
|
|
#include <QDesktopServices>
|
|
|
|
#include <QFont>
|
2011-11-13 08:55:20 -05:00
|
|
|
|
2020-06-06 09:54:57 -04:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
constexpr int clearFormsDelay = 30000;
|
2022-02-21 20:40:01 -05:00
|
|
|
|
|
|
|
bool isQuickUnlockAvailable()
|
|
|
|
{
|
|
|
|
if (config()->get(Config::Security_QuickUnlock).toBool()) {
|
|
|
|
#if defined(Q_CC_MSVC)
|
|
|
|
return getWindowsHello()->isAvailable();
|
|
|
|
#elif defined(Q_OS_MACOS)
|
|
|
|
return TouchID::getInstance().isAvailable();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool canPerformQuickUnlock(const QString& filename)
|
|
|
|
{
|
|
|
|
if (isQuickUnlockAvailable()) {
|
|
|
|
#if defined(Q_CC_MSVC)
|
|
|
|
return getWindowsHello()->hasKey(filename);
|
|
|
|
#elif defined(Q_OS_MACOS)
|
|
|
|
return TouchID::getInstance().containsKey(filename);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
Q_UNUSED(filename);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} // namespace
|
2020-06-06 09:54:57 -04:00
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
DatabaseOpenWidget::DatabaseOpenWidget(QWidget* parent)
|
2017-03-10 14:42:59 -05:00
|
|
|
: DialogyWidget(parent)
|
|
|
|
, m_ui(new Ui::DatabaseOpenWidget())
|
|
|
|
, m_db(nullptr)
|
2011-11-13 08:55:20 -05:00
|
|
|
{
|
|
|
|
m_ui->setupUi(this);
|
|
|
|
|
2015-01-19 18:14:59 -05:00
|
|
|
m_ui->messageWidget->setHidden(true);
|
|
|
|
|
2020-06-06 09:54:57 -04:00
|
|
|
m_hideTimer.setInterval(clearFormsDelay);
|
|
|
|
m_hideTimer.setSingleShot(true);
|
|
|
|
connect(&m_hideTimer, &QTimer::timeout, this, [this] {
|
|
|
|
// Reset the password field after being hidden for a set time
|
|
|
|
m_ui->editPassword->setText("");
|
|
|
|
m_ui->editPassword->setShowPassword(false);
|
|
|
|
});
|
|
|
|
|
2019-06-22 12:00:31 -04:00
|
|
|
QFont font;
|
|
|
|
font.setPointSize(font.pointSize() + 4);
|
2012-07-02 12:47:12 -04:00
|
|
|
font.setBold(true);
|
|
|
|
m_ui->labelHeadline->setFont(font);
|
2019-06-22 12:00:31 -04:00
|
|
|
m_ui->labelHeadline->setText(tr("Unlock KeePassXC Database"));
|
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
m_ui->quickUnlockButton->setFont(font);
|
|
|
|
m_ui->quickUnlockButton->setIcon(
|
|
|
|
icons()->icon("fingerprint", true, palette().color(QPalette::Active, QPalette::HighlightedText)));
|
|
|
|
m_ui->quickUnlockButton->setIconSize({32, 32});
|
|
|
|
|
2011-12-24 13:19:52 -05:00
|
|
|
connect(m_ui->buttonBrowseFile, SIGNAL(clicked()), SLOT(browseKeyFile()));
|
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
auto okBtn = m_ui->buttonBox->button(QDialogButtonBox::Ok);
|
|
|
|
okBtn->setText(tr("Unlock"));
|
|
|
|
okBtn->setDefault(true);
|
2012-04-05 13:03:55 -04:00
|
|
|
connect(m_ui->buttonBox, SIGNAL(accepted()), SLOT(openDatabase()));
|
2011-12-24 13:19:52 -05:00
|
|
|
connect(m_ui->buttonBox, SIGNAL(rejected()), SLOT(reject()));
|
2017-01-14 20:08:48 -05:00
|
|
|
|
2020-10-05 20:41:00 -04:00
|
|
|
m_ui->hardwareKeyLabelHelp->setIcon(icons()->icon("system-help").pixmap(QSize(12, 12)));
|
2019-06-22 12:00:31 -04:00
|
|
|
connect(m_ui->hardwareKeyLabelHelp, SIGNAL(clicked(bool)), SLOT(openHardwareKeyHelp()));
|
2020-10-05 20:41:00 -04:00
|
|
|
m_ui->keyFileLabelHelp->setIcon(icons()->icon("system-help").pixmap(QSize(12, 12)));
|
2019-11-06 05:10:02 -05:00
|
|
|
connect(m_ui->keyFileLabelHelp, SIGNAL(clicked(bool)), SLOT(openKeyFileHelp()));
|
2019-06-22 12:00:31 -04:00
|
|
|
|
2017-02-20 19:06:32 -05:00
|
|
|
#ifdef WITH_XC_YUBIKEY
|
2020-04-06 08:42:20 -04:00
|
|
|
m_ui->hardwareKeyProgress->setVisible(false);
|
|
|
|
QSizePolicy sp = m_ui->hardwareKeyProgress->sizePolicy();
|
2017-02-20 19:06:32 -05:00
|
|
|
sp.setRetainSizeWhenHidden(true);
|
2020-04-06 08:42:20 -04:00
|
|
|
m_ui->hardwareKeyProgress->setSizePolicy(sp);
|
2017-02-20 19:06:32 -05:00
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
connect(m_ui->buttonRedetectYubikey, SIGNAL(clicked()), SLOT(pollHardwareKey()));
|
|
|
|
connect(YubiKey::instance(), SIGNAL(detectComplete(bool)), SLOT(hardwareKeyResponse(bool)), Qt::QueuedConnection);
|
|
|
|
|
|
|
|
connect(YubiKey::instance(), &YubiKey::userInteractionRequest, this, [this] {
|
|
|
|
// Show the press notification if we are in an independent window (e.g., DatabaseOpenDialog)
|
|
|
|
if (window() != getMainWindow()) {
|
Implement support for Yubikeys and potential other tokens via wireless NFC using smartcard readers (Rebase) (#6895)
* Support NFC readers for hardware tokens using PC/SC
This requires a new library dependency: PCSC.
The PCSC library provides methods to access smartcards. On Linux, the third-party pcsc-lite package is used. On Windows, the native Windows API (Winscard.dll) is used. On Mac OSX, the native OSX API (framework-PCSC) is used.
* Split hardware key access into multiple classes to handle different methods of communicating with the keys.
* Since the Yubikey can now be a wireless token as well, the verb "plug in" was replaced with a more
generic "interface with". This shall indicate that the user has to present their token to the reader, or plug it in via USB.
* Add PC/SC interface for YubiKey challenge-response
This new interface uses the PC/SC protocol and API
instead of the USB protocol via ykpers. Many YubiKeys expose their functionality as a CCID device, which can be interfaced with using PC/SC. This is especially useful for NFC-only or NFC-capable Yubikeys, when they are used together with a PC/SC compliant NFC reader device.
Although many (not all) Yubikeys expose their CCID functionality over their own USB connection as well, the HMAC-SHA1 functionality is often locked in this mode, as it requires eg. a touch on the gold button. When accessing the CCID functionality wirelessly via NFC (like this code can do using a reader), then the user interaction is to present the key to the reader.
This implementation has been tested on Linux using pcsc-lite, Windows using the native Winscard.dll library, and Mac OSX using the native PCSC-framework library.
* Remove PC/SC ATR whitelist, instead scan for AIDs
Before, a whitelist of ATR codes (answer to reset, hardware-specific)
was used to scan for compatible (Yubi)Keys.
Now, every connected smartcard is scanned for AIDs (applet identifier),
which are known to implement the HMAC-SHA1 protocol.
This enables the support of currently unknown or unreleased hardware.
Co-authored-by: Jonathan White <support@dmapps.us>
2021-10-01 10:39:07 -04:00
|
|
|
m_ui->messageWidget->showMessage(tr("Please present or touch your YubiKey to continue…"),
|
2020-04-06 08:42:20 -04:00
|
|
|
MessageWidget::Information,
|
|
|
|
MessageWidget::DisableAutoHide);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
connect(YubiKey::instance(), &YubiKey::challengeCompleted, this, [this] { m_ui->messageWidget->hide(); });
|
2017-02-20 19:06:32 -05:00
|
|
|
#else
|
2019-10-25 13:35:16 -04:00
|
|
|
m_ui->hardwareKeyLabel->setVisible(false);
|
|
|
|
m_ui->hardwareKeyLabelHelp->setVisible(false);
|
2017-02-20 19:06:32 -05:00
|
|
|
m_ui->buttonRedetectYubikey->setVisible(false);
|
2020-04-06 08:42:20 -04:00
|
|
|
m_ui->challengeResponseCombo->setVisible(false);
|
|
|
|
m_ui->hardwareKeyProgress->setVisible(false);
|
2017-02-20 19:06:32 -05:00
|
|
|
#endif
|
2017-02-20 14:24:38 -05:00
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
// QuickUnlock actions
|
|
|
|
connect(m_ui->quickUnlockButton, &QPushButton::pressed, this, [this] { openDatabase(); });
|
|
|
|
connect(m_ui->resetQuickUnlockButton, &QPushButton::pressed, this, [this] { resetQuickUnlock(); });
|
2012-06-28 03:21:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
DatabaseOpenWidget::~DatabaseOpenWidget()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-11-26 09:37:25 -05:00
|
|
|
void DatabaseOpenWidget::showEvent(QShowEvent* event)
|
|
|
|
{
|
|
|
|
DialogyWidget::showEvent(event);
|
2022-02-21 20:40:01 -05:00
|
|
|
if (isOnQuickUnlockScreen()) {
|
|
|
|
m_ui->quickUnlockButton->setFocus();
|
|
|
|
if (!canPerformQuickUnlock(m_filename)) {
|
|
|
|
resetQuickUnlock();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m_ui->editPassword->setFocus();
|
|
|
|
}
|
2020-06-06 09:54:57 -04:00
|
|
|
m_hideTimer.stop();
|
2016-11-26 09:37:25 -05:00
|
|
|
}
|
|
|
|
|
2017-10-07 21:25:42 -04:00
|
|
|
void DatabaseOpenWidget::hideEvent(QHideEvent* event)
|
|
|
|
{
|
|
|
|
DialogyWidget::hideEvent(event);
|
|
|
|
|
2020-06-06 09:54:57 -04:00
|
|
|
// Schedule form clearing if we are hidden
|
2020-04-06 08:42:20 -04:00
|
|
|
if (!isVisible()) {
|
2020-06-06 09:54:57 -04:00
|
|
|
m_hideTimer.start();
|
2018-11-23 07:49:55 -05:00
|
|
|
}
|
2017-10-07 21:25:42 -04:00
|
|
|
}
|
|
|
|
|
2022-06-12 16:35:42 -04:00
|
|
|
bool DatabaseOpenWidget::unlockingDatabase()
|
|
|
|
{
|
|
|
|
return m_unlockingDatabase;
|
|
|
|
}
|
|
|
|
|
2012-07-06 12:50:52 -04:00
|
|
|
void DatabaseOpenWidget::load(const QString& filename)
|
2012-06-28 03:21:15 -04:00
|
|
|
{
|
2020-06-06 09:54:57 -04:00
|
|
|
clearForms();
|
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
m_filename = filename;
|
2019-06-22 12:00:31 -04:00
|
|
|
m_ui->fileNameLabel->setRawText(m_filename);
|
2011-12-25 14:36:45 -05:00
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::RememberLastKeyFiles).toBool()) {
|
2020-04-06 08:42:20 -04:00
|
|
|
auto lastKeyFiles = config()->get(Config::LastKeyFiles).toHash();
|
2015-03-14 23:06:53 -04:00
|
|
|
if (lastKeyFiles.contains(m_filename)) {
|
2020-06-06 09:54:57 -04:00
|
|
|
m_ui->keyFileLineEdit->setText(lastKeyFiles[m_filename].toString());
|
2015-03-14 23:06:53 -04:00
|
|
|
}
|
2011-12-25 14:36:45 -05:00
|
|
|
}
|
2012-04-22 14:57:42 -04:00
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
if (canPerformQuickUnlock(m_filename)) {
|
|
|
|
m_ui->centralStack->setCurrentIndex(1);
|
|
|
|
m_ui->quickUnlockButton->setFocus();
|
|
|
|
} else {
|
|
|
|
m_ui->editPassword->setFocus();
|
|
|
|
}
|
2020-04-06 08:42:20 -04:00
|
|
|
|
|
|
|
#ifdef WITH_XC_YUBIKEY
|
|
|
|
// Only auto-poll for hardware keys if we previously used one with this database file
|
|
|
|
if (config()->get(Config::RememberLastKeyFiles).toBool()) {
|
|
|
|
auto lastChallengeResponse = config()->get(Config::LastChallengeResponse).toHash();
|
|
|
|
if (lastChallengeResponse.contains(m_filename)) {
|
|
|
|
pollHardwareKey();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2011-11-13 08:55:20 -05:00
|
|
|
}
|
|
|
|
|
2017-09-23 11:43:29 -04:00
|
|
|
void DatabaseOpenWidget::clearForms()
|
|
|
|
{
|
2022-02-21 20:40:01 -05:00
|
|
|
setUserInteractionLock(false);
|
2020-06-06 09:54:57 -04:00
|
|
|
m_ui->editPassword->setText("");
|
|
|
|
m_ui->editPassword->setShowPassword(false);
|
|
|
|
m_ui->keyFileLineEdit->clear();
|
2020-12-04 08:01:36 -05:00
|
|
|
m_ui->keyFileLineEdit->setShowPassword(false);
|
2020-06-06 09:54:57 -04:00
|
|
|
m_ui->challengeResponseCombo->clear();
|
2022-02-21 20:40:01 -05:00
|
|
|
m_ui->centralStack->setCurrentIndex(0);
|
2020-06-06 09:54:57 -04:00
|
|
|
m_db.reset();
|
2017-09-23 11:43:29 -04:00
|
|
|
}
|
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
QSharedPointer<Database> DatabaseOpenWidget::database()
|
2011-11-13 08:55:20 -05:00
|
|
|
{
|
2012-04-05 13:03:55 -04:00
|
|
|
return m_db;
|
2011-11-13 08:55:20 -05:00
|
|
|
}
|
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
QString DatabaseOpenWidget::filename()
|
|
|
|
{
|
|
|
|
return m_filename;
|
|
|
|
}
|
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
void DatabaseOpenWidget::enterKey(const QString& pw, const QString& keyFile)
|
2012-04-24 05:47:16 -04:00
|
|
|
{
|
2019-03-26 22:23:16 -04:00
|
|
|
m_ui->editPassword->setText(pw);
|
2020-06-06 09:54:57 -04:00
|
|
|
m_ui->keyFileLineEdit->setText(keyFile);
|
2022-02-21 20:40:01 -05:00
|
|
|
m_blockQuickUnlock = true;
|
2012-04-24 05:47:16 -04:00
|
|
|
openDatabase();
|
|
|
|
}
|
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
void DatabaseOpenWidget::openDatabase()
|
2011-11-13 08:55:20 -05:00
|
|
|
{
|
2022-02-21 20:40:01 -05:00
|
|
|
// Cache this variable for future use then reset
|
|
|
|
bool blockQuickUnlock = m_blockQuickUnlock || isOnQuickUnlockScreen();
|
|
|
|
m_blockQuickUnlock = false;
|
|
|
|
|
|
|
|
setUserInteractionLock(true);
|
2022-03-29 15:31:54 -04:00
|
|
|
m_ui->editPassword->setShowPassword(false);
|
2020-04-06 08:42:20 -04:00
|
|
|
m_ui->messageWidget->hide();
|
2022-02-21 20:40:01 -05:00
|
|
|
QCoreApplication::processEvents();
|
2020-04-06 08:42:20 -04:00
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
const auto databaseKey = buildDatabaseKey();
|
2020-07-01 19:16:40 -04:00
|
|
|
if (!databaseKey) {
|
2022-02-21 20:40:01 -05:00
|
|
|
setUserInteractionLock(false);
|
2017-10-05 14:44:05 -04:00
|
|
|
return;
|
|
|
|
}
|
2012-10-12 06:12:00 -04:00
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
QString error;
|
2022-02-21 20:40:01 -05:00
|
|
|
m_db.reset(new Database());
|
2022-01-28 21:26:53 -05:00
|
|
|
bool ok = m_db->open(m_filename, databaseKey, &error);
|
2021-11-19 18:32:09 -05:00
|
|
|
|
2020-01-20 09:47:02 -05:00
|
|
|
if (ok) {
|
2022-02-21 20:40:01 -05:00
|
|
|
// Warn user about minor version mismatch to halt loading if necessary
|
|
|
|
if (m_db->hasMinorVersionMismatch()) {
|
|
|
|
QScopedPointer<QMessageBox> msgBox(new QMessageBox(this));
|
|
|
|
msgBox->setIcon(QMessageBox::Warning);
|
|
|
|
msgBox->setWindowTitle(tr("Database Version Mismatch"));
|
|
|
|
msgBox->setText(tr("The database you are trying to open was most likely\n"
|
|
|
|
"created by a newer version of KeePassXC.\n\n"
|
|
|
|
"You can try to open it anyway, but it may be incomplete\n"
|
|
|
|
"and saving any changes may incur data loss.\n\n"
|
|
|
|
"We recommend you update your KeePassXC installation."));
|
|
|
|
auto btn = msgBox->addButton(tr("Open database anyway"), QMessageBox::ButtonRole::AcceptRole);
|
|
|
|
msgBox->setDefaultButton(btn);
|
|
|
|
msgBox->addButton(QMessageBox::Cancel);
|
|
|
|
msgBox->exec();
|
|
|
|
if (msgBox->clickedButton() != btn) {
|
|
|
|
m_db.reset(new Database());
|
|
|
|
m_ui->messageWidget->showMessage(tr("Database unlock canceled."), MessageWidget::MessageType::Error);
|
|
|
|
setUserInteractionLock(false);
|
|
|
|
return;
|
2020-01-20 09:47:02 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
// Save Quick Unlock credentials if available
|
|
|
|
if (!blockQuickUnlock && isQuickUnlockAvailable()) {
|
|
|
|
auto keyData = databaseKey->serialize();
|
|
|
|
#if defined(Q_CC_MSVC)
|
|
|
|
// Store the password using Windows Hello
|
|
|
|
getWindowsHello()->storeKey(m_filename, keyData);
|
|
|
|
#elif defined(Q_OS_MACOS)
|
|
|
|
// Store the password using TouchID
|
|
|
|
TouchID::getInstance().storeKey(m_filename, keyData);
|
2020-01-20 09:47:02 -05:00
|
|
|
#endif
|
2022-02-21 20:40:01 -05:00
|
|
|
m_ui->messageWidget->hideMessage();
|
|
|
|
}
|
|
|
|
|
2020-01-20 09:47:02 -05:00
|
|
|
emit dialogFinished(true);
|
|
|
|
clearForms();
|
|
|
|
} else {
|
2022-02-21 20:40:01 -05:00
|
|
|
if (!isOnQuickUnlockScreen() && m_ui->editPassword->text().isEmpty() && !m_retryUnlockWithEmptyPassword) {
|
2019-06-22 12:00:31 -04:00
|
|
|
QScopedPointer<QMessageBox> msgBox(new QMessageBox(this));
|
|
|
|
msgBox->setIcon(QMessageBox::Critical);
|
|
|
|
msgBox->setWindowTitle(tr("Unlock failed and no password given"));
|
|
|
|
msgBox->setText(tr("Unlocking the database failed and you did not enter a password.\n"
|
|
|
|
"Do you want to retry with an \"empty\" password instead?\n\n"
|
|
|
|
"To prevent this error from appearing, you must go to "
|
|
|
|
"\"Database Settings / Security\" and reset your password."));
|
|
|
|
auto btn = msgBox->addButton(tr("Retry with empty password"), QMessageBox::ButtonRole::AcceptRole);
|
|
|
|
msgBox->setDefaultButton(btn);
|
|
|
|
msgBox->addButton(QMessageBox::Cancel);
|
|
|
|
msgBox->exec();
|
|
|
|
|
|
|
|
if (msgBox->clickedButton() == btn) {
|
|
|
|
m_retryUnlockWithEmptyPassword = true;
|
2022-02-21 20:40:01 -05:00
|
|
|
setUserInteractionLock(false);
|
2019-06-22 12:00:31 -04:00
|
|
|
openDatabase();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2020-01-20 09:47:02 -05:00
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
setUserInteractionLock(false);
|
|
|
|
|
2019-06-22 12:00:31 -04:00
|
|
|
m_retryUnlockWithEmptyPassword = false;
|
2019-04-05 08:31:37 -04:00
|
|
|
m_ui->messageWidget->showMessage(error, MessageWidget::MessageType::Error);
|
2022-03-10 16:13:13 -05:00
|
|
|
|
|
|
|
if (!isOnQuickUnlockScreen()) {
|
|
|
|
// Focus on the password field and select the input for easy retry
|
|
|
|
m_ui->editPassword->selectAll();
|
|
|
|
m_ui->editPassword->setFocus();
|
|
|
|
}
|
2012-10-12 06:12:00 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
QSharedPointer<CompositeKey> DatabaseOpenWidget::buildDatabaseKey()
|
2012-10-12 06:12:00 -04:00
|
|
|
{
|
2020-07-01 19:16:40 -04:00
|
|
|
auto databaseKey = QSharedPointer<CompositeKey>::create();
|
2012-04-05 13:03:55 -04:00
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
if (canPerformQuickUnlock(m_filename)) {
|
|
|
|
// try to retrieve the stored password using Windows Hello
|
|
|
|
QByteArray keyData;
|
|
|
|
#ifdef Q_CC_MSVC
|
|
|
|
if (!getWindowsHello()->getKey(m_filename, keyData)) {
|
|
|
|
// Failed to retrieve Quick Unlock data
|
|
|
|
m_ui->messageWidget->showMessage(tr("Failed to authenticate with Windows Hello"), MessageWidget::Error);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
#elif defined(Q_OS_MACOS)
|
|
|
|
if (!TouchID::getInstance().getKey(m_filename, keyData)) {
|
|
|
|
// Failed to retrieve Quick Unlock data
|
|
|
|
m_ui->messageWidget->showMessage(tr("Failed to authenticate with Touch ID"), MessageWidget::Error);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
databaseKey->setRawKey(keyData);
|
|
|
|
return databaseKey;
|
2011-11-13 08:55:20 -05:00
|
|
|
}
|
|
|
|
|
2022-02-21 20:40:01 -05:00
|
|
|
if (!m_ui->editPassword->text().isEmpty() || m_retryUnlockWithEmptyPassword) {
|
|
|
|
databaseKey->addKey(QSharedPointer<PasswordKey>::create(m_ui->editPassword->text()));
|
2018-04-04 11:39:26 -04:00
|
|
|
}
|
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
auto lastKeyFiles = config()->get(Config::LastKeyFiles).toHash();
|
2019-06-22 12:00:31 -04:00
|
|
|
lastKeyFiles.remove(m_filename);
|
2011-12-25 14:36:45 -05:00
|
|
|
|
2019-06-22 12:00:31 -04:00
|
|
|
auto key = QSharedPointer<FileKey>::create();
|
2020-06-06 09:54:57 -04:00
|
|
|
QString keyFilename = m_ui->keyFileLineEdit->text();
|
|
|
|
if (!keyFilename.isEmpty()) {
|
2011-12-24 13:19:52 -05:00
|
|
|
QString errorMsg;
|
2018-05-13 17:21:43 -04:00
|
|
|
if (!key->load(keyFilename, &errorMsg)) {
|
2019-04-05 08:31:37 -04:00
|
|
|
m_ui->messageWidget->showMessage(tr("Failed to open key file: %1").arg(errorMsg), MessageWidget::Error);
|
2018-05-13 17:21:43 -04:00
|
|
|
return {};
|
2011-12-24 13:19:52 -05:00
|
|
|
}
|
2020-12-09 19:28:01 -05:00
|
|
|
if (key->type() != FileKey::KeePass2XMLv2 && key->type() != FileKey::Hashed
|
|
|
|
&& !config()->get(Config::Messages_NoLegacyKeyFileWarning).toBool()) {
|
2017-12-27 08:20:28 -05:00
|
|
|
QMessageBox legacyWarning;
|
2020-12-09 19:28:01 -05:00
|
|
|
legacyWarning.setWindowTitle(tr("Old key file format"));
|
|
|
|
legacyWarning.setText(tr("You are using an old key file format which KeePassXC may<br>"
|
|
|
|
"stop supporting in the future.<br><br>"
|
|
|
|
"Please consider generating a new key file by going to:<br>"
|
2021-02-14 19:15:10 -05:00
|
|
|
"<strong>Database > Database Security > Change Key File.</strong><br>"));
|
2017-12-27 08:20:28 -05:00
|
|
|
legacyWarning.setIcon(QMessageBox::Icon::Warning);
|
|
|
|
legacyWarning.addButton(QMessageBox::Ok);
|
|
|
|
legacyWarning.setDefaultButton(QMessageBox::Ok);
|
|
|
|
legacyWarning.setCheckBox(new QCheckBox(tr("Don't show this warning again")));
|
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
connect(legacyWarning.checkBox(), &QCheckBox::stateChanged, this, [](int state) {
|
2020-04-25 19:31:38 -04:00
|
|
|
config()->set(Config::Messages_NoLegacyKeyFileWarning, state == Qt::CheckState::Checked);
|
2017-12-27 08:20:28 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
legacyWarning.exec();
|
|
|
|
}
|
2020-07-01 19:16:40 -04:00
|
|
|
databaseKey->addKey(key);
|
2020-04-06 08:42:20 -04:00
|
|
|
lastKeyFiles.insert(m_filename, keyFilename);
|
2017-02-20 16:07:01 -05:00
|
|
|
}
|
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::RememberLastKeyFiles).toBool()) {
|
|
|
|
config()->set(Config::LastKeyFiles, lastKeyFiles);
|
2015-03-14 23:06:53 -04:00
|
|
|
}
|
2011-12-24 13:19:52 -05:00
|
|
|
|
2017-02-20 19:06:32 -05:00
|
|
|
#ifdef WITH_XC_YUBIKEY
|
2020-04-06 08:42:20 -04:00
|
|
|
auto lastChallengeResponse = config()->get(Config::LastChallengeResponse).toHash();
|
2019-06-22 12:00:31 -04:00
|
|
|
lastChallengeResponse.remove(m_filename);
|
2014-05-26 03:49:28 -04:00
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
int selectionIndex = m_ui->challengeResponseCombo->currentIndex();
|
2019-06-22 12:00:31 -04:00
|
|
|
if (selectionIndex > 0) {
|
2020-04-06 08:42:20 -04:00
|
|
|
auto slot = m_ui->challengeResponseCombo->itemData(selectionIndex).value<YubiKeySlot>();
|
2021-04-22 23:07:49 -04:00
|
|
|
auto crKey = QSharedPointer<ChallengeResponseKey>(new ChallengeResponseKey(slot));
|
2020-07-01 19:16:40 -04:00
|
|
|
databaseKey->addChallengeResponseKey(crKey);
|
2020-04-06 08:42:20 -04:00
|
|
|
|
|
|
|
// Qt doesn't read custom types in settings so stuff into a QString
|
|
|
|
lastChallengeResponse.insert(m_filename, QStringLiteral("%1:%2").arg(slot.first).arg(slot.second));
|
2019-06-22 12:00:31 -04:00
|
|
|
}
|
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::RememberLastKeyFiles).toBool()) {
|
|
|
|
config()->set(Config::LastChallengeResponse, lastChallengeResponse);
|
2014-05-26 03:49:28 -04:00
|
|
|
}
|
2017-02-20 19:06:32 -05:00
|
|
|
#endif
|
2014-05-26 03:49:28 -04:00
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
return databaseKey;
|
2011-11-13 08:55:20 -05:00
|
|
|
}
|
2011-11-16 12:46:09 -05:00
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
void DatabaseOpenWidget::reject()
|
|
|
|
{
|
2018-11-22 05:47:31 -05:00
|
|
|
emit dialogFinished(false);
|
2012-06-28 03:21:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void DatabaseOpenWidget::browseKeyFile()
|
2011-12-24 13:19:52 -05:00
|
|
|
{
|
2020-12-09 19:28:01 -05:00
|
|
|
QString filters = QString("%1 (*);;%2 (*.keyx; *.key)").arg(tr("All files"), tr("Key files"));
|
2011-12-25 19:21:29 -05:00
|
|
|
QString filename = fileDialog()->getOpenFileName(this, tr("Select key file"), QString(), filters);
|
2011-12-24 13:19:52 -05:00
|
|
|
|
2019-11-06 05:10:02 -05:00
|
|
|
if (QFileInfo(filename).canonicalFilePath() == QFileInfo(m_filename).canonicalFilePath()) {
|
2019-11-18 01:57:04 -05:00
|
|
|
MessageBox::warning(this,
|
|
|
|
tr("Cannot use database file as key file"),
|
|
|
|
tr("You cannot use your database file as a key file.\nIf you do not have a key file, "
|
|
|
|
"please leave the field empty."),
|
|
|
|
MessageBox::Button::Ok);
|
2019-11-06 05:10:02 -05:00
|
|
|
filename = "";
|
|
|
|
}
|
|
|
|
|
2011-12-24 13:19:52 -05:00
|
|
|
if (!filename.isEmpty()) {
|
2020-06-06 09:54:57 -04:00
|
|
|
m_ui->keyFileLineEdit->setText(filename);
|
2011-12-24 13:19:52 -05:00
|
|
|
}
|
|
|
|
}
|
2014-05-26 03:49:28 -04:00
|
|
|
|
2020-06-06 09:54:57 -04:00
|
|
|
void DatabaseOpenWidget::clearKeyFileText()
|
2019-06-22 12:00:31 -04:00
|
|
|
{
|
2020-06-06 09:54:57 -04:00
|
|
|
m_ui->keyFileLineEdit->clear();
|
2020-12-04 08:01:36 -05:00
|
|
|
m_ui->keyFileLineEdit->setShowPassword(false);
|
2019-06-22 12:00:31 -04:00
|
|
|
}
|
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
void DatabaseOpenWidget::pollHardwareKey()
|
2017-02-20 16:07:01 -05:00
|
|
|
{
|
2020-04-06 08:42:20 -04:00
|
|
|
if (m_pollingHardwareKey) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_ui->challengeResponseCombo->clear();
|
|
|
|
m_ui->challengeResponseCombo->addItem(tr("Detecting hardware keys…"));
|
|
|
|
|
2017-02-20 16:07:01 -05:00
|
|
|
m_ui->buttonRedetectYubikey->setEnabled(false);
|
2020-04-06 08:42:20 -04:00
|
|
|
m_ui->challengeResponseCombo->setEnabled(false);
|
|
|
|
m_ui->hardwareKeyProgress->setVisible(true);
|
|
|
|
m_pollingHardwareKey = true;
|
2017-02-23 17:52:36 -05:00
|
|
|
|
2022-03-22 15:12:52 -04:00
|
|
|
YubiKey::instance()->findValidKeysAsync();
|
2017-02-20 16:07:01 -05:00
|
|
|
}
|
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
void DatabaseOpenWidget::hardwareKeyResponse(bool found)
|
2014-05-26 03:49:28 -04:00
|
|
|
{
|
2020-04-06 08:42:20 -04:00
|
|
|
m_ui->challengeResponseCombo->clear();
|
|
|
|
m_ui->buttonRedetectYubikey->setEnabled(true);
|
|
|
|
m_ui->hardwareKeyProgress->setVisible(false);
|
|
|
|
m_pollingHardwareKey = false;
|
2017-02-20 16:07:01 -05:00
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
if (!found) {
|
|
|
|
m_ui->challengeResponseCombo->addItem(tr("No hardware keys detected"));
|
|
|
|
m_ui->challengeResponseCombo->setEnabled(false);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
m_ui->challengeResponseCombo->addItem(tr("Select hardware key…"));
|
|
|
|
}
|
|
|
|
|
|
|
|
YubiKeySlot lastUsedSlot;
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::RememberLastKeyFiles).toBool()) {
|
2020-04-06 08:42:20 -04:00
|
|
|
auto lastChallengeResponse = config()->get(Config::LastChallengeResponse).toHash();
|
2017-02-20 16:07:01 -05:00
|
|
|
if (lastChallengeResponse.contains(m_filename)) {
|
2020-04-06 08:42:20 -04:00
|
|
|
// Qt doesn't read custom types in settings so extract from QString
|
|
|
|
auto split = lastChallengeResponse.value(m_filename).toString().split(":");
|
|
|
|
if (split.size() > 1) {
|
|
|
|
lastUsedSlot = YubiKeySlot(split[0].toUInt(), split[1].toInt());
|
|
|
|
}
|
2017-02-20 16:07:01 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
int selectedIndex = 0;
|
|
|
|
for (auto& slot : YubiKey::instance()->foundKeys()) {
|
|
|
|
// add detected YubiKey to combo box
|
|
|
|
m_ui->challengeResponseCombo->addItem(YubiKey::instance()->getDisplayName(slot), QVariant::fromValue(slot));
|
|
|
|
// Select this YubiKey + Slot if we used it in the past
|
|
|
|
if (lastUsedSlot == slot) {
|
|
|
|
selectedIndex = m_ui->challengeResponseCombo->count() - 1;
|
|
|
|
}
|
|
|
|
}
|
2017-10-07 21:25:42 -04:00
|
|
|
|
2020-04-06 08:42:20 -04:00
|
|
|
m_ui->challengeResponseCombo->setCurrentIndex(selectedIndex);
|
|
|
|
m_ui->challengeResponseCombo->setEnabled(true);
|
2014-05-26 03:49:28 -04:00
|
|
|
}
|
2019-06-22 12:00:31 -04:00
|
|
|
|
|
|
|
void DatabaseOpenWidget::openHardwareKeyHelp()
|
|
|
|
{
|
2019-11-06 05:10:02 -05:00
|
|
|
QDesktopServices::openUrl(QUrl("https://keepassxc.org/docs#faq-cat-yubikey"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void DatabaseOpenWidget::openKeyFileHelp()
|
|
|
|
{
|
|
|
|
QDesktopServices::openUrl(QUrl("https://keepassxc.org/docs#faq-cat-keyfile"));
|
|
|
|
}
|
2022-02-21 20:40:01 -05:00
|
|
|
|
|
|
|
void DatabaseOpenWidget::setUserInteractionLock(bool state)
|
|
|
|
{
|
|
|
|
if (state) {
|
|
|
|
QApplication::setOverrideCursor(QCursor(Qt::WaitCursor));
|
|
|
|
m_ui->centralStack->setEnabled(false);
|
|
|
|
} else {
|
|
|
|
// Ensure no override cursors remain
|
|
|
|
while (QApplication::overrideCursor()) {
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
|
|
}
|
|
|
|
m_ui->centralStack->setEnabled(true);
|
|
|
|
}
|
2022-06-12 16:35:42 -04:00
|
|
|
m_unlockingDatabase = state;
|
2022-02-21 20:40:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DatabaseOpenWidget::isOnQuickUnlockScreen()
|
|
|
|
{
|
|
|
|
return m_ui->centralStack->currentIndex() == 1;
|
|
|
|
}
|
|
|
|
|
2022-03-29 18:03:35 -04:00
|
|
|
/**
|
|
|
|
* Reset installed quick unlock secrets.
|
|
|
|
*
|
|
|
|
* It's safe to call this method even if quick unlock is unavailable.
|
|
|
|
*/
|
2022-02-21 20:40:01 -05:00
|
|
|
void DatabaseOpenWidget::resetQuickUnlock()
|
|
|
|
{
|
2022-03-29 18:03:35 -04:00
|
|
|
if (!isQuickUnlockAvailable()) {
|
|
|
|
return;
|
|
|
|
}
|
2022-02-21 20:40:01 -05:00
|
|
|
#if defined(Q_CC_MSVC)
|
|
|
|
getWindowsHello()->reset(m_filename);
|
|
|
|
#elif defined(Q_OS_MACOS)
|
|
|
|
TouchID::getInstance().reset(m_filename);
|
|
|
|
#endif
|
|
|
|
load(m_filename);
|
|
|
|
}
|