2018-05-13 17:21:43 -04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 KeePassXC Team <team@keepassxc.org>
|
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
#include "DatabaseSettingsWidgetDatabaseKey.h"
|
2018-10-31 23:27:38 -04:00
|
|
|
|
2018-05-13 17:21:43 -04:00
|
|
|
#include "core/Database.h"
|
|
|
|
#include "gui/MessageBox.h"
|
2020-07-01 19:16:40 -04:00
|
|
|
#include "gui/databasekey/KeyFileEditWidget.h"
|
|
|
|
#include "gui/databasekey/PasswordEditWidget.h"
|
|
|
|
#include "gui/databasekey/YubiKeyEditWidget.h"
|
2021-04-22 23:07:49 -04:00
|
|
|
#include "keys/ChallengeResponseKey.h"
|
2018-10-31 23:27:38 -04:00
|
|
|
#include "keys/FileKey.h"
|
|
|
|
#include "keys/PasswordKey.h"
|
2018-05-13 17:21:43 -04:00
|
|
|
|
2021-07-11 22:10:29 -04:00
|
|
|
#include <QLayout>
|
2018-05-13 17:21:43 -04:00
|
|
|
#include <QPushButton>
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
DatabaseSettingsWidgetDatabaseKey::DatabaseSettingsWidgetDatabaseKey(QWidget* parent)
|
2018-05-13 17:21:43 -04:00
|
|
|
: DatabaseSettingsWidget(parent)
|
2021-02-14 19:15:10 -05:00
|
|
|
, m_additionalKeyOptionsToggle(new QPushButton(tr("Add additional protection…"), this))
|
2018-05-13 17:21:43 -04:00
|
|
|
, m_additionalKeyOptions(new QWidget(this))
|
|
|
|
, m_passwordEditWidget(new PasswordEditWidget(this))
|
|
|
|
, m_keyFileEditWidget(new KeyFileEditWidget(this))
|
|
|
|
#ifdef WITH_XC_YUBIKEY
|
|
|
|
, m_yubiKeyEditWidget(new YubiKeyEditWidget(this))
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
auto* vbox = new QVBoxLayout(this);
|
|
|
|
vbox->setSizeConstraint(QLayout::SetMinimumSize);
|
|
|
|
|
|
|
|
// primary password option
|
|
|
|
vbox->addWidget(m_passwordEditWidget);
|
|
|
|
|
|
|
|
// additional key options
|
|
|
|
m_additionalKeyOptionsToggle->setObjectName("additionalKeyOptionsToggle");
|
|
|
|
vbox->addWidget(m_additionalKeyOptionsToggle);
|
|
|
|
vbox->addWidget(m_additionalKeyOptions);
|
|
|
|
vbox->setSizeConstraint(QLayout::SetMinimumSize);
|
|
|
|
m_additionalKeyOptions->setLayout(new QVBoxLayout());
|
|
|
|
m_additionalKeyOptions->layout()->setMargin(0);
|
|
|
|
m_additionalKeyOptions->layout()->addWidget(m_keyFileEditWidget);
|
|
|
|
#ifdef WITH_XC_YUBIKEY
|
|
|
|
m_additionalKeyOptions->layout()->addWidget(m_yubiKeyEditWidget);
|
|
|
|
#endif
|
|
|
|
m_additionalKeyOptions->setVisible(false);
|
|
|
|
|
|
|
|
connect(m_additionalKeyOptionsToggle, SIGNAL(clicked()), SLOT(showAdditionalKeyOptions()));
|
|
|
|
|
|
|
|
vbox->addStretch();
|
|
|
|
setLayout(vbox);
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
DatabaseSettingsWidgetDatabaseKey::~DatabaseSettingsWidgetDatabaseKey()
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
void DatabaseSettingsWidgetDatabaseKey::load(QSharedPointer<Database> db)
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
DatabaseSettingsWidget::load(db);
|
|
|
|
|
|
|
|
if (!m_db->key() || m_db->key()->keys().isEmpty()) {
|
|
|
|
// database has no key, we are about to add a new one
|
|
|
|
m_passwordEditWidget->changeVisiblePage(KeyComponentWidget::Page::Edit);
|
|
|
|
m_passwordEditWidget->setPasswordVisible(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool hasAdditionalKeys = false;
|
2018-10-31 23:27:38 -04:00
|
|
|
for (const auto& key : m_db->key()->keys()) {
|
2018-05-13 17:21:43 -04:00
|
|
|
if (key->uuid() == PasswordKey::UUID) {
|
|
|
|
m_passwordEditWidget->setComponentAdded(true);
|
|
|
|
} else if (key->uuid() == FileKey::UUID) {
|
|
|
|
m_keyFileEditWidget->setComponentAdded(true);
|
|
|
|
hasAdditionalKeys = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef WITH_XC_YUBIKEY
|
2018-10-31 23:27:38 -04:00
|
|
|
for (const auto& key : m_db->key()->challengeResponseKeys()) {
|
2021-04-22 23:07:49 -04:00
|
|
|
if (key->uuid() == ChallengeResponseKey::UUID) {
|
2018-05-13 17:21:43 -04:00
|
|
|
m_yubiKeyEditWidget->setComponentAdded(true);
|
|
|
|
hasAdditionalKeys = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
setAdditionalKeyOptionsVisible(hasAdditionalKeys);
|
|
|
|
|
2019-03-30 21:31:32 -04:00
|
|
|
connect(m_passwordEditWidget->findChild<QPushButton*>("removeButton"), SIGNAL(clicked()), SLOT(markDirty()));
|
|
|
|
connect(m_keyFileEditWidget->findChild<QPushButton*>("removeButton"), SIGNAL(clicked()), SLOT(markDirty()));
|
2019-04-12 05:57:09 -04:00
|
|
|
#ifdef WITH_XC_YUBIKEY
|
2019-03-30 21:31:32 -04:00
|
|
|
connect(m_yubiKeyEditWidget->findChild<QPushButton*>("removeButton"), SIGNAL(clicked()), SLOT(markDirty()));
|
2019-04-12 05:57:09 -04:00
|
|
|
#endif
|
2018-05-13 17:21:43 -04:00
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
void DatabaseSettingsWidgetDatabaseKey::initialize()
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
bool blocked = blockSignals(true);
|
|
|
|
m_passwordEditWidget->setComponentAdded(false);
|
|
|
|
m_keyFileEditWidget->setComponentAdded(false);
|
|
|
|
#ifdef WITH_XC_YUBIKEY
|
|
|
|
m_yubiKeyEditWidget->setComponentAdded(false);
|
|
|
|
#endif
|
|
|
|
blockSignals(blocked);
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
void DatabaseSettingsWidgetDatabaseKey::uninitialize()
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
bool DatabaseSettingsWidgetDatabaseKey::save()
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
m_isDirty |= (m_passwordEditWidget->visiblePage() == KeyComponentWidget::Page::Edit);
|
|
|
|
m_isDirty |= (m_keyFileEditWidget->visiblePage() == KeyComponentWidget::Page::Edit);
|
|
|
|
#ifdef WITH_XC_YUBIKEY
|
|
|
|
m_isDirty |= (m_yubiKeyEditWidget->visiblePage() == KeyComponentWidget::Page::Edit);
|
|
|
|
#endif
|
|
|
|
|
2018-10-31 23:27:38 -04:00
|
|
|
if (m_db->key() && !m_db->key()->keys().isEmpty() && !m_isDirty) {
|
2018-05-13 17:21:43 -04:00
|
|
|
// key unchanged
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto newKey = QSharedPointer<CompositeKey>::create();
|
|
|
|
|
2019-03-19 08:25:30 -04:00
|
|
|
QSharedPointer<Key> oldPasswordKey;
|
|
|
|
QSharedPointer<Key> oldFileKey;
|
|
|
|
QSharedPointer<ChallengeResponseKey> oldChallengeResponse;
|
2018-05-13 17:21:43 -04:00
|
|
|
|
2018-10-31 23:27:38 -04:00
|
|
|
for (const auto& key : m_db->key()->keys()) {
|
2018-05-13 17:21:43 -04:00
|
|
|
if (key->uuid() == PasswordKey::UUID) {
|
2019-03-19 08:25:30 -04:00
|
|
|
oldPasswordKey = key;
|
2018-05-13 17:21:43 -04:00
|
|
|
} else if (key->uuid() == FileKey::UUID) {
|
2019-03-19 08:25:30 -04:00
|
|
|
oldFileKey = key;
|
2018-05-13 17:21:43 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-31 23:27:38 -04:00
|
|
|
for (const auto& key : m_db->key()->challengeResponseKeys()) {
|
2021-04-22 23:07:49 -04:00
|
|
|
if (key->uuid() == ChallengeResponseKey::UUID) {
|
2019-03-19 08:25:30 -04:00
|
|
|
oldChallengeResponse = key;
|
2018-05-13 17:21:43 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-22 12:00:31 -04:00
|
|
|
if (m_passwordEditWidget->visiblePage() == KeyComponentWidget::Page::AddNew || m_passwordEditWidget->isEmpty()) {
|
|
|
|
QScopedPointer<QMessageBox> msgBox(new QMessageBox(this));
|
|
|
|
msgBox->setIcon(QMessageBox::Warning);
|
|
|
|
msgBox->setWindowTitle(tr("No password set"));
|
|
|
|
msgBox->setText(tr("WARNING! You have not set a password. Using a database without "
|
|
|
|
"a password is strongly discouraged!\n\n"
|
|
|
|
"Are you sure you want to continue without a password?"));
|
|
|
|
auto btn = msgBox->addButton(tr("Continue without password"), QMessageBox::ButtonRole::AcceptRole);
|
|
|
|
msgBox->addButton(QMessageBox::Cancel);
|
|
|
|
msgBox->setDefaultButton(QMessageBox::Cancel);
|
|
|
|
msgBox->exec();
|
|
|
|
if (msgBox->clickedButton() != btn) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (!addToCompositeKey(m_passwordEditWidget, newKey, oldPasswordKey)) {
|
2018-05-13 17:21:43 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-19 08:25:30 -04:00
|
|
|
if (!addToCompositeKey(m_keyFileEditWidget, newKey, oldFileKey)) {
|
2018-05-13 17:21:43 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef WITH_XC_YUBIKEY
|
2019-03-19 08:25:30 -04:00
|
|
|
if (!addToCompositeKey(m_yubiKeyEditWidget, newKey, oldChallengeResponse)) {
|
2018-05-13 17:21:43 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (newKey->keys().isEmpty() && newKey->challengeResponseKeys().isEmpty()) {
|
2018-10-31 23:27:38 -04:00
|
|
|
MessageBox::critical(this,
|
|
|
|
tr("No encryption key added"),
|
2018-05-13 17:21:43 -04:00
|
|
|
tr("You must add at least one encryption key to secure your database!"),
|
2018-12-19 23:14:11 -05:00
|
|
|
MessageBox::Ok,
|
|
|
|
MessageBox::Ok);
|
2018-05-13 17:21:43 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-02-18 11:21:02 -05:00
|
|
|
m_db->setKey(newKey, true, false, false);
|
2018-05-13 17:21:43 -04:00
|
|
|
|
|
|
|
emit editFinished(true);
|
2019-02-18 11:21:02 -05:00
|
|
|
if (m_isDirty) {
|
|
|
|
m_db->markAsModified();
|
|
|
|
}
|
|
|
|
|
2018-05-13 17:21:43 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
void DatabaseSettingsWidgetDatabaseKey::discard()
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
emit editFinished(false);
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
void DatabaseSettingsWidgetDatabaseKey::showAdditionalKeyOptions()
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
setAdditionalKeyOptionsVisible(true);
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
void DatabaseSettingsWidgetDatabaseKey::setAdditionalKeyOptionsVisible(bool show)
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
m_additionalKeyOptionsToggle->setVisible(!show);
|
|
|
|
m_additionalKeyOptions->setVisible(show);
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
bool DatabaseSettingsWidgetDatabaseKey::addToCompositeKey(KeyComponentWidget* widget,
|
|
|
|
QSharedPointer<CompositeKey>& newKey,
|
|
|
|
QSharedPointer<Key>& oldKey)
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
if (widget->visiblePage() == KeyComponentWidget::Edit) {
|
|
|
|
QString error = tr("Unknown error");
|
|
|
|
if (!widget->validate(error) || !widget->addToCompositeKey(newKey)) {
|
2020-07-01 19:16:40 -04:00
|
|
|
MessageBox::critical(this, tr("Failed to change database credentials"), error, MessageBox::Ok);
|
2018-05-13 17:21:43 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (widget->visiblePage() == KeyComponentWidget::LeaveOrRemove) {
|
|
|
|
Q_ASSERT(oldKey);
|
|
|
|
newKey->addKey(oldKey);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
bool DatabaseSettingsWidgetDatabaseKey::addToCompositeKey(KeyComponentWidget* widget,
|
|
|
|
QSharedPointer<CompositeKey>& newKey,
|
|
|
|
QSharedPointer<ChallengeResponseKey>& oldKey)
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
if (widget->visiblePage() == KeyComponentWidget::Edit) {
|
|
|
|
QString error = tr("Unknown error");
|
|
|
|
if (!widget->validate(error) || !widget->addToCompositeKey(newKey)) {
|
2020-07-01 19:16:40 -04:00
|
|
|
MessageBox::critical(this, tr("Failed to change database credentials"), error, MessageBox::Ok);
|
2018-05-13 17:21:43 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (widget->visiblePage() == KeyComponentWidget::LeaveOrRemove) {
|
|
|
|
Q_ASSERT(oldKey);
|
|
|
|
newKey->addChallengeResponseKey(oldKey);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:16:40 -04:00
|
|
|
void DatabaseSettingsWidgetDatabaseKey::markDirty()
|
2018-05-13 17:21:43 -04:00
|
|
|
{
|
|
|
|
m_isDirty = true;
|
|
|
|
}
|