2011-11-13 08:55:20 -05:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2011 Felix Geyer <debfx@fobos.de>
|
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2014-05-26 03:49:28 -04:00
|
|
|
#include <QtConcurrentRun>
|
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
#include "DatabaseOpenWidget.h"
|
|
|
|
#include "ui_DatabaseOpenWidget.h"
|
2011-11-13 08:55:20 -05:00
|
|
|
|
2011-12-25 14:36:45 -05:00
|
|
|
#include "core/Config.h"
|
2012-06-28 03:21:15 -04:00
|
|
|
#include "core/Database.h"
|
2014-01-12 11:08:36 -05:00
|
|
|
#include "core/FilePath.h"
|
2015-10-08 14:30:03 -04:00
|
|
|
#include "gui/MainWindow.h"
|
2011-12-25 19:21:29 -05:00
|
|
|
#include "gui/FileDialog.h"
|
2013-10-08 16:09:20 -04:00
|
|
|
#include "gui/MessageBox.h"
|
2012-04-05 13:03:55 -04:00
|
|
|
#include "format/KeePass2Reader.h"
|
2011-12-24 13:19:52 -05:00
|
|
|
#include "keys/FileKey.h"
|
2011-11-13 08:55:20 -05:00
|
|
|
#include "keys/PasswordKey.h"
|
2014-05-26 03:49:28 -04:00
|
|
|
#include "keys/YkChallengeResponseKey.h"
|
|
|
|
#include "crypto/Random.h"
|
|
|
|
|
2011-11-13 08:55:20 -05:00
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
DatabaseOpenWidget::DatabaseOpenWidget(QWidget* parent)
|
|
|
|
: DialogyWidget(parent)
|
|
|
|
, m_ui(new Ui::DatabaseOpenWidget())
|
2015-07-24 12:28:12 -04:00
|
|
|
, 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);
|
|
|
|
|
2012-07-02 12:47:12 -04:00
|
|
|
QFont font = m_ui->labelHeadline->font();
|
|
|
|
font.setBold(true);
|
|
|
|
font.setPointSize(font.pointSize() + 2);
|
|
|
|
m_ui->labelHeadline->setFont(font);
|
|
|
|
|
2011-12-24 13:19:52 -05:00
|
|
|
m_ui->buttonBox->button(QDialogButtonBox::Ok)->setEnabled(false);
|
|
|
|
|
2014-01-12 11:08:36 -05:00
|
|
|
m_ui->buttonTogglePassword->setIcon(filePath()->onOffIcon("actions", "password-show"));
|
|
|
|
connect(m_ui->buttonTogglePassword, SIGNAL(toggled(bool)),
|
|
|
|
m_ui->editPassword, SLOT(setShowPassword(bool)));
|
2011-12-24 13:19:52 -05:00
|
|
|
connect(m_ui->buttonBrowseFile, SIGNAL(clicked()), SLOT(browseKeyFile()));
|
|
|
|
|
2011-11-16 12:46:09 -05:00
|
|
|
connect(m_ui->editPassword, SIGNAL(textChanged(QString)), SLOT(activatePassword()));
|
2011-12-24 13:19:52 -05:00
|
|
|
connect(m_ui->comboKeyFile, SIGNAL(editTextChanged(QString)), SLOT(activateKeyFile()));
|
2014-05-26 03:49:28 -04:00
|
|
|
connect(m_ui->comboChallengeResponse, SIGNAL(activated(int)), SLOT(activateChallengeResponse()));
|
|
|
|
|
|
|
|
connect(m_ui->checkPassword, SIGNAL(toggled(bool)), SLOT(setOkButtonEnabled()));
|
|
|
|
connect(m_ui->checkKeyFile, SIGNAL(toggled(bool)), SLOT(setOkButtonEnabled()));
|
|
|
|
connect(m_ui->comboKeyFile, SIGNAL(editTextChanged(QString)), SLOT(setOkButtonEnabled()));
|
|
|
|
connect(m_ui->checkChallengeResponse, SIGNAL(toggled(bool)), SLOT(setOkButtonEnabled()));
|
|
|
|
connect(m_ui->comboChallengeResponse, SIGNAL(activated(int)), SLOT(setOkButtonEnabled()));
|
2011-12-24 13:19:52 -05:00
|
|
|
|
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
|
|
|
|
2017-02-20 16:07:01 -05:00
|
|
|
connect(m_ui->buttonRedetectYubikey, SIGNAL(clicked()), SLOT(pollYubikey()));
|
|
|
|
|
|
|
|
connect(YubiKey::instance(), SIGNAL(detected(int,bool)), SLOT(yubikeyDetected(int,bool)), Qt::QueuedConnection);
|
|
|
|
connect(YubiKey::instance(), SIGNAL(notFound()), SLOT(noYubikeyFound()), Qt::QueuedConnection);
|
2017-02-20 14:24:38 -05:00
|
|
|
|
2017-02-12 10:57:01 -05:00
|
|
|
#ifdef Q_OS_MACOS
|
|
|
|
// add random padding to layouts to align widgets properly
|
|
|
|
m_ui->dialogButtonsLayout->setContentsMargins(10, 0, 15, 0);
|
|
|
|
m_ui->gridLayout->setContentsMargins(10, 0, 0, 0);
|
|
|
|
m_ui->labelLayout->setContentsMargins(10, 0, 10, 0);
|
|
|
|
#endif
|
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);
|
|
|
|
m_ui->editPassword->setFocus();
|
|
|
|
}
|
|
|
|
|
2012-07-06 12:50:52 -04:00
|
|
|
void DatabaseOpenWidget::load(const QString& filename)
|
2012-06-28 03:21:15 -04:00
|
|
|
{
|
|
|
|
m_filename = filename;
|
2011-12-25 14:36:45 -05:00
|
|
|
|
2012-07-02 12:47:12 -04:00
|
|
|
m_ui->labelFilename->setText(filename);
|
|
|
|
|
2015-04-14 17:10:37 -04:00
|
|
|
if (config()->get("RememberLastKeyFiles").toBool()) {
|
2015-03-14 23:06:53 -04:00
|
|
|
QHash<QString, QVariant> lastKeyFiles = config()->get("LastKeyFiles").toHash();
|
|
|
|
if (lastKeyFiles.contains(m_filename)) {
|
|
|
|
m_ui->checkKeyFile->setChecked(true);
|
|
|
|
m_ui->comboKeyFile->addItem(lastKeyFiles[m_filename].toString());
|
|
|
|
}
|
2011-12-25 14:36:45 -05:00
|
|
|
}
|
2012-04-22 14:57:42 -04:00
|
|
|
|
2016-01-31 10:17:24 -05:00
|
|
|
m_ui->buttonBox->button(QDialogButtonBox::Ok)->setEnabled(true);
|
2014-05-26 03:49:28 -04:00
|
|
|
|
2017-01-14 20:08:48 -05:00
|
|
|
m_ui->comboChallengeResponse->clear();
|
2017-02-20 16:07:01 -05:00
|
|
|
pollYubikey();
|
2014-05-26 03:49:28 -04:00
|
|
|
|
2012-04-22 14:57:42 -04:00
|
|
|
m_ui->editPassword->setFocus();
|
2011-11-13 08:55:20 -05:00
|
|
|
}
|
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
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
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
|
|
|
if (!pw.isNull()) {
|
|
|
|
m_ui->editPassword->setText(pw);
|
|
|
|
}
|
|
|
|
if (!keyFile.isEmpty()) {
|
2014-12-01 16:45:11 -05:00
|
|
|
m_ui->comboKeyFile->setEditText(keyFile);
|
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
|
|
|
{
|
2012-04-05 13:03:55 -04:00
|
|
|
KeePass2Reader reader;
|
2012-10-12 06:12:00 -04:00
|
|
|
CompositeKey masterKey = databaseKey();
|
|
|
|
|
|
|
|
QFile file(m_filename);
|
|
|
|
if (!file.open(QIODevice::ReadOnly)) {
|
2015-01-21 11:19:05 -05:00
|
|
|
m_ui->messageWidget->showMessage(
|
|
|
|
tr("Unable to open the database.").append("\n").append(file.errorString()), MessageWidget::Error);
|
2012-10-12 06:12:00 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (m_db) {
|
|
|
|
delete m_db;
|
|
|
|
}
|
|
|
|
QApplication::setOverrideCursor(QCursor(Qt::WaitCursor));
|
|
|
|
m_db = reader.readDatabase(&file, masterKey);
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
|
|
|
|
|
|
if (m_db) {
|
2015-01-19 18:14:59 -05:00
|
|
|
if (m_ui->messageWidget->isVisible()) {
|
|
|
|
m_ui->messageWidget->animatedHide();
|
|
|
|
}
|
2012-10-12 06:12:00 -04:00
|
|
|
Q_EMIT editFinished(true);
|
|
|
|
}
|
|
|
|
else {
|
2015-01-21 11:19:05 -05:00
|
|
|
m_ui->messageWidget->showMessage(tr("Unable to open the database.")
|
|
|
|
.append("\n").append(reader.errorString()), MessageWidget::Error);
|
2012-10-12 06:12:00 -04:00
|
|
|
m_ui->editPassword->clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CompositeKey DatabaseOpenWidget::databaseKey()
|
|
|
|
{
|
2012-04-05 13:03:55 -04:00
|
|
|
CompositeKey masterKey;
|
|
|
|
|
2011-11-13 08:55:20 -05:00
|
|
|
if (m_ui->checkPassword->isChecked()) {
|
2012-04-05 13:03:55 -04:00
|
|
|
masterKey.addKey(PasswordKey(m_ui->editPassword->text()));
|
2011-11-13 08:55:20 -05:00
|
|
|
}
|
|
|
|
|
2012-04-18 18:25:57 -04:00
|
|
|
QHash<QString, QVariant> lastKeyFiles = config()->get("LastKeyFiles").toHash();
|
2017-02-20 16:07:01 -05:00
|
|
|
QHash<QString, QVariant> lastChallengeResponse = config()->get("LastChallengeResponse").toHash();
|
2011-12-25 14:36:45 -05:00
|
|
|
|
2011-11-13 08:55:20 -05:00
|
|
|
if (m_ui->checkKeyFile->isChecked()) {
|
2011-12-24 13:19:52 -05:00
|
|
|
FileKey key;
|
2011-12-25 14:36:45 -05:00
|
|
|
QString keyFilename = m_ui->comboKeyFile->currentText();
|
2011-12-24 13:19:52 -05:00
|
|
|
QString errorMsg;
|
2011-12-25 14:36:45 -05:00
|
|
|
if (!key.load(keyFilename, &errorMsg)) {
|
2015-01-21 11:19:05 -05:00
|
|
|
m_ui->messageWidget->showMessage(tr("Can't open key file").append(":\n")
|
|
|
|
.append(errorMsg), MessageWidget::Error);
|
2012-10-12 06:12:00 -04:00
|
|
|
return CompositeKey();
|
2011-12-24 13:19:52 -05:00
|
|
|
}
|
2012-04-05 13:03:55 -04:00
|
|
|
masterKey.addKey(key);
|
2011-12-25 14:36:45 -05:00
|
|
|
lastKeyFiles[m_filename] = keyFilename;
|
2017-02-20 16:07:01 -05:00
|
|
|
} else {
|
2011-12-25 14:36:45 -05:00
|
|
|
lastKeyFiles.remove(m_filename);
|
|
|
|
}
|
|
|
|
|
2017-02-20 16:07:01 -05:00
|
|
|
if (m_ui->checkChallengeResponse->isChecked()) {
|
|
|
|
lastChallengeResponse[m_filename] = true;
|
|
|
|
} else {
|
|
|
|
lastChallengeResponse.remove(m_filename);
|
|
|
|
}
|
|
|
|
|
2015-04-14 17:10:37 -04:00
|
|
|
if (config()->get("RememberLastKeyFiles").toBool()) {
|
2015-03-14 23:06:53 -04:00
|
|
|
config()->set("LastKeyFiles", lastKeyFiles);
|
2017-02-20 16:07:01 -05:00
|
|
|
config()->set("LastChallengeResponse", lastChallengeResponse);
|
2015-03-14 23:06:53 -04:00
|
|
|
}
|
2011-12-24 13:19:52 -05:00
|
|
|
|
2014-05-26 03:49:28 -04:00
|
|
|
|
|
|
|
if (m_ui->checkChallengeResponse->isChecked()) {
|
|
|
|
int i = m_ui->comboChallengeResponse->currentIndex();
|
|
|
|
i = m_ui->comboChallengeResponse->itemData(i).toInt();
|
|
|
|
YkChallengeResponseKey key(i);
|
|
|
|
|
|
|
|
masterKey.addChallengeResponseKey(key);
|
|
|
|
}
|
|
|
|
|
2012-10-12 06:12:00 -04:00
|
|
|
return masterKey;
|
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()
|
|
|
|
{
|
|
|
|
Q_EMIT editFinished(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DatabaseOpenWidget::activatePassword()
|
2011-11-16 12:46:09 -05:00
|
|
|
{
|
|
|
|
m_ui->checkPassword->setChecked(true);
|
|
|
|
}
|
2011-12-24 13:19:52 -05:00
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
void DatabaseOpenWidget::activateKeyFile()
|
2011-12-24 13:19:52 -05:00
|
|
|
{
|
|
|
|
m_ui->checkKeyFile->setChecked(true);
|
|
|
|
}
|
|
|
|
|
2014-05-26 03:49:28 -04:00
|
|
|
void DatabaseOpenWidget::activateChallengeResponse()
|
|
|
|
{
|
|
|
|
m_ui->checkChallengeResponse->setChecked(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DatabaseOpenWidget::setOkButtonEnabled()
|
|
|
|
{
|
|
|
|
bool enable = m_ui->checkPassword->isChecked() || m_ui->checkChallengeResponse->isChecked()
|
|
|
|
|| (m_ui->checkKeyFile->isChecked() && !m_ui->comboKeyFile->currentText().isEmpty());
|
|
|
|
|
|
|
|
m_ui->buttonBox->button(QDialogButtonBox::Ok)->setEnabled(enable);
|
|
|
|
}
|
|
|
|
|
2012-06-28 03:21:15 -04:00
|
|
|
void DatabaseOpenWidget::browseKeyFile()
|
2011-12-24 13:19:52 -05:00
|
|
|
{
|
|
|
|
QString filters = QString("%1 (*);;%2 (*.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
|
|
|
|
|
|
|
if (!filename.isEmpty()) {
|
2011-12-25 14:36:45 -05:00
|
|
|
m_ui->comboKeyFile->lineEdit()->setText(filename);
|
2011-12-24 13:19:52 -05:00
|
|
|
}
|
|
|
|
}
|
2014-05-26 03:49:28 -04:00
|
|
|
|
2017-02-20 16:07:01 -05:00
|
|
|
void DatabaseOpenWidget::pollYubikey()
|
|
|
|
{
|
|
|
|
// YubiKey init is slow, detect asynchronously to not block the UI
|
|
|
|
m_ui->buttonRedetectYubikey->setEnabled(false);
|
|
|
|
m_ui->checkChallengeResponse->setEnabled(false);
|
|
|
|
m_ui->checkChallengeResponse->setChecked(false);
|
|
|
|
m_ui->comboChallengeResponse->setEnabled(false);
|
|
|
|
m_ui->comboChallengeResponse->clear();
|
|
|
|
QtConcurrent::run(YubiKey::instance(), &YubiKey::detect);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DatabaseOpenWidget::yubikeyDetected(int slot, bool blocking)
|
2014-05-26 03:49:28 -04:00
|
|
|
{
|
|
|
|
YkChallengeResponseKey yk(slot, blocking);
|
|
|
|
m_ui->comboChallengeResponse->addItem(yk.getName(), QVariant(slot));
|
|
|
|
m_ui->comboChallengeResponse->setEnabled(true);
|
|
|
|
m_ui->checkChallengeResponse->setEnabled(true);
|
2017-02-20 16:07:01 -05:00
|
|
|
m_ui->buttonRedetectYubikey->setEnabled(true);
|
|
|
|
|
|
|
|
if (config()->get("RememberLastKeyFiles").toBool()) {
|
|
|
|
QHash<QString, QVariant> lastChallengeResponse = config()->get("LastChallengeResponse").toHash();
|
|
|
|
if (lastChallengeResponse.contains(m_filename)) {
|
|
|
|
m_ui->checkChallengeResponse->setChecked(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DatabaseOpenWidget::noYubikeyFound()
|
|
|
|
{
|
|
|
|
m_ui->buttonRedetectYubikey->setEnabled(true);
|
2014-05-26 03:49:28 -04:00
|
|
|
}
|