keepassxc/src/core/Metadata.cpp

514 lines
11 KiB
C++
Raw Normal View History

2010-08-07 15:10:44 +02:00
/*
* Copyright (C) 2010 Felix Geyer <debfx@fobos.de>
* Copyright (C) 2021 KeePassXC Team <team@keepassxc.org>
2010-08-07 15:10:44 +02: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/>.
*/
#include "Metadata.h"
#include "core/Clock.h"
#include "core/Entry.h"
#include "core/Group.h"
2021-07-11 22:10:29 -04:00
#include <QApplication>
#include <QCryptographicHash>
#include <QJsonDocument>
2010-08-13 18:08:06 +02:00
2012-07-01 21:58:45 +02:00
const int Metadata::DefaultHistoryMaxItems = 10;
const int Metadata::DefaultHistoryMaxSize = 6 * 1024 * 1024;
// Fallback icon for return by reference
Fix syntax issue & add a missing header (#7160) * Fix syntax to prevent compilation issue Fix build failure with AppleClang 7 & 8. Error was: src/core/CustomData.cpp:30:41: error: default initialization of an object of const type 'const CustomData::CustomDataItem' without a user-provided default constructor static const CustomData::CustomDataItem NULL_ITEM; ^ {} src/core/Metadata.cpp:32:39: error: default initialization of an object of const type 'const Metadata::CustomIconData' without a user-provided default constructor static const Metadata::CustomIconData NULL_ICON; ^ src/core/Metadata.cpp:32:48: note: add an explicit initializer to initialize 'NULL_ICON' static const Metadata::CustomIconData NULL_ICON; ^ {} * Add missing QUuid header Fixes this compilation issue with LLVM clang 9, possibly also due to an old QT5 tests/TestTools.cpp:96:39: error: incomplete type 'QUuid' named in nested name specifier auto validUuid = Tools::uuidToHex(QUuid::createUuid()); ^~~~~~~ /opt/local/libexec/qt5/lib/QtCore.framework/Headers/qmetatype.h:1887:1: note: forward declaration of 'QUuid' QT_FOR_EACH_STATIC_CORE_CLASS(QT_FORWARD_DECLARE_STATIC_TYPES_ITER) ^ /opt/local/libexec/qt5/lib/QtCore.framework/Headers/qmetatype.h:108:18: note: expanded from macro 'QT_FOR_EACH_STATIC_CORE_CLASS' F(QUuid, 30, QUuid) \ ^ tests/TestTools.cpp:101:40: error: incomplete type 'QUuid' named in nested name specifier auto nonHexUuid = Tools::uuidToHex(QUuid::createUuid()).replace(0, 1, 'p'); ^~~~~~~ /opt/local/libexec/qt5/lib/QtCore.framework/Headers/qmetatype.h:1887:1: note: forward declaration of 'QUuid' QT_FOR_EACH_STATIC_CORE_CLASS(QT_FORWARD_DECLARE_STATIC_TYPES_ITER) ^ /opt/local/libexec/qt5/lib/QtCore.framework/Headers/qmetatype.h:108:18: note: expanded from macro 'QT_FOR_EACH_STATIC_CORE_CLASS' F(QUuid, 30, QUuid) \ ^
2021-11-25 04:36:31 +01:00
static const Metadata::CustomIconData NULL_ICON{};
Metadata::Metadata(QObject* parent)
: ModifiableObject(parent)
, m_customData(new CustomData(this))
, m_updateDatetime(true)
2010-08-07 15:10:44 +02:00
{
init();
connect(m_customData, &CustomData::modified, this, &Metadata::modified);
}
void Metadata::init()
{
m_data.generator = QStringLiteral("KeePassXC");
2013-11-22 10:28:11 +01:00
m_data.maintenanceHistoryDays = 365;
m_data.masterKeyChangeRec = -1;
m_data.masterKeyChangeForce = -1;
m_data.historyMaxItems = DefaultHistoryMaxItems;
m_data.historyMaxSize = DefaultHistoryMaxSize;
m_data.recycleBinEnabled = true;
m_data.protectTitle = false;
m_data.protectUsername = false;
m_data.protectPassword = true;
m_data.protectUrl = false;
m_data.protectNotes = false;
QDateTime now = Clock::currentDateTimeUtc();
2013-11-22 10:28:11 +01:00
m_data.nameChanged = now;
m_data.descriptionChanged = now;
m_data.defaultUserNameChanged = now;
m_recycleBinChanged = now;
m_entryTemplatesGroupChanged = now;
m_masterKeyChanged = now;
m_settingsChanged = now;
}
void Metadata::clear()
{
init();
m_customIcons.clear();
m_customIconsOrder.clear();
m_customIconsHashes.clear();
m_customData->clear();
2012-04-11 15:57:11 +02:00
}
template <class P, class V> bool Metadata::set(P& property, const V& value)
{
2012-04-11 15:57:11 +02:00
if (property != value) {
property = value;
emitModified();
2012-04-11 15:57:11 +02:00
return true;
2018-03-31 16:01:30 -04:00
} else {
2012-04-11 15:57:11 +02:00
return false;
}
}
2018-03-31 16:01:30 -04:00
template <class P, class V> bool Metadata::set(P& property, const V& value, QDateTime& dateTime)
{
2012-04-11 15:57:11 +02:00
if (property != value) {
property = value;
if (m_updateDatetime) {
dateTime = Clock::currentDateTimeUtc();
2012-04-11 15:57:11 +02:00
}
emitModified();
2012-04-11 15:57:11 +02:00
return true;
2018-03-31 16:01:30 -04:00
} else {
2012-04-11 15:57:11 +02:00
return false;
}
}
2012-06-30 00:22:07 +02:00
void Metadata::setUpdateDatetime(bool value)
{
2012-04-11 15:57:11 +02:00
m_updateDatetime = value;
2010-08-07 15:10:44 +02:00
}
2013-11-22 10:28:11 +01:00
void Metadata::copyAttributesFrom(const Metadata* other)
{
m_data = other->m_data;
}
2010-08-07 15:10:44 +02:00
QString Metadata::generator() const
{
2013-11-22 10:28:11 +01:00
return m_data.generator;
}
2010-08-07 15:10:44 +02:00
QString Metadata::name() const
{
2013-11-22 10:28:11 +01:00
return m_data.name;
}
2010-08-07 15:10:44 +02:00
QDateTime Metadata::nameChanged() const
{
2013-11-22 10:28:11 +01:00
return m_data.nameChanged;
}
2010-08-07 15:10:44 +02:00
QString Metadata::description() const
{
2013-11-22 10:28:11 +01:00
return m_data.description;
}
2010-08-07 15:10:44 +02:00
QDateTime Metadata::descriptionChanged() const
{
2013-11-22 10:28:11 +01:00
return m_data.descriptionChanged;
}
2010-08-07 15:10:44 +02:00
QString Metadata::defaultUserName() const
{
2013-11-22 10:28:11 +01:00
return m_data.defaultUserName;
}
QDateTime Metadata::defaultUserNameChanged() const
{
2013-11-22 10:28:11 +01:00
return m_data.defaultUserNameChanged;
}
2010-08-07 15:10:44 +02:00
int Metadata::maintenanceHistoryDays() const
{
2013-11-22 10:28:11 +01:00
return m_data.maintenanceHistoryDays;
}
2010-08-07 15:10:44 +02:00
QString Metadata::color() const
{
2013-11-22 10:28:11 +01:00
return m_data.color;
}
2010-08-07 15:10:44 +02:00
bool Metadata::protectTitle() const
{
2013-11-22 10:28:11 +01:00
return m_data.protectTitle;
}
2010-08-07 15:10:44 +02:00
bool Metadata::protectUsername() const
{
2013-11-22 10:28:11 +01:00
return m_data.protectUsername;
}
2010-08-07 15:10:44 +02:00
bool Metadata::protectPassword() const
{
2013-11-22 10:28:11 +01:00
return m_data.protectPassword;
}
2010-08-07 15:10:44 +02:00
bool Metadata::protectUrl() const
{
2013-11-22 10:28:11 +01:00
return m_data.protectUrl;
}
2010-08-07 15:10:44 +02:00
bool Metadata::protectNotes() const
{
2013-11-22 10:28:11 +01:00
return m_data.protectNotes;
}
2010-08-07 15:10:44 +02:00
const Metadata::CustomIconData& Metadata::customIcon(const QUuid& uuid) const
2010-09-19 21:22:24 +02:00
{
auto icon = m_customIcons.find(uuid);
Q_ASSERT(icon != m_customIcons.end());
if (icon == m_customIcons.end()) {
return NULL_ICON;
}
return icon.value();
}
bool Metadata::hasCustomIcon(const QUuid& uuid) const
{
return m_customIcons.contains(uuid);
}
2018-03-22 22:56:05 +01:00
QList<QUuid> Metadata::customIconsOrder() const
{
return m_customIconsOrder;
}
2010-08-07 15:10:44 +02:00
bool Metadata::recycleBinEnabled() const
{
2013-11-22 10:28:11 +01:00
return m_data.recycleBinEnabled;
}
2010-08-07 15:10:44 +02:00
Group* Metadata::recycleBin()
{
return m_recycleBin;
}
2010-08-13 18:08:06 +02:00
const Group* Metadata::recycleBin() const
2010-08-07 15:10:44 +02:00
{
2010-08-13 18:08:06 +02:00
return m_recycleBin;
}
2010-08-07 15:10:44 +02:00
QDateTime Metadata::recycleBinChanged() const
{
return m_recycleBinChanged;
}
2010-08-07 15:10:44 +02:00
2010-08-13 18:08:06 +02:00
const Group* Metadata::entryTemplatesGroup() const
2010-08-07 15:10:44 +02:00
{
return m_entryTemplatesGroup;
}
2010-08-07 15:10:44 +02:00
QDateTime Metadata::entryTemplatesGroupChanged() const
{
return m_entryTemplatesGroupChanged;
}
2010-08-07 15:10:44 +02:00
2010-08-13 18:08:06 +02:00
const Group* Metadata::lastSelectedGroup() const
2010-08-07 15:10:44 +02:00
{
return m_lastSelectedGroup;
}
2010-08-07 15:10:44 +02:00
2010-08-13 18:08:06 +02:00
const Group* Metadata::lastTopVisibleGroup() const
2010-08-07 15:10:44 +02:00
{
return m_lastTopVisibleGroup;
}
2010-08-07 15:10:44 +02:00
QDateTime Metadata::databaseKeyChanged() const
{
return m_masterKeyChanged;
}
int Metadata::databaseKeyChangeRec() const
{
2013-11-22 10:28:11 +01:00
return m_data.masterKeyChangeRec;
}
int Metadata::databaseKeyChangeForce() const
{
2013-11-22 10:28:11 +01:00
return m_data.masterKeyChangeForce;
}
int Metadata::historyMaxItems() const
{
2013-11-22 10:28:11 +01:00
return m_data.historyMaxItems;
}
int Metadata::historyMaxSize() const
{
2013-11-22 10:28:11 +01:00
return m_data.historyMaxSize;
}
CustomData* Metadata::customData()
{
return m_customData;
}
const CustomData* Metadata::customData() const
2010-08-07 15:10:44 +02:00
{
return m_customData;
}
2010-08-07 15:10:44 +02:00
void Metadata::setGenerator(const QString& value)
{
2013-11-22 10:28:11 +01:00
set(m_data.generator, value);
2010-08-07 15:10:44 +02:00
}
void Metadata::setName(const QString& value)
{
set(m_data.name, value, m_data.nameChanged);
2010-08-07 15:10:44 +02:00
}
void Metadata::setNameChanged(const QDateTime& value)
{
Q_ASSERT(value.timeSpec() == Qt::UTC);
2013-11-22 10:28:11 +01:00
m_data.nameChanged = value;
2010-08-07 15:10:44 +02:00
}
void Metadata::setDescription(const QString& value)
{
2013-11-22 10:28:11 +01:00
set(m_data.description, value, m_data.descriptionChanged);
2010-08-07 15:10:44 +02:00
}
void Metadata::setDescriptionChanged(const QDateTime& value)
{
Q_ASSERT(value.timeSpec() == Qt::UTC);
2013-11-22 10:28:11 +01:00
m_data.descriptionChanged = value;
2010-08-07 15:10:44 +02:00
}
void Metadata::setDefaultUserName(const QString& value)
{
2013-11-22 10:28:11 +01:00
set(m_data.defaultUserName, value, m_data.defaultUserNameChanged);
2010-08-07 15:10:44 +02:00
}
void Metadata::setDefaultUserNameChanged(const QDateTime& value)
{
Q_ASSERT(value.timeSpec() == Qt::UTC);
2013-11-22 10:28:11 +01:00
m_data.defaultUserNameChanged = value;
}
2010-08-07 15:10:44 +02:00
void Metadata::setMaintenanceHistoryDays(int value)
{
2013-11-22 10:28:11 +01:00
set(m_data.maintenanceHistoryDays, value);
2010-08-07 15:10:44 +02:00
}
void Metadata::setColor(const QString& value)
{
2013-11-22 10:28:11 +01:00
set(m_data.color, value);
}
2010-08-07 15:10:44 +02:00
void Metadata::setProtectTitle(bool value)
{
2013-11-22 10:28:11 +01:00
set(m_data.protectTitle, value);
2010-08-07 15:10:44 +02:00
}
void Metadata::setProtectUsername(bool value)
{
2013-11-22 10:28:11 +01:00
set(m_data.protectUsername, value);
2010-08-07 15:10:44 +02:00
}
void Metadata::setProtectPassword(bool value)
{
2013-11-22 10:28:11 +01:00
set(m_data.protectPassword, value);
2010-08-07 15:10:44 +02:00
}
void Metadata::setProtectUrl(bool value)
{
2013-11-22 10:28:11 +01:00
set(m_data.protectUrl, value);
2010-08-07 15:10:44 +02:00
}
void Metadata::setProtectNotes(bool value)
{
2013-11-22 10:28:11 +01:00
set(m_data.protectNotes, value);
2010-08-07 15:10:44 +02:00
}
void Metadata::addCustomIcon(const QUuid& uuid, const CustomIconData& iconData)
2010-08-07 15:10:44 +02:00
{
2010-08-13 18:08:06 +02:00
Q_ASSERT(!uuid.isNull());
Q_ASSERT(!m_customIcons.contains(uuid));
2010-08-07 15:10:44 +02:00
// remove all uuids to prevent duplicates in release mode
m_customIcons[uuid] = iconData;
m_customIconsOrder.removeAll(uuid);
m_customIconsOrder.append(uuid);
// Associate image hash to uuid
QByteArray hash = hashIcon(iconData.data);
m_customIconsHashes[hash] = uuid;
Q_ASSERT(m_customIcons.count() == m_customIconsOrder.count());
emitModified();
}
void Metadata::addCustomIcon(const QUuid& uuid,
const QByteArray& iconBytes,
const QString& name,
const QDateTime& lastModified)
{
addCustomIcon(uuid, {iconBytes, name, lastModified});
}
2018-03-22 22:56:05 +01:00
void Metadata::removeCustomIcon(const QUuid& uuid)
2010-08-07 15:10:44 +02:00
{
2010-08-13 18:08:06 +02:00
Q_ASSERT(!uuid.isNull());
Q_ASSERT(m_customIcons.contains(uuid));
2010-08-07 15:10:44 +02:00
// Remove hash record only if this is the same uuid
QByteArray hash = hashIcon(m_customIcons[uuid].data);
if (m_customIconsHashes.contains(hash) && m_customIconsHashes[hash] == uuid) {
m_customIconsHashes.remove(hash);
}
2010-08-07 15:10:44 +02:00
m_customIcons.remove(uuid);
m_customIconsOrder.removeAll(uuid);
Q_ASSERT(m_customIcons.count() == m_customIconsOrder.count());
dynamic_cast<Database*>(parent())->addDeletedObject(uuid);
emitModified();
2010-08-07 15:10:44 +02:00
}
QUuid Metadata::findCustomIcon(const QByteArray& candidate)
{
QByteArray hash = hashIcon(candidate);
2018-03-22 22:56:05 +01:00
return m_customIconsHashes.value(hash, QUuid());
}
2018-03-22 22:56:05 +01:00
void Metadata::copyCustomIcons(const QSet<QUuid>& iconList, const Metadata* otherMetadata)
{
2018-03-22 22:56:05 +01:00
for (const QUuid& uuid : iconList) {
Q_ASSERT(otherMetadata->hasCustomIcon(uuid));
if (!hasCustomIcon(uuid) && otherMetadata->hasCustomIcon(uuid)) {
addCustomIcon(uuid, otherMetadata->customIcon(uuid));
}
}
}
QByteArray Metadata::hashIcon(const QByteArray& iconData)
{
return QCryptographicHash::hash(iconData, QCryptographicHash::Md5);
}
2010-08-07 15:10:44 +02:00
void Metadata::setRecycleBinEnabled(bool value)
{
2013-11-22 10:28:11 +01:00
set(m_data.recycleBinEnabled, value);
2010-08-07 15:10:44 +02:00
}
2010-08-13 18:08:06 +02:00
void Metadata::setRecycleBin(Group* group)
2010-08-07 15:10:44 +02:00
{
2012-04-11 15:57:11 +02:00
set(m_recycleBin, group, m_recycleBinChanged);
2010-08-07 15:10:44 +02:00
}
void Metadata::setRecycleBinChanged(const QDateTime& value)
{
Q_ASSERT(value.timeSpec() == Qt::UTC);
2010-08-07 15:10:44 +02:00
m_recycleBinChanged = value;
}
2010-08-13 18:08:06 +02:00
void Metadata::setEntryTemplatesGroup(Group* group)
2010-08-07 15:10:44 +02:00
{
2012-04-11 15:57:11 +02:00
set(m_entryTemplatesGroup, group, m_entryTemplatesGroupChanged);
2010-08-07 15:10:44 +02:00
}
void Metadata::setEntryTemplatesGroupChanged(const QDateTime& value)
{
Q_ASSERT(value.timeSpec() == Qt::UTC);
2010-08-07 15:10:44 +02:00
m_entryTemplatesGroupChanged = value;
}
2010-08-13 18:08:06 +02:00
void Metadata::setLastSelectedGroup(Group* group)
2010-08-07 15:10:44 +02:00
{
2012-04-11 15:57:11 +02:00
set(m_lastSelectedGroup, group);
2010-08-07 15:10:44 +02:00
}
2010-08-13 18:08:06 +02:00
void Metadata::setLastTopVisibleGroup(Group* group)
2010-08-07 15:10:44 +02:00
{
2012-04-11 15:57:11 +02:00
set(m_lastTopVisibleGroup, group);
2010-08-07 15:10:44 +02:00
}
void Metadata::setDatabaseKeyChanged(const QDateTime& value)
{
Q_ASSERT(value.timeSpec() == Qt::UTC);
m_masterKeyChanged = value;
}
void Metadata::setMasterKeyChangeRec(int value)
{
2013-11-22 10:28:11 +01:00
set(m_data.masterKeyChangeRec, value);
}
void Metadata::setMasterKeyChangeForce(int value)
{
2013-11-22 10:28:11 +01:00
set(m_data.masterKeyChangeForce, value);
}
void Metadata::setHistoryMaxItems(int value)
{
2013-11-22 10:28:11 +01:00
set(m_data.historyMaxItems, value);
}
void Metadata::setHistoryMaxSize(int value)
{
2013-11-22 10:28:11 +01:00
set(m_data.historyMaxSize, value);
}
QDateTime Metadata::settingsChanged() const
2010-08-07 15:10:44 +02:00
{
return m_settingsChanged;
}
void Metadata::setSettingsChanged(const QDateTime& value)
{
Q_ASSERT(value.timeSpec() == Qt::UTC);
m_settingsChanged = value;
}
void Metadata::addSavedSearch(const QString& name, const QString& searchtext)
{
auto searches = savedSearches();
searches.insert(name, searchtext);
auto json = QJsonDocument::fromVariant(searches);
m_customData->set("KPXC_SavedSearch", json.toJson());
}
void Metadata::deleteSavedSearch(const QString& name)
{
auto searches = savedSearches();
searches.remove(name);
auto json = QJsonDocument::fromVariant(searches);
m_customData->set("KPXC_SavedSearch", json.toJson());
}
QVariantMap Metadata::savedSearches()
{
auto searches = m_customData->value("KPXC_SavedSearch");
auto json = QJsonDocument::fromJson(searches.toUtf8());
return json.toVariant().toMap();
}