2010-08-07 09:10:44 -04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2010 Felix Geyer <debfx@fobos.de>
|
2017-06-09 17:40:36 -04:00
|
|
|
* Copyright (C) 2017 KeePassXC Team <team@keepassxc.org>
|
2010-08-07 09:10:44 -04: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 "Entry.h"
|
|
|
|
|
2017-07-01 08:51:20 -04:00
|
|
|
#include "config-keepassx.h"
|
|
|
|
|
2018-09-30 08:45:06 -04:00
|
|
|
#include "core/Clock.h"
|
2021-02-15 17:28:16 -05:00
|
|
|
#include "core/Config.h"
|
2011-07-08 08:51:14 -04:00
|
|
|
#include "core/Database.h"
|
|
|
|
#include "core/DatabaseIcons.h"
|
|
|
|
#include "core/Group.h"
|
|
|
|
#include "core/Metadata.h"
|
2018-12-24 18:15:46 -05:00
|
|
|
#include "core/Tools.h"
|
2017-04-13 06:05:36 -04:00
|
|
|
#include "totp/totp.h"
|
2010-08-12 15:38:59 -04:00
|
|
|
|
2018-09-20 23:49:56 -04:00
|
|
|
#include <QDir>
|
2017-11-20 13:01:22 -05:00
|
|
|
#include <QRegularExpression>
|
2018-10-28 11:13:58 -04:00
|
|
|
#include <utility>
|
2017-11-20 13:01:22 -05:00
|
|
|
|
2012-07-01 15:58:45 -04:00
|
|
|
const int Entry::DefaultIconNumber = 0;
|
2017-10-16 03:35:40 -04:00
|
|
|
const int Entry::ResolveMaximumDepth = 10;
|
2018-01-18 17:45:09 -05:00
|
|
|
const QString Entry::AutoTypeSequenceUsername = "{USERNAME}{ENTER}";
|
|
|
|
const QString Entry::AutoTypeSequencePassword = "{PASSWORD}{ENTER}";
|
2017-10-16 03:35:40 -04:00
|
|
|
|
2010-08-13 12:08:06 -04:00
|
|
|
Entry::Entry()
|
2012-07-16 03:54:04 -04:00
|
|
|
: m_attributes(new EntryAttributes(this))
|
|
|
|
, m_attachments(new EntryAttachments(this))
|
|
|
|
, m_autoTypeAssociations(new AutoTypeAssociations(this))
|
2018-02-06 10:37:06 -05:00
|
|
|
, m_customData(new CustomData(this))
|
2014-03-22 07:07:06 -04:00
|
|
|
, m_modifiedSinceBegin(false)
|
2012-07-19 07:57:55 -04:00
|
|
|
, m_updateTimeinfo(true)
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2012-05-15 10:47:46 -04:00
|
|
|
m_data.iconNumber = DefaultIconNumber;
|
2012-04-23 15:06:04 -04:00
|
|
|
m_data.autoTypeEnabled = true;
|
|
|
|
m_data.autoTypeObfuscation = 0;
|
2011-07-07 06:42:08 -04:00
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
connect(m_attributes, SIGNAL(entryAttributesModified()), SLOT(updateTotp()));
|
|
|
|
connect(m_attributes, SIGNAL(entryAttributesModified()), this, SIGNAL(entryModified()));
|
2012-04-14 09:38:20 -04:00
|
|
|
connect(m_attributes, SIGNAL(defaultKeyModified()), SLOT(emitDataChanged()));
|
2018-11-22 05:47:31 -05:00
|
|
|
connect(m_attachments, SIGNAL(entryAttachmentsModified()), this, SIGNAL(entryModified()));
|
|
|
|
connect(m_autoTypeAssociations, SIGNAL(modified()), SIGNAL(entryModified()));
|
|
|
|
connect(m_customData, SIGNAL(customDataModified()), this, SIGNAL(entryModified()));
|
2012-04-23 15:06:04 -04:00
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
connect(this, SIGNAL(entryModified()), SLOT(updateTimeinfo()));
|
|
|
|
connect(this, SIGNAL(entryModified()), SLOT(updateModifiedSinceBegin()));
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2010-08-18 10:22:48 -04:00
|
|
|
Entry::~Entry()
|
|
|
|
{
|
2018-09-30 08:45:06 -04:00
|
|
|
setUpdateTimeinfo(false);
|
2011-07-09 15:54:01 -04:00
|
|
|
if (m_group) {
|
|
|
|
m_group->removeEntry(this);
|
2012-04-21 13:06:28 -04:00
|
|
|
|
2012-04-21 18:29:39 -04:00
|
|
|
if (m_group->database()) {
|
|
|
|
m_group->database()->addDeletedObject(m_uuid);
|
|
|
|
}
|
2011-07-09 15:54:01 -04:00
|
|
|
}
|
|
|
|
|
2010-08-25 07:52:59 -04:00
|
|
|
qDeleteAll(m_history);
|
2010-08-18 10:22:48 -04:00
|
|
|
}
|
|
|
|
|
2012-08-01 04:40:38 -04:00
|
|
|
template <class T> inline bool Entry::set(T& property, const T& value)
|
2012-04-16 19:02:02 -04:00
|
|
|
{
|
2012-04-11 13:51:54 -04:00
|
|
|
if (property != value) {
|
|
|
|
property = value;
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryModified();
|
2012-04-11 13:51:54 -04:00
|
|
|
return true;
|
|
|
|
}
|
2018-09-30 08:45:06 -04:00
|
|
|
return false;
|
2012-04-11 13:51:54 -04:00
|
|
|
}
|
|
|
|
|
2012-04-16 19:02:02 -04:00
|
|
|
void Entry::updateTimeinfo()
|
|
|
|
{
|
|
|
|
if (m_updateTimeinfo) {
|
2018-09-30 08:45:06 -04:00
|
|
|
m_data.timeInfo.setLastModificationTime(Clock::currentDateTimeUtc());
|
|
|
|
m_data.timeInfo.setLastAccessTime(Clock::currentDateTimeUtc());
|
2012-04-16 19:02:02 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-30 08:45:06 -04:00
|
|
|
bool Entry::canUpdateTimeinfo() const
|
|
|
|
{
|
|
|
|
return m_updateTimeinfo;
|
|
|
|
}
|
|
|
|
|
2012-04-16 19:02:02 -04:00
|
|
|
void Entry::setUpdateTimeinfo(bool value)
|
|
|
|
{
|
2012-04-11 13:51:54 -04:00
|
|
|
m_updateTimeinfo = value;
|
|
|
|
}
|
|
|
|
|
2018-12-24 18:15:46 -05:00
|
|
|
QString Entry::buildReference(const QUuid& uuid, const QString& field)
|
|
|
|
{
|
|
|
|
Q_ASSERT(EntryAttributes::DefaultAttributes.count(field) > 0);
|
|
|
|
|
|
|
|
QString uuidStr = Tools::uuidToHex(uuid).toUpper();
|
|
|
|
QString shortField;
|
|
|
|
|
|
|
|
if (field == EntryAttributes::TitleKey) {
|
|
|
|
shortField = "T";
|
|
|
|
} else if (field == EntryAttributes::UserNameKey) {
|
|
|
|
shortField = "U";
|
|
|
|
} else if (field == EntryAttributes::PasswordKey) {
|
|
|
|
shortField = "P";
|
|
|
|
} else if (field == EntryAttributes::URLKey) {
|
|
|
|
shortField = "A";
|
|
|
|
} else if (field == EntryAttributes::NotesKey) {
|
|
|
|
shortField = "N";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shortField.isEmpty()) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
return QString("{REF:%1@I:%2}").arg(shortField, uuidStr);
|
|
|
|
}
|
|
|
|
|
2017-11-12 10:35:54 -05:00
|
|
|
EntryReferenceType Entry::referenceType(const QString& referenceStr)
|
|
|
|
{
|
|
|
|
const QString referenceLowerStr = referenceStr.toLower();
|
|
|
|
EntryReferenceType result = EntryReferenceType::Unknown;
|
|
|
|
if (referenceLowerStr == QLatin1String("t")) {
|
|
|
|
result = EntryReferenceType::Title;
|
|
|
|
} else if (referenceLowerStr == QLatin1String("u")) {
|
|
|
|
result = EntryReferenceType::UserName;
|
|
|
|
} else if (referenceLowerStr == QLatin1String("p")) {
|
|
|
|
result = EntryReferenceType::Password;
|
|
|
|
} else if (referenceLowerStr == QLatin1String("a")) {
|
|
|
|
result = EntryReferenceType::Url;
|
|
|
|
} else if (referenceLowerStr == QLatin1String("n")) {
|
|
|
|
result = EntryReferenceType::Notes;
|
|
|
|
} else if (referenceLowerStr == QLatin1String("i")) {
|
2018-03-22 17:56:05 -04:00
|
|
|
result = EntryReferenceType::QUuid;
|
2017-11-12 10:35:54 -05:00
|
|
|
} else if (referenceLowerStr == QLatin1String("o")) {
|
|
|
|
result = EntryReferenceType::CustomAttributes;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-03-22 17:56:05 -04:00
|
|
|
const QUuid& Entry::uuid() const
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
|
|
|
return m_uuid;
|
|
|
|
}
|
|
|
|
|
2018-09-30 08:45:06 -04:00
|
|
|
const QString Entry::uuidToHex() const
|
|
|
|
{
|
2018-12-24 18:15:46 -05:00
|
|
|
return Tools::uuidToHex(m_uuid);
|
2018-09-30 08:45:06 -04:00
|
|
|
}
|
|
|
|
|
2012-01-01 15:52:54 -05:00
|
|
|
QImage Entry::icon() const
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
if (m_data.customIcon.isNull()) {
|
2020-05-29 10:00:07 -04:00
|
|
|
return databaseIcons()->icon(m_data.iconNumber).toImage();
|
2018-03-31 16:01:30 -04:00
|
|
|
} else {
|
2013-07-04 07:31:38 -04:00
|
|
|
Q_ASSERT(database());
|
|
|
|
|
|
|
|
if (database()) {
|
|
|
|
return database()->metadata()->customIcon(m_data.customIcon);
|
2018-03-31 16:01:30 -04:00
|
|
|
} else {
|
2013-07-04 07:31:38 -04:00
|
|
|
return QImage();
|
|
|
|
}
|
2010-08-24 17:12:01 -04:00
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 10:00:07 -04:00
|
|
|
QPixmap Entry::iconPixmap(IconSize size) const
|
2012-01-01 15:52:54 -05:00
|
|
|
{
|
2020-05-29 10:00:07 -04:00
|
|
|
QPixmap icon(size, size);
|
2012-04-23 15:06:04 -04:00
|
|
|
if (m_data.customIcon.isNull()) {
|
2020-05-29 10:00:07 -04:00
|
|
|
icon = databaseIcons()->icon(m_data.iconNumber, size);
|
|
|
|
} else {
|
|
|
|
Q_ASSERT(database());
|
|
|
|
if (database()) {
|
|
|
|
icon = database()->metadata()->customIconPixmap(m_data.customIcon, size);
|
|
|
|
}
|
2018-09-30 08:45:06 -04:00
|
|
|
}
|
2013-07-04 07:31:38 -04:00
|
|
|
|
2020-05-29 10:00:07 -04:00
|
|
|
if (isExpired()) {
|
|
|
|
icon = databaseIcons()->applyBadge(icon, DatabaseIcons::Badges::Expired);
|
2012-01-01 15:52:54 -05:00
|
|
|
}
|
|
|
|
|
2020-05-29 10:00:07 -04:00
|
|
|
return icon;
|
2012-01-01 15:52:54 -05:00
|
|
|
}
|
|
|
|
|
2010-08-18 16:57:26 -04:00
|
|
|
int Entry::iconNumber() const
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.iconNumber;
|
2010-08-18 16:57:26 -04:00
|
|
|
}
|
|
|
|
|
2018-03-22 17:56:05 -04:00
|
|
|
const QUuid& Entry::iconUuid() const
|
2010-08-18 16:57:26 -04:00
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.customIcon;
|
2010-08-18 16:57:26 -04:00
|
|
|
}
|
|
|
|
|
2020-01-26 21:38:43 -05:00
|
|
|
QString Entry::foregroundColor() const
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.foregroundColor;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2020-01-26 21:38:43 -05:00
|
|
|
QString Entry::backgroundColor() const
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.backgroundColor;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
QString Entry::overrideUrl() const
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.overrideUrl;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2010-09-25 06:41:00 -04:00
|
|
|
QString Entry::tags() const
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.tags;
|
2010-09-25 06:41:00 -04:00
|
|
|
}
|
|
|
|
|
2018-09-30 08:45:06 -04:00
|
|
|
const TimeInfo& Entry::timeInfo() const
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.timeInfo;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Entry::autoTypeEnabled() const
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.autoTypeEnabled;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int Entry::autoTypeObfuscation() const
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.autoTypeObfuscation;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
QString Entry::defaultAutoTypeSequence() const
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
return m_data.defaultAutoTypeSequence;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2018-01-16 11:23:29 -05:00
|
|
|
/**
|
|
|
|
* Determine the effective sequence that will be injected
|
|
|
|
* This function return an empty string if a parent group has autotype disabled or if the entry has no parent
|
|
|
|
*/
|
2016-11-11 16:26:07 -05:00
|
|
|
QString Entry::effectiveAutoTypeSequence() const
|
|
|
|
{
|
2018-01-24 14:08:56 -05:00
|
|
|
if (!autoTypeEnabled()) {
|
|
|
|
return {};
|
2016-11-11 16:26:07 -05:00
|
|
|
}
|
|
|
|
|
2018-01-16 11:23:29 -05:00
|
|
|
const Group* parent = group();
|
|
|
|
if (!parent) {
|
2018-01-24 14:08:56 -05:00
|
|
|
return {};
|
2018-01-16 11:23:29 -05:00
|
|
|
}
|
2018-01-18 17:45:09 -05:00
|
|
|
|
2018-01-16 11:23:29 -05:00
|
|
|
QString sequence = parent->effectiveAutoTypeSequence();
|
|
|
|
if (sequence.isEmpty()) {
|
2018-01-24 14:08:56 -05:00
|
|
|
return {};
|
2016-11-11 16:26:07 -05:00
|
|
|
}
|
|
|
|
|
2018-01-16 11:23:29 -05:00
|
|
|
if (!m_data.defaultAutoTypeSequence.isEmpty()) {
|
|
|
|
return m_data.defaultAutoTypeSequence;
|
2016-11-11 16:26:07 -05:00
|
|
|
}
|
|
|
|
|
2018-01-18 17:45:09 -05:00
|
|
|
if (sequence == Group::RootAutoTypeSequence && (!username().isEmpty() || !password().isEmpty())) {
|
|
|
|
if (username().isEmpty()) {
|
|
|
|
return AutoTypeSequencePassword;
|
|
|
|
} else if (password().isEmpty()) {
|
|
|
|
return AutoTypeSequenceUsername;
|
|
|
|
}
|
|
|
|
return Group::RootAutoTypeSequence;
|
|
|
|
}
|
|
|
|
|
2016-11-11 16:26:07 -05:00
|
|
|
return sequence;
|
|
|
|
}
|
|
|
|
|
2021-02-15 17:28:16 -05:00
|
|
|
/**
|
|
|
|
* Retrive the autotype sequences matches for a given windowTitle
|
|
|
|
* This returns a list with priority ordering. If you don't want duplicates call .toSet() on it.
|
|
|
|
*/
|
|
|
|
QList<QString> Entry::autoTypeSequences(const QString& windowTitle) const
|
|
|
|
{
|
|
|
|
// If no window just return the effective sequence
|
|
|
|
if (windowTitle.isEmpty()) {
|
|
|
|
return {effectiveAutoTypeSequence()};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Define helper functions to match window titles
|
|
|
|
auto windowMatches = [&](const QString& pattern) {
|
|
|
|
// Regex searching
|
|
|
|
if (pattern.startsWith("//") && pattern.endsWith("//") && pattern.size() >= 4) {
|
|
|
|
QRegExp regExp(pattern.mid(2, pattern.size() - 4), Qt::CaseInsensitive, QRegExp::RegExp2);
|
|
|
|
return (regExp.indexIn(windowTitle) != -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wildcard searching
|
|
|
|
auto regex = Tools::convertToRegex(pattern, true, false, false);
|
|
|
|
return windowTitle.contains(regex);
|
|
|
|
};
|
|
|
|
|
|
|
|
auto windowMatchesTitle = [&](const QString& entryTitle) {
|
|
|
|
return !entryTitle.isEmpty() && windowTitle.contains(entryTitle, Qt::CaseInsensitive);
|
|
|
|
};
|
|
|
|
|
|
|
|
auto windowMatchesUrl = [&](const QString& entryUrl) {
|
|
|
|
if (!entryUrl.isEmpty() && windowTitle.contains(entryUrl, Qt::CaseInsensitive)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
QUrl url(entryUrl);
|
|
|
|
if (url.isValid() && !url.host().isEmpty()) {
|
|
|
|
return windowTitle.contains(url.host(), Qt::CaseInsensitive);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
QList<QString> sequenceList;
|
|
|
|
|
|
|
|
// Add window association matches
|
|
|
|
const auto assocList = autoTypeAssociations()->getAll();
|
|
|
|
for (const auto& assoc : assocList) {
|
|
|
|
auto window = resolveMultiplePlaceholders(assoc.window);
|
|
|
|
if (windowMatches(window)) {
|
|
|
|
if (!assoc.sequence.isEmpty()) {
|
|
|
|
sequenceList << assoc.sequence;
|
|
|
|
} else {
|
|
|
|
sequenceList << effectiveAutoTypeSequence();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to match window title
|
|
|
|
if (config()->get(Config::AutoTypeEntryTitleMatch).toBool() && windowMatchesTitle(resolvePlaceholder(title()))) {
|
|
|
|
sequenceList << effectiveAutoTypeSequence();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to match url in window title
|
|
|
|
if (config()->get(Config::AutoTypeEntryURLMatch).toBool() && windowMatchesUrl(resolvePlaceholder(url()))) {
|
|
|
|
sequenceList << effectiveAutoTypeSequence();
|
|
|
|
}
|
|
|
|
|
|
|
|
return sequenceList;
|
|
|
|
}
|
|
|
|
|
2012-07-16 03:54:04 -04:00
|
|
|
AutoTypeAssociations* Entry::autoTypeAssociations()
|
|
|
|
{
|
|
|
|
return m_autoTypeAssociations;
|
|
|
|
}
|
|
|
|
|
|
|
|
const AutoTypeAssociations* Entry::autoTypeAssociations() const
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2012-07-16 03:54:04 -04:00
|
|
|
return m_autoTypeAssociations;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
QString Entry::title() const
|
2012-04-06 13:33:29 -04:00
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
return m_attributes->value(EntryAttributes::TitleKey);
|
2012-04-06 13:33:29 -04:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
QString Entry::url() const
|
2012-04-06 13:33:29 -04:00
|
|
|
{
|
2013-12-01 18:10:47 -05:00
|
|
|
return m_attributes->value(EntryAttributes::URLKey);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2017-06-29 13:54:49 -04:00
|
|
|
QString Entry::webUrl() const
|
|
|
|
{
|
2017-10-25 00:01:29 -04:00
|
|
|
QString url = resolveMultiplePlaceholders(m_attributes->value(EntryAttributes::URLKey));
|
|
|
|
return resolveUrl(url);
|
2017-06-29 13:54:49 -04:00
|
|
|
}
|
|
|
|
|
2017-10-25 23:29:01 -04:00
|
|
|
QString Entry::displayUrl() const
|
|
|
|
{
|
|
|
|
QString url = maskPasswordPlaceholders(m_attributes->value(EntryAttributes::URLKey));
|
2018-07-14 17:08:04 -04:00
|
|
|
return resolveMultiplePlaceholders(url);
|
2017-10-25 23:29:01 -04:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
QString Entry::username() const
|
2011-01-13 16:31:17 -05:00
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
return m_attributes->value(EntryAttributes::UserNameKey);
|
2011-01-13 16:31:17 -05:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
QString Entry::password() const
|
2011-01-13 16:31:17 -05:00
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
return m_attributes->value(EntryAttributes::PasswordKey);
|
2011-01-13 16:31:17 -05:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
QString Entry::notes() const
|
2010-08-18 09:08:17 -04:00
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
return m_attributes->value(EntryAttributes::NotesKey);
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
2018-12-24 18:15:46 -05:00
|
|
|
QString Entry::attribute(const QString& key) const
|
|
|
|
{
|
|
|
|
return m_attributes->value(key);
|
|
|
|
}
|
|
|
|
|
2020-04-17 09:51:36 -04:00
|
|
|
int Entry::size() const
|
|
|
|
{
|
|
|
|
int size = 0;
|
|
|
|
const QRegularExpression delimiter(",|:|;");
|
|
|
|
|
|
|
|
size += this->attributes()->attributesSize();
|
|
|
|
size += this->autoTypeAssociations()->associationsSize();
|
|
|
|
size += this->attachments()->attachmentsSize();
|
|
|
|
size += this->customData()->dataSize();
|
|
|
|
const QStringList tags = this->tags().split(delimiter, QString::SkipEmptyParts);
|
|
|
|
for (const QString& tag : tags) {
|
|
|
|
size += tag.toUtf8().size();
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2012-05-15 13:58:10 -04:00
|
|
|
bool Entry::isExpired() const
|
|
|
|
{
|
2018-09-30 08:45:06 -04:00
|
|
|
return m_data.timeInfo.expires() && m_data.timeInfo.expiryTime() < Clock::currentDateTimeUtc();
|
2012-05-15 13:58:10 -04:00
|
|
|
}
|
|
|
|
|
2019-06-24 18:03:42 -04:00
|
|
|
bool Entry::isRecycled() const
|
|
|
|
{
|
|
|
|
const Database* db = database();
|
|
|
|
if (!db) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_group == db->metadata()->recycleBin() || m_group->isRecycled();
|
|
|
|
}
|
|
|
|
|
2019-06-22 09:38:02 -04:00
|
|
|
bool Entry::isAttributeReference(const QString& key) const
|
|
|
|
{
|
|
|
|
return m_attributes->isReference(key);
|
|
|
|
}
|
|
|
|
|
2018-12-24 18:15:46 -05:00
|
|
|
bool Entry::isAttributeReferenceOf(const QString& key, const QUuid& uuid) const
|
|
|
|
{
|
|
|
|
if (!m_attributes->isReference(key)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_attributes->value(key).contains(Tools::uuidToHex(uuid), Qt::CaseInsensitive);
|
|
|
|
}
|
|
|
|
|
2017-03-05 17:47:08 -05:00
|
|
|
bool Entry::hasReferences() const
|
|
|
|
{
|
|
|
|
const QList<QString> keyList = EntryAttributes::DefaultAttributes;
|
|
|
|
for (const QString& key : keyList) {
|
|
|
|
if (m_attributes->isReference(key)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-12-24 18:15:46 -05:00
|
|
|
bool Entry::hasReferencesTo(const QUuid& uuid) const
|
|
|
|
{
|
|
|
|
const QList<QString> keyList = EntryAttributes::DefaultAttributes;
|
|
|
|
for (const QString& key : keyList) {
|
|
|
|
if (isAttributeReferenceOf(key, uuid)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::replaceReferencesWithValues(const Entry* other)
|
|
|
|
{
|
|
|
|
for (const QString& key : EntryAttributes::DefaultAttributes) {
|
|
|
|
if (isAttributeReferenceOf(key, other->uuid())) {
|
|
|
|
setDefaultAttribute(key, other->attribute(key));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
EntryAttributes* Entry::attributes()
|
2010-08-18 09:08:17 -04:00
|
|
|
{
|
2012-04-14 09:38:20 -04:00
|
|
|
return m_attributes;
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
const EntryAttributes* Entry::attributes() const
|
2010-08-18 09:08:17 -04:00
|
|
|
{
|
2012-04-14 09:38:20 -04:00
|
|
|
return m_attributes;
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
EntryAttachments* Entry::attachments()
|
2010-08-18 09:08:17 -04:00
|
|
|
{
|
2012-04-14 09:38:20 -04:00
|
|
|
return m_attachments;
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
const EntryAttachments* Entry::attachments() const
|
2010-08-18 09:08:17 -04:00
|
|
|
{
|
2012-04-14 09:38:20 -04:00
|
|
|
return m_attachments;
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
2018-02-18 15:01:22 -05:00
|
|
|
CustomData* Entry::customData()
|
2018-02-06 10:37:06 -05:00
|
|
|
{
|
|
|
|
return m_customData;
|
|
|
|
}
|
|
|
|
|
2018-02-18 15:01:22 -05:00
|
|
|
const CustomData* Entry::customData() const
|
2018-02-06 10:37:06 -05:00
|
|
|
{
|
|
|
|
return m_customData;
|
|
|
|
}
|
|
|
|
|
2017-04-13 06:05:36 -04:00
|
|
|
bool Entry::hasTotp() const
|
|
|
|
{
|
2018-09-05 16:20:57 -04:00
|
|
|
return !m_data.totpSettings.isNull();
|
2017-04-13 06:05:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
QString Entry::totp() const
|
|
|
|
{
|
|
|
|
if (hasTotp()) {
|
2018-09-05 16:20:57 -04:00
|
|
|
return Totp::generateTotp(m_data.totpSettings);
|
2017-04-13 06:05:36 -04:00
|
|
|
}
|
2018-01-22 15:42:22 -05:00
|
|
|
return {};
|
2017-04-13 06:05:36 -04:00
|
|
|
}
|
|
|
|
|
2018-09-05 16:20:57 -04:00
|
|
|
void Entry::setTotp(QSharedPointer<Totp::Settings> settings)
|
2017-04-13 06:05:36 -04:00
|
|
|
{
|
2018-02-01 11:58:15 -05:00
|
|
|
beginUpdate();
|
2019-10-14 09:25:45 -04:00
|
|
|
m_attributes->remove(Totp::ATTRIBUTE_OTP);
|
|
|
|
m_attributes->remove(Totp::ATTRIBUTE_SEED);
|
|
|
|
m_attributes->remove(Totp::ATTRIBUTE_SETTINGS);
|
|
|
|
|
2019-02-01 21:01:51 -05:00
|
|
|
if (settings->key.isEmpty()) {
|
|
|
|
m_data.totpSettings.reset();
|
2017-04-13 06:05:36 -04:00
|
|
|
} else {
|
2019-02-01 21:01:51 -05:00
|
|
|
m_data.totpSettings = std::move(settings);
|
2020-08-26 15:17:31 -04:00
|
|
|
auto text = Totp::writeSettings(
|
|
|
|
m_data.totpSettings, resolveMultiplePlaceholders(title()), resolveMultiplePlaceholders(username()));
|
2019-10-14 09:25:45 -04:00
|
|
|
if (m_data.totpSettings->format != Totp::StorageFormat::LEGACY) {
|
2019-02-01 21:01:51 -05:00
|
|
|
m_attributes->set(Totp::ATTRIBUTE_OTP, text, true);
|
|
|
|
} else {
|
|
|
|
m_attributes->set(Totp::ATTRIBUTE_SEED, m_data.totpSettings->key, true);
|
|
|
|
m_attributes->set(Totp::ATTRIBUTE_SETTINGS, text);
|
|
|
|
}
|
2017-04-13 06:05:36 -04:00
|
|
|
}
|
2018-02-01 11:58:15 -05:00
|
|
|
endUpdate();
|
2017-04-13 06:05:36 -04:00
|
|
|
}
|
|
|
|
|
2018-09-05 16:20:57 -04:00
|
|
|
void Entry::updateTotp()
|
2017-04-13 06:05:36 -04:00
|
|
|
{
|
2018-09-05 16:20:57 -04:00
|
|
|
if (m_attributes->contains(Totp::ATTRIBUTE_SETTINGS)) {
|
|
|
|
m_data.totpSettings = Totp::parseSettings(m_attributes->value(Totp::ATTRIBUTE_SETTINGS),
|
|
|
|
m_attributes->value(Totp::ATTRIBUTE_SEED));
|
|
|
|
} else if (m_attributes->contains(Totp::ATTRIBUTE_OTP)) {
|
|
|
|
m_data.totpSettings = Totp::parseSettings(m_attributes->value(Totp::ATTRIBUTE_OTP));
|
2020-07-16 18:45:30 -04:00
|
|
|
} else {
|
|
|
|
m_data.totpSettings.reset();
|
2017-04-13 06:05:36 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-05 16:20:57 -04:00
|
|
|
QSharedPointer<Totp::Settings> Entry::totpSettings() const
|
2017-04-13 06:05:36 -04:00
|
|
|
{
|
2018-09-05 16:20:57 -04:00
|
|
|
return m_data.totpSettings;
|
2017-04-13 06:05:36 -04:00
|
|
|
}
|
|
|
|
|
2020-08-26 15:17:31 -04:00
|
|
|
QString Entry::totpSettingsString() const
|
|
|
|
{
|
|
|
|
if (m_data.totpSettings) {
|
|
|
|
return Totp::writeSettings(
|
|
|
|
m_data.totpSettings, resolveMultiplePlaceholders(title()), resolveMultiplePlaceholders(username()), true);
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-10-17 10:05:02 -04:00
|
|
|
QString Entry::path() const
|
|
|
|
{
|
|
|
|
auto path = group()->hierarchy();
|
|
|
|
path << title();
|
|
|
|
return path.mid(1).join("/");
|
|
|
|
}
|
|
|
|
|
2018-03-22 17:56:05 -04:00
|
|
|
void Entry::setUuid(const QUuid& uuid)
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2010-08-13 12:08:06 -04:00
|
|
|
Q_ASSERT(!uuid.isNull());
|
2012-04-11 13:51:54 -04:00
|
|
|
set(m_uuid, uuid);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setIcon(int iconNumber)
|
|
|
|
{
|
2010-08-14 06:24:35 -04:00
|
|
|
Q_ASSERT(iconNumber >= 0);
|
|
|
|
|
2012-04-23 15:06:04 -04:00
|
|
|
if (m_data.iconNumber != iconNumber || !m_data.customIcon.isNull()) {
|
|
|
|
m_data.iconNumber = iconNumber;
|
2018-03-22 17:56:05 -04:00
|
|
|
m_data.customIcon = QUuid();
|
2012-04-11 13:51:54 -04:00
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryModified();
|
2012-05-11 08:14:12 -04:00
|
|
|
emitDataChanged();
|
2012-04-11 13:51:54 -04:00
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2018-03-22 17:56:05 -04:00
|
|
|
void Entry::setIcon(const QUuid& uuid)
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2010-08-14 06:24:35 -04:00
|
|
|
Q_ASSERT(!uuid.isNull());
|
|
|
|
|
2012-04-23 15:06:04 -04:00
|
|
|
if (m_data.customIcon != uuid) {
|
|
|
|
m_data.customIcon = uuid;
|
|
|
|
m_data.iconNumber = 0;
|
2012-01-01 15:52:54 -05:00
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryModified();
|
2012-05-11 08:14:12 -04:00
|
|
|
emitDataChanged();
|
2012-04-11 13:51:54 -04:00
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2020-01-26 21:38:43 -05:00
|
|
|
void Entry::setForegroundColor(const QString& colorStr)
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2020-01-26 21:38:43 -05:00
|
|
|
set(m_data.foregroundColor, colorStr);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2020-01-26 21:38:43 -05:00
|
|
|
void Entry::setBackgroundColor(const QString& colorStr)
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2020-01-26 21:38:43 -05:00
|
|
|
set(m_data.backgroundColor, colorStr);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setOverrideUrl(const QString& url)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
set(m_data.overrideUrl, url);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2010-09-25 06:41:00 -04:00
|
|
|
void Entry::setTags(const QString& tags)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
set(m_data.tags, tags);
|
2010-09-25 06:41:00 -04:00
|
|
|
}
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
void Entry::setTimeInfo(const TimeInfo& timeInfo)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
m_data.timeInfo = timeInfo;
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setAutoTypeEnabled(bool enable)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
set(m_data.autoTypeEnabled, enable);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setAutoTypeObfuscation(int obfuscation)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
set(m_data.autoTypeObfuscation, obfuscation);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setDefaultAutoTypeSequence(const QString& sequence)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
set(m_data.defaultAutoTypeSequence, sequence);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
|
2010-08-18 09:08:17 -04:00
|
|
|
void Entry::setTitle(const QString& title)
|
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
m_attributes->set(EntryAttributes::TitleKey, title, m_attributes->isProtected(EntryAttributes::TitleKey));
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setUrl(const QString& url)
|
|
|
|
{
|
2018-03-31 16:01:30 -04:00
|
|
|
bool remove = url != m_attributes->value(EntryAttributes::URLKey)
|
|
|
|
&& (m_attributes->value(EntryAttributes::RememberCmdExecAttr) == "1"
|
|
|
|
|| m_attributes->value(EntryAttributes::RememberCmdExecAttr) == "0");
|
2017-01-28 08:18:43 -05:00
|
|
|
if (remove) {
|
|
|
|
m_attributes->remove(EntryAttributes::RememberCmdExecAttr);
|
|
|
|
}
|
2013-12-01 18:10:47 -05:00
|
|
|
m_attributes->set(EntryAttributes::URLKey, url, m_attributes->isProtected(EntryAttributes::URLKey));
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setUsername(const QString& username)
|
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
m_attributes->set(EntryAttributes::UserNameKey, username, m_attributes->isProtected(EntryAttributes::UserNameKey));
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setPassword(const QString& password)
|
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
m_attributes->set(EntryAttributes::PasswordKey, password, m_attributes->isProtected(EntryAttributes::PasswordKey));
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setNotes(const QString& notes)
|
|
|
|
{
|
2013-12-01 03:43:41 -05:00
|
|
|
m_attributes->set(EntryAttributes::NotesKey, notes, m_attributes->isProtected(EntryAttributes::NotesKey));
|
2010-08-18 09:08:17 -04:00
|
|
|
}
|
|
|
|
|
2018-12-24 18:15:46 -05:00
|
|
|
void Entry::setDefaultAttribute(const QString& attribute, const QString& value)
|
|
|
|
{
|
|
|
|
Q_ASSERT(EntryAttributes::isDefaultAttribute(attribute));
|
|
|
|
|
|
|
|
if (!EntryAttributes::isDefaultAttribute(attribute)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_attributes->set(attribute, value, m_attributes->isProtected(attribute));
|
|
|
|
}
|
|
|
|
|
2012-04-16 20:27:38 -04:00
|
|
|
void Entry::setExpires(const bool& value)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
if (m_data.timeInfo.expires() != value) {
|
|
|
|
m_data.timeInfo.setExpires(value);
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryModified();
|
2012-04-16 20:27:38 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::setExpiryTime(const QDateTime& dateTime)
|
|
|
|
{
|
2012-04-23 15:06:04 -04:00
|
|
|
if (m_data.timeInfo.expiryTime() != dateTime) {
|
|
|
|
m_data.timeInfo.setExpiryTime(dateTime);
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryModified();
|
2012-04-16 20:27:38 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-25 07:52:59 -04:00
|
|
|
QList<Entry*> Entry::historyItems()
|
|
|
|
{
|
|
|
|
return m_history;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QList<Entry*>& Entry::historyItems() const
|
|
|
|
{
|
|
|
|
return m_history;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::addHistoryItem(Entry* entry)
|
|
|
|
{
|
|
|
|
Q_ASSERT(!entry->parent());
|
|
|
|
|
|
|
|
m_history.append(entry);
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryModified();
|
2010-08-25 07:52:59 -04:00
|
|
|
}
|
|
|
|
|
2013-03-26 18:53:34 -04:00
|
|
|
void Entry::removeHistoryItems(const QList<Entry*>& historyEntries)
|
2012-05-18 13:22:22 -04:00
|
|
|
{
|
2013-03-26 18:53:34 -04:00
|
|
|
if (historyEntries.isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-09-02 13:51:51 -04:00
|
|
|
for (Entry* entry : historyEntries) {
|
2012-05-18 13:22:22 -04:00
|
|
|
Q_ASSERT(!entry->parent());
|
2018-09-30 08:45:06 -04:00
|
|
|
Q_ASSERT(entry->uuid().isNull() || entry->uuid() == uuid());
|
2013-03-26 18:53:34 -04:00
|
|
|
Q_ASSERT(m_history.contains(entry));
|
|
|
|
|
|
|
|
m_history.removeOne(entry);
|
2012-05-18 13:22:22 -04:00
|
|
|
delete entry;
|
|
|
|
}
|
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryModified();
|
2012-05-18 13:22:22 -04:00
|
|
|
}
|
|
|
|
|
2012-06-29 18:22:07 -04:00
|
|
|
void Entry::truncateHistory()
|
|
|
|
{
|
2012-05-14 10:29:22 -04:00
|
|
|
const Database* db = database();
|
2012-05-14 10:27:59 -04:00
|
|
|
|
|
|
|
if (!db) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-31 10:00:11 -04:00
|
|
|
bool changed = false;
|
2012-05-14 10:27:59 -04:00
|
|
|
int histMaxItems = db->metadata()->historyMaxItems();
|
|
|
|
if (histMaxItems > -1) {
|
|
|
|
int historyCount = 0;
|
|
|
|
QMutableListIterator<Entry*> i(m_history);
|
|
|
|
i.toBack();
|
|
|
|
while (i.hasPrevious()) {
|
|
|
|
historyCount++;
|
|
|
|
Entry* entry = i.previous();
|
|
|
|
if (historyCount > histMaxItems) {
|
|
|
|
delete entry;
|
|
|
|
i.remove();
|
2020-05-31 10:00:11 -04:00
|
|
|
changed = true;
|
2012-05-10 03:56:41 -04:00
|
|
|
}
|
2012-05-04 17:45:34 -04:00
|
|
|
}
|
2012-05-14 10:27:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int histMaxSize = db->metadata()->historyMaxSize();
|
|
|
|
if (histMaxSize > -1) {
|
|
|
|
int size = 0;
|
2018-01-15 11:20:16 -05:00
|
|
|
QSet<QByteArray> foundAttachments = attachments()->values();
|
2012-05-14 10:27:59 -04:00
|
|
|
|
|
|
|
QMutableListIterator<Entry*> i(m_history);
|
|
|
|
i.toBack();
|
|
|
|
while (i.hasPrevious()) {
|
2012-07-19 18:45:34 -04:00
|
|
|
Entry* historyItem = i.previous();
|
2012-05-14 10:27:59 -04:00
|
|
|
|
|
|
|
// don't calculate size if it's already above the maximum
|
|
|
|
if (size <= histMaxSize) {
|
2020-04-17 09:51:36 -04:00
|
|
|
size += historyItem->size();
|
2018-01-15 11:20:16 -05:00
|
|
|
foundAttachments += historyItem->attachments()->values();
|
2012-05-14 10:27:59 -04:00
|
|
|
}
|
2012-05-10 03:56:41 -04:00
|
|
|
|
2012-05-14 10:27:59 -04:00
|
|
|
if (size > histMaxSize) {
|
2012-07-19 18:45:34 -04:00
|
|
|
delete historyItem;
|
2012-05-14 10:27:59 -04:00
|
|
|
i.remove();
|
2020-05-31 10:00:11 -04:00
|
|
|
changed = true;
|
2012-05-10 03:56:41 -04:00
|
|
|
}
|
2012-05-04 17:45:34 -04:00
|
|
|
}
|
|
|
|
}
|
2020-05-31 10:00:11 -04:00
|
|
|
|
|
|
|
if (changed) {
|
|
|
|
emit entryModified();
|
|
|
|
}
|
2012-05-04 17:45:34 -04:00
|
|
|
}
|
|
|
|
|
2018-09-30 08:45:06 -04:00
|
|
|
bool Entry::equals(const Entry* other, CompareItemOptions options) const
|
|
|
|
{
|
|
|
|
if (!other) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (m_uuid != other->uuid()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!m_data.equals(other->m_data, options)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (*m_customData != *other->m_customData) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (*m_attributes != *other->m_attributes) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (*m_attachments != *other->m_attachments) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (*m_autoTypeAssociations != *other->m_autoTypeAssociations) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!options.testFlag(CompareItemIgnoreHistory)) {
|
|
|
|
if (m_history.count() != other->m_history.count()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (int i = 0; i < m_history.count(); ++i) {
|
|
|
|
if (!m_history[i]->equals(other->m_history[i], options)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-11-22 07:27:49 -05:00
|
|
|
Entry* Entry::clone(CloneFlags flags) const
|
2012-05-15 15:10:39 -04:00
|
|
|
{
|
|
|
|
Entry* entry = new Entry();
|
|
|
|
entry->setUpdateTimeinfo(false);
|
2013-11-22 07:27:49 -05:00
|
|
|
if (flags & CloneNewUuid) {
|
2018-03-22 17:56:05 -04:00
|
|
|
entry->m_uuid = QUuid::createUuid();
|
2018-03-31 16:01:30 -04:00
|
|
|
} else {
|
2013-11-22 07:27:49 -05:00
|
|
|
entry->m_uuid = m_uuid;
|
|
|
|
}
|
2012-05-15 15:10:39 -04:00
|
|
|
entry->m_data = m_data;
|
2018-02-06 10:37:06 -05:00
|
|
|
entry->m_customData->copyDataFrom(m_customData);
|
2012-07-20 06:13:26 -04:00
|
|
|
entry->m_attributes->copyDataFrom(m_attributes);
|
|
|
|
entry->m_attachments->copyDataFrom(m_attachments);
|
2017-02-27 20:25:56 -05:00
|
|
|
|
|
|
|
if (flags & CloneUserAsRef) {
|
2019-01-20 09:50:20 -05:00
|
|
|
entry->m_attributes->set(EntryAttributes::UserNameKey,
|
|
|
|
buildReference(uuid(), EntryAttributes::UserNameKey),
|
|
|
|
m_attributes->isProtected(EntryAttributes::UserNameKey));
|
2017-02-27 20:25:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & ClonePassAsRef) {
|
2019-01-20 09:50:20 -05:00
|
|
|
entry->m_attributes->set(EntryAttributes::PasswordKey,
|
|
|
|
buildReference(uuid(), EntryAttributes::PasswordKey),
|
|
|
|
m_attributes->isProtected(EntryAttributes::PasswordKey));
|
2017-02-27 20:25:56 -05:00
|
|
|
}
|
|
|
|
|
2018-01-15 12:17:56 -05:00
|
|
|
entry->m_autoTypeAssociations->copyDataFrom(m_autoTypeAssociations);
|
2013-11-22 07:27:49 -05:00
|
|
|
if (flags & CloneIncludeHistory) {
|
2016-09-02 13:51:51 -04:00
|
|
|
for (Entry* historyItem : m_history) {
|
2019-08-15 05:35:11 -04:00
|
|
|
Entry* historyItemClone =
|
|
|
|
historyItem->clone(flags & ~CloneIncludeHistory & ~CloneNewUuid & ~CloneResetTimeInfo);
|
2013-11-22 07:27:49 -05:00
|
|
|
historyItemClone->setUpdateTimeinfo(false);
|
|
|
|
historyItemClone->setUuid(entry->uuid());
|
|
|
|
historyItemClone->setUpdateTimeinfo(true);
|
|
|
|
entry->addHistoryItem(historyItemClone);
|
|
|
|
}
|
|
|
|
}
|
2012-05-15 15:10:39 -04:00
|
|
|
|
2013-11-22 07:27:49 -05:00
|
|
|
if (flags & CloneResetTimeInfo) {
|
2018-09-30 08:45:06 -04:00
|
|
|
QDateTime now = Clock::currentDateTimeUtc();
|
2013-11-22 07:27:49 -05:00
|
|
|
entry->m_data.timeInfo.setCreationTime(now);
|
|
|
|
entry->m_data.timeInfo.setLastModificationTime(now);
|
|
|
|
entry->m_data.timeInfo.setLastAccessTime(now);
|
|
|
|
entry->m_data.timeInfo.setLocationChanged(now);
|
|
|
|
}
|
|
|
|
|
2020-01-07 22:06:31 -05:00
|
|
|
if (flags & CloneRenameTitle) {
|
2018-03-13 16:02:38 -04:00
|
|
|
entry->setTitle(tr("%1 - Clone").arg(entry->title()));
|
2020-01-07 22:06:31 -05:00
|
|
|
}
|
2012-11-06 17:34:16 -05:00
|
|
|
|
2019-10-05 13:58:00 -04:00
|
|
|
entry->setUpdateTimeinfo(true);
|
|
|
|
|
2012-05-15 15:10:39 -04:00
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2013-04-14 08:21:42 -04:00
|
|
|
void Entry::copyDataFrom(const Entry* other)
|
|
|
|
{
|
|
|
|
setUpdateTimeinfo(false);
|
|
|
|
m_data = other->m_data;
|
2018-02-06 10:37:06 -05:00
|
|
|
m_customData->copyDataFrom(other->m_customData);
|
2013-04-14 08:21:42 -04:00
|
|
|
m_attributes->copyDataFrom(other->m_attributes);
|
|
|
|
m_attachments->copyDataFrom(other->m_attachments);
|
|
|
|
m_autoTypeAssociations->copyDataFrom(other->m_autoTypeAssociations);
|
|
|
|
setUpdateTimeinfo(true);
|
|
|
|
}
|
|
|
|
|
2012-04-23 15:06:04 -04:00
|
|
|
void Entry::beginUpdate()
|
|
|
|
{
|
2018-12-01 16:08:55 -05:00
|
|
|
Q_ASSERT(m_tmpHistoryItem.isNull());
|
2012-04-23 15:06:04 -04:00
|
|
|
|
2018-12-01 16:08:55 -05:00
|
|
|
m_tmpHistoryItem.reset(new Entry());
|
2012-04-23 15:06:04 -04:00
|
|
|
m_tmpHistoryItem->setUpdateTimeinfo(false);
|
|
|
|
m_tmpHistoryItem->m_uuid = m_uuid;
|
|
|
|
m_tmpHistoryItem->m_data = m_data;
|
2012-07-20 06:13:26 -04:00
|
|
|
m_tmpHistoryItem->m_attributes->copyDataFrom(m_attributes);
|
|
|
|
m_tmpHistoryItem->m_attachments->copyDataFrom(m_attachments);
|
2018-01-15 12:17:56 -05:00
|
|
|
m_tmpHistoryItem->m_autoTypeAssociations->copyDataFrom(m_autoTypeAssociations);
|
2012-04-23 15:06:04 -04:00
|
|
|
|
|
|
|
m_modifiedSinceBegin = false;
|
|
|
|
}
|
|
|
|
|
2016-07-31 17:49:38 -04:00
|
|
|
bool Entry::endUpdate()
|
2012-04-23 15:06:04 -04:00
|
|
|
{
|
2018-12-01 16:08:55 -05:00
|
|
|
Q_ASSERT(!m_tmpHistoryItem.isNull());
|
2012-04-23 15:06:04 -04:00
|
|
|
if (m_modifiedSinceBegin) {
|
|
|
|
m_tmpHistoryItem->setUpdateTimeinfo(true);
|
2018-12-01 16:08:55 -05:00
|
|
|
addHistoryItem(m_tmpHistoryItem.take());
|
2012-05-10 03:56:41 -04:00
|
|
|
truncateHistory();
|
2012-04-23 15:06:04 -04:00
|
|
|
}
|
|
|
|
|
2018-12-01 16:08:55 -05:00
|
|
|
m_tmpHistoryItem.reset();
|
2016-07-31 17:49:38 -04:00
|
|
|
|
|
|
|
return m_modifiedSinceBegin;
|
2012-04-23 15:06:04 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::updateModifiedSinceBegin()
|
|
|
|
{
|
|
|
|
m_modifiedSinceBegin = true;
|
|
|
|
}
|
|
|
|
|
2017-11-19 14:06:09 -05:00
|
|
|
QString Entry::resolveMultiplePlaceholdersRecursive(const QString& str, int maxDepth) const
|
2017-10-16 03:35:40 -04:00
|
|
|
{
|
|
|
|
if (maxDepth <= 0) {
|
2018-11-22 05:47:31 -05:00
|
|
|
qWarning("Maximum depth of replacement has been reached. Entry uuid: %s", uuid().toString().toLatin1().data());
|
2017-10-16 03:35:40 -04:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString result = str;
|
|
|
|
QRegExp placeholderRegEx("(\\{[^\\}]+\\})", Qt::CaseInsensitive, QRegExp::RegExp2);
|
|
|
|
placeholderRegEx.setMinimal(true);
|
|
|
|
int pos = 0;
|
|
|
|
while ((pos = placeholderRegEx.indexIn(str, pos)) != -1) {
|
|
|
|
const QString found = placeholderRegEx.cap(1);
|
|
|
|
result.replace(found, resolvePlaceholderRecursive(found, maxDepth - 1));
|
|
|
|
pos += placeholderRegEx.matchedLength();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result != str) {
|
|
|
|
result = resolveMultiplePlaceholdersRecursive(result, maxDepth - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-11-19 14:06:09 -05:00
|
|
|
QString Entry::resolvePlaceholderRecursive(const QString& placeholder, int maxDepth) const
|
2017-10-16 03:35:40 -04:00
|
|
|
{
|
2018-02-17 10:38:27 -05:00
|
|
|
if (maxDepth <= 0) {
|
2018-11-22 05:47:31 -05:00
|
|
|
qWarning("Maximum depth of replacement has been reached. Entry uuid: %s", uuid().toString().toLatin1().data());
|
2018-02-17 10:38:27 -05:00
|
|
|
return placeholder;
|
|
|
|
}
|
|
|
|
|
2017-10-16 03:35:40 -04:00
|
|
|
const PlaceholderType typeOfPlaceholder = placeholderType(placeholder);
|
|
|
|
switch (typeOfPlaceholder) {
|
|
|
|
case PlaceholderType::NotPlaceholder:
|
|
|
|
case PlaceholderType::Unknown:
|
2018-03-05 17:32:04 -05:00
|
|
|
return resolveMultiplePlaceholdersRecursive(placeholder, maxDepth - 1);
|
2017-10-16 03:35:40 -04:00
|
|
|
case PlaceholderType::Title:
|
2018-02-14 18:21:07 -05:00
|
|
|
if (placeholderType(title()) == PlaceholderType::Title) {
|
|
|
|
return title();
|
|
|
|
}
|
2018-03-05 11:02:02 -05:00
|
|
|
return resolveMultiplePlaceholdersRecursive(title(), maxDepth - 1);
|
2017-10-16 03:35:40 -04:00
|
|
|
case PlaceholderType::UserName:
|
2018-02-14 18:21:07 -05:00
|
|
|
if (placeholderType(username()) == PlaceholderType::UserName) {
|
|
|
|
return username();
|
|
|
|
}
|
2018-03-05 11:02:02 -05:00
|
|
|
return resolveMultiplePlaceholdersRecursive(username(), maxDepth - 1);
|
2017-10-16 03:35:40 -04:00
|
|
|
case PlaceholderType::Password:
|
2018-02-14 18:21:07 -05:00
|
|
|
if (placeholderType(password()) == PlaceholderType::Password) {
|
|
|
|
return password();
|
|
|
|
}
|
2018-03-05 11:02:02 -05:00
|
|
|
return resolveMultiplePlaceholdersRecursive(password(), maxDepth - 1);
|
2017-10-16 03:35:40 -04:00
|
|
|
case PlaceholderType::Notes:
|
2018-02-14 18:21:07 -05:00
|
|
|
if (placeholderType(notes()) == PlaceholderType::Notes) {
|
|
|
|
return notes();
|
|
|
|
}
|
2018-03-05 11:02:02 -05:00
|
|
|
return resolveMultiplePlaceholdersRecursive(notes(), maxDepth - 1);
|
2017-10-16 03:35:40 -04:00
|
|
|
case PlaceholderType::Url:
|
2018-02-14 18:21:07 -05:00
|
|
|
if (placeholderType(url()) == PlaceholderType::Url) {
|
|
|
|
return url();
|
|
|
|
}
|
2018-03-05 11:02:02 -05:00
|
|
|
return resolveMultiplePlaceholdersRecursive(url(), maxDepth - 1);
|
2020-05-20 22:44:38 -04:00
|
|
|
case PlaceholderType::DbDir: {
|
|
|
|
QFileInfo fileInfo(database()->filePath());
|
|
|
|
return fileInfo.absoluteDir().absolutePath();
|
|
|
|
}
|
2017-10-16 03:35:40 -04:00
|
|
|
case PlaceholderType::UrlWithoutScheme:
|
|
|
|
case PlaceholderType::UrlScheme:
|
|
|
|
case PlaceholderType::UrlHost:
|
|
|
|
case PlaceholderType::UrlPort:
|
|
|
|
case PlaceholderType::UrlPath:
|
|
|
|
case PlaceholderType::UrlQuery:
|
|
|
|
case PlaceholderType::UrlFragment:
|
|
|
|
case PlaceholderType::UrlUserInfo:
|
|
|
|
case PlaceholderType::UrlUserName:
|
|
|
|
case PlaceholderType::UrlPassword: {
|
|
|
|
const QString strUrl = resolveMultiplePlaceholdersRecursive(url(), maxDepth - 1);
|
|
|
|
return resolveUrlPlaceholder(strUrl, typeOfPlaceholder);
|
|
|
|
}
|
2018-02-14 18:21:07 -05:00
|
|
|
case PlaceholderType::Totp:
|
|
|
|
// totp can't have placeholder inside
|
|
|
|
return totp();
|
2017-10-16 03:35:40 -04:00
|
|
|
case PlaceholderType::CustomAttribute: {
|
|
|
|
const QString key = placeholder.mid(3, placeholder.length() - 4); // {S:attr} => mid(3, len - 4)
|
|
|
|
return attributes()->hasKey(key) ? attributes()->value(key) : QString();
|
|
|
|
}
|
2017-11-12 10:35:54 -05:00
|
|
|
case PlaceholderType::Reference:
|
|
|
|
return resolveReferencePlaceholderRecursive(placeholder, maxDepth);
|
2020-03-18 22:00:24 -04:00
|
|
|
case PlaceholderType::DateTimeSimple:
|
|
|
|
case PlaceholderType::DateTimeYear:
|
|
|
|
case PlaceholderType::DateTimeMonth:
|
|
|
|
case PlaceholderType::DateTimeDay:
|
|
|
|
case PlaceholderType::DateTimeHour:
|
|
|
|
case PlaceholderType::DateTimeMinute:
|
|
|
|
case PlaceholderType::DateTimeSecond:
|
|
|
|
case PlaceholderType::DateTimeUtcSimple:
|
|
|
|
case PlaceholderType::DateTimeUtcYear:
|
|
|
|
case PlaceholderType::DateTimeUtcMonth:
|
|
|
|
case PlaceholderType::DateTimeUtcDay:
|
|
|
|
case PlaceholderType::DateTimeUtcHour:
|
|
|
|
case PlaceholderType::DateTimeUtcMinute:
|
|
|
|
case PlaceholderType::DateTimeUtcSecond:
|
|
|
|
return resolveMultiplePlaceholdersRecursive(resolveDateTimePlaceholder(typeOfPlaceholder), maxDepth - 1);
|
2017-11-12 10:35:54 -05:00
|
|
|
}
|
2017-10-16 03:35:40 -04:00
|
|
|
|
2017-11-12 10:35:54 -05:00
|
|
|
return placeholder;
|
|
|
|
}
|
|
|
|
|
2020-03-18 22:00:24 -04:00
|
|
|
QString Entry::resolveDateTimePlaceholder(Entry::PlaceholderType placeholderType) const
|
|
|
|
{
|
|
|
|
QDateTime time = Clock::currentDateTime();
|
|
|
|
QDateTime time_utc = Clock::currentDateTimeUtc();
|
|
|
|
QString date_formatted{};
|
|
|
|
|
|
|
|
switch (placeholderType) {
|
|
|
|
case PlaceholderType::DateTimeSimple:
|
|
|
|
date_formatted = time.toString("yyyyMMddhhmmss");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeYear:
|
|
|
|
date_formatted = time.toString("yyyy");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeMonth:
|
|
|
|
date_formatted = time.toString("MM");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeDay:
|
|
|
|
date_formatted = time.toString("dd");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeHour:
|
|
|
|
date_formatted = time.toString("hh");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeMinute:
|
|
|
|
date_formatted = time.toString("mm");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeSecond:
|
|
|
|
date_formatted = time.toString("ss");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeUtcSimple:
|
|
|
|
date_formatted = time_utc.toString("yyyyMMddhhmmss");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeUtcYear:
|
|
|
|
date_formatted = time_utc.toString("yyyy");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeUtcMonth:
|
|
|
|
date_formatted = time_utc.toString("MM");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeUtcDay:
|
|
|
|
date_formatted = time_utc.toString("dd");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeUtcHour:
|
|
|
|
date_formatted = time_utc.toString("hh");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeUtcMinute:
|
|
|
|
date_formatted = time_utc.toString("mm");
|
|
|
|
break;
|
|
|
|
case PlaceholderType::DateTimeUtcSecond:
|
|
|
|
date_formatted = time_utc.toString("ss");
|
|
|
|
break;
|
|
|
|
default: {
|
|
|
|
Q_ASSERT_X(false, "Entry::resolveDateTimePlaceholder", "Bad DateTime placeholder type");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return date_formatted;
|
|
|
|
}
|
|
|
|
|
2017-11-19 14:06:09 -05:00
|
|
|
QString Entry::resolveReferencePlaceholderRecursive(const QString& placeholder, int maxDepth) const
|
2017-11-12 10:35:54 -05:00
|
|
|
{
|
2018-02-17 10:38:27 -05:00
|
|
|
if (maxDepth <= 0) {
|
2018-11-22 05:47:31 -05:00
|
|
|
qWarning("Maximum depth of replacement has been reached. Entry uuid: %s", uuid().toString().toLatin1().data());
|
2018-02-17 10:38:27 -05:00
|
|
|
return placeholder;
|
|
|
|
}
|
|
|
|
|
2017-11-12 10:35:54 -05:00
|
|
|
// resolving references in format: {REF:<WantedField>@<SearchIn>:<SearchText>}
|
|
|
|
// using format from http://keepass.info/help/base/fieldrefs.html at the time of writing
|
|
|
|
|
2017-11-19 14:06:09 -05:00
|
|
|
QRegularExpressionMatch match = EntryAttributes::matchReference(placeholder);
|
2017-11-12 10:35:54 -05:00
|
|
|
if (!match.hasMatch()) {
|
|
|
|
return placeholder;
|
2017-10-16 03:35:40 -04:00
|
|
|
}
|
2017-11-12 10:35:54 -05:00
|
|
|
|
|
|
|
QString result;
|
|
|
|
const QString searchIn = match.captured(EntryAttributes::SearchInGroupName);
|
|
|
|
const QString searchText = match.captured(EntryAttributes::SearchTextGroupName);
|
|
|
|
|
|
|
|
const EntryReferenceType searchInType = Entry::referenceType(searchIn);
|
2018-02-17 10:38:27 -05:00
|
|
|
|
|
|
|
Q_ASSERT(m_group);
|
|
|
|
Q_ASSERT(m_group->database());
|
2019-02-18 08:26:56 -05:00
|
|
|
const Entry* refEntry = m_group->database()->rootGroup()->findEntryBySearchTerm(searchText, searchInType);
|
2017-11-12 10:35:54 -05:00
|
|
|
|
|
|
|
if (refEntry) {
|
|
|
|
const QString wantedField = match.captured(EntryAttributes::WantedFieldGroupName);
|
|
|
|
result = refEntry->referenceFieldValue(Entry::referenceType(wantedField));
|
|
|
|
|
|
|
|
// Referencing fields of other entries only works with standard fields, not with custom user strings.
|
|
|
|
// If you want to reference a custom user string, you need to place a redirection in a standard field
|
|
|
|
// of the entry with the custom string, using {S:<Name>}, and reference the standard field.
|
|
|
|
result = refEntry->resolveMultiplePlaceholdersRecursive(result, maxDepth - 1);
|
2017-10-16 03:35:40 -04:00
|
|
|
}
|
|
|
|
|
2017-11-12 10:35:54 -05:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Entry::referenceFieldValue(EntryReferenceType referenceType) const
|
|
|
|
{
|
|
|
|
switch (referenceType) {
|
|
|
|
case EntryReferenceType::Title:
|
|
|
|
return title();
|
|
|
|
case EntryReferenceType::UserName:
|
|
|
|
return username();
|
|
|
|
case EntryReferenceType::Password:
|
|
|
|
return password();
|
|
|
|
case EntryReferenceType::Url:
|
|
|
|
return url();
|
|
|
|
case EntryReferenceType::Notes:
|
|
|
|
return notes();
|
2018-03-22 17:56:05 -04:00
|
|
|
case EntryReferenceType::QUuid:
|
2018-09-30 08:45:06 -04:00
|
|
|
return uuidToHex();
|
2017-11-12 10:35:54 -05:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return QString();
|
2017-10-16 03:35:40 -04:00
|
|
|
}
|
|
|
|
|
2020-05-21 21:43:00 -04:00
|
|
|
void Entry::moveUp()
|
|
|
|
{
|
|
|
|
if (m_group) {
|
|
|
|
m_group->moveEntryUp(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Entry::moveDown()
|
|
|
|
{
|
|
|
|
if (m_group) {
|
|
|
|
m_group->moveEntryDown(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-06 16:54:07 -04:00
|
|
|
Group* Entry::group()
|
|
|
|
{
|
|
|
|
return m_group;
|
|
|
|
}
|
|
|
|
|
2012-07-23 12:06:04 -04:00
|
|
|
const Group* Entry::group() const
|
|
|
|
{
|
|
|
|
return m_group;
|
|
|
|
}
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
void Entry::setGroup(Group* group)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
2012-04-27 04:50:32 -04:00
|
|
|
Q_ASSERT(group);
|
|
|
|
|
2012-04-22 06:09:12 -04:00
|
|
|
if (m_group == group) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
if (m_group) {
|
2010-08-13 12:08:06 -04:00
|
|
|
m_group->removeEntry(this);
|
2012-04-27 04:50:32 -04:00
|
|
|
if (m_group->database() && m_group->database() != group->database()) {
|
2012-04-21 18:29:39 -04:00
|
|
|
m_group->database()->addDeletedObject(m_uuid);
|
2012-04-27 05:22:02 -04:00
|
|
|
|
|
|
|
// copy custom icon to the new database
|
2020-05-29 10:00:07 -04:00
|
|
|
if (!iconUuid().isNull() && group->database() && m_group->database()->metadata()->hasCustomIcon(iconUuid())
|
|
|
|
&& !group->database()->metadata()->hasCustomIcon(iconUuid())) {
|
2012-04-27 05:22:02 -04:00
|
|
|
group->database()->metadata()->addCustomIcon(iconUuid(), icon());
|
|
|
|
}
|
2012-04-21 13:06:28 -04:00
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
2012-04-21 13:06:28 -04:00
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
m_group = group;
|
2012-05-12 07:22:41 -04:00
|
|
|
group->addEntry(this);
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
QObject::setParent(group);
|
2012-04-22 06:09:12 -04:00
|
|
|
|
|
|
|
if (m_updateTimeinfo) {
|
2018-09-30 08:45:06 -04:00
|
|
|
m_data.timeInfo.setLocationChanged(Clock::currentDateTimeUtc());
|
2012-04-22 06:09:12 -04:00
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
2011-07-07 06:42:08 -04:00
|
|
|
|
2012-04-14 09:38:20 -04:00
|
|
|
void Entry::emitDataChanged()
|
2011-07-07 06:42:08 -04:00
|
|
|
{
|
2018-11-22 05:47:31 -05:00
|
|
|
emit entryDataChanged(this);
|
2011-07-07 06:42:08 -04:00
|
|
|
}
|
2012-04-18 07:57:57 -04:00
|
|
|
|
|
|
|
const Database* Entry::database() const
|
|
|
|
{
|
|
|
|
if (m_group) {
|
|
|
|
return m_group->database();
|
|
|
|
}
|
2018-09-30 08:45:06 -04:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
Database* Entry::database()
|
|
|
|
{
|
|
|
|
if (m_group) {
|
|
|
|
return m_group->database();
|
|
|
|
}
|
|
|
|
return nullptr;
|
2012-04-18 07:57:57 -04:00
|
|
|
}
|
2012-05-12 07:22:41 -04:00
|
|
|
|
2017-11-19 14:06:09 -05:00
|
|
|
QString Entry::maskPasswordPlaceholders(const QString& str) const
|
2017-09-29 14:10:54 -04:00
|
|
|
{
|
|
|
|
QString result = str;
|
|
|
|
result.replace(QRegExp("(\\{PASSWORD\\})", Qt::CaseInsensitive, QRegExp::RegExp2), "******");
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-10-01 10:26:24 -04:00
|
|
|
Entry* Entry::resolveReference(const QString& str) const
|
|
|
|
{
|
|
|
|
QRegularExpressionMatch match = EntryAttributes::matchReference(str);
|
|
|
|
if (!match.hasMatch()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString searchIn = match.captured(EntryAttributes::SearchInGroupName);
|
|
|
|
const QString searchText = match.captured(EntryAttributes::SearchTextGroupName);
|
|
|
|
|
|
|
|
const EntryReferenceType searchInType = Entry::referenceType(searchIn);
|
2018-12-18 22:28:56 -05:00
|
|
|
return m_group->database()->rootGroup()->findEntryBySearchTerm(searchText, searchInType);
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
|
|
|
|
2016-11-25 12:26:59 -05:00
|
|
|
QString Entry::resolveMultiplePlaceholders(const QString& str) const
|
2012-07-12 16:33:20 -04:00
|
|
|
{
|
2017-10-16 03:35:40 -04:00
|
|
|
return resolveMultiplePlaceholdersRecursive(str, ResolveMaximumDepth);
|
2016-11-25 12:26:59 -05:00
|
|
|
}
|
2012-07-12 16:33:20 -04:00
|
|
|
|
2017-10-14 12:23:22 -04:00
|
|
|
QString Entry::resolvePlaceholder(const QString& placeholder) const
|
2016-11-25 12:26:59 -05:00
|
|
|
{
|
2017-10-16 03:35:40 -04:00
|
|
|
return resolvePlaceholderRecursive(placeholder, ResolveMaximumDepth);
|
2017-10-14 12:23:22 -04:00
|
|
|
}
|
|
|
|
|
2017-11-19 14:06:09 -05:00
|
|
|
QString Entry::resolveUrlPlaceholder(const QString& str, Entry::PlaceholderType placeholderType) const
|
2017-10-14 12:23:22 -04:00
|
|
|
{
|
2020-01-07 22:06:31 -05:00
|
|
|
if (str.isEmpty()) {
|
2017-10-14 12:23:22 -04:00
|
|
|
return QString();
|
2020-01-07 22:06:31 -05:00
|
|
|
}
|
2017-10-14 12:23:22 -04:00
|
|
|
|
2017-10-16 03:35:40 -04:00
|
|
|
const QUrl qurl(str);
|
2017-10-14 12:23:22 -04:00
|
|
|
switch (placeholderType) {
|
|
|
|
case PlaceholderType::UrlWithoutScheme:
|
|
|
|
return qurl.toString(QUrl::RemoveScheme | QUrl::FullyDecoded);
|
|
|
|
case PlaceholderType::UrlScheme:
|
|
|
|
return qurl.scheme();
|
|
|
|
case PlaceholderType::UrlHost:
|
|
|
|
return qurl.host();
|
|
|
|
case PlaceholderType::UrlPort:
|
|
|
|
return QString::number(qurl.port());
|
|
|
|
case PlaceholderType::UrlPath:
|
|
|
|
return qurl.path();
|
|
|
|
case PlaceholderType::UrlQuery:
|
|
|
|
return qurl.query();
|
|
|
|
case PlaceholderType::UrlFragment:
|
|
|
|
return qurl.fragment();
|
|
|
|
case PlaceholderType::UrlUserInfo:
|
|
|
|
return qurl.userInfo();
|
|
|
|
case PlaceholderType::UrlUserName:
|
|
|
|
return qurl.userName();
|
|
|
|
case PlaceholderType::UrlPassword:
|
|
|
|
return qurl.password();
|
2017-10-16 03:35:40 -04:00
|
|
|
default: {
|
|
|
|
Q_ASSERT_X(false, "Entry::resolveUrlPlaceholder", "Bad url placeholder type");
|
2017-10-14 12:23:22 -04:00
|
|
|
break;
|
2017-02-27 20:25:56 -05:00
|
|
|
}
|
2017-10-16 03:35:40 -04:00
|
|
|
}
|
2017-02-27 20:25:56 -05:00
|
|
|
|
2017-10-16 03:35:40 -04:00
|
|
|
return QString();
|
2017-10-14 12:23:22 -04:00
|
|
|
}
|
|
|
|
|
2017-11-19 14:06:09 -05:00
|
|
|
Entry::PlaceholderType Entry::placeholderType(const QString& placeholder) const
|
2017-10-14 12:23:22 -04:00
|
|
|
{
|
|
|
|
if (!placeholder.startsWith(QLatin1Char('{')) || !placeholder.endsWith(QLatin1Char('}'))) {
|
|
|
|
return PlaceholderType::NotPlaceholder;
|
2018-09-30 08:45:06 -04:00
|
|
|
}
|
|
|
|
if (placeholder.startsWith(QLatin1Literal("{S:"))) {
|
2017-10-14 12:23:22 -04:00
|
|
|
return PlaceholderType::CustomAttribute;
|
2018-09-30 08:45:06 -04:00
|
|
|
}
|
|
|
|
if (placeholder.startsWith(QLatin1Literal("{REF:"))) {
|
2017-10-14 12:23:22 -04:00
|
|
|
return PlaceholderType::Reference;
|
|
|
|
}
|
|
|
|
|
2018-03-31 16:01:30 -04:00
|
|
|
static const QMap<QString, PlaceholderType> placeholders{
|
|
|
|
{QStringLiteral("{TITLE}"), PlaceholderType::Title},
|
|
|
|
{QStringLiteral("{USERNAME}"), PlaceholderType::UserName},
|
|
|
|
{QStringLiteral("{PASSWORD}"), PlaceholderType::Password},
|
|
|
|
{QStringLiteral("{NOTES}"), PlaceholderType::Notes},
|
|
|
|
{QStringLiteral("{TOTP}"), PlaceholderType::Totp},
|
|
|
|
{QStringLiteral("{URL}"), PlaceholderType::Url},
|
|
|
|
{QStringLiteral("{URL:RMVSCM}"), PlaceholderType::UrlWithoutScheme},
|
|
|
|
{QStringLiteral("{URL:WITHOUTSCHEME}"), PlaceholderType::UrlWithoutScheme},
|
|
|
|
{QStringLiteral("{URL:SCM}"), PlaceholderType::UrlScheme},
|
|
|
|
{QStringLiteral("{URL:SCHEME}"), PlaceholderType::UrlScheme},
|
|
|
|
{QStringLiteral("{URL:HOST}"), PlaceholderType::UrlHost},
|
|
|
|
{QStringLiteral("{URL:PORT}"), PlaceholderType::UrlPort},
|
|
|
|
{QStringLiteral("{URL:PATH}"), PlaceholderType::UrlPath},
|
|
|
|
{QStringLiteral("{URL:QUERY}"), PlaceholderType::UrlQuery},
|
|
|
|
{QStringLiteral("{URL:FRAGMENT}"), PlaceholderType::UrlFragment},
|
|
|
|
{QStringLiteral("{URL:USERINFO}"), PlaceholderType::UrlUserInfo},
|
|
|
|
{QStringLiteral("{URL:USERNAME}"), PlaceholderType::UrlUserName},
|
2020-03-18 22:00:24 -04:00
|
|
|
{QStringLiteral("{URL:PASSWORD}"), PlaceholderType::UrlPassword},
|
|
|
|
{QStringLiteral("{DT_SIMPLE}"), PlaceholderType::DateTimeSimple},
|
|
|
|
{QStringLiteral("{DT_YEAR}"), PlaceholderType::DateTimeYear},
|
|
|
|
{QStringLiteral("{DT_MONTH}"), PlaceholderType::DateTimeMonth},
|
|
|
|
{QStringLiteral("{DT_DAY}"), PlaceholderType::DateTimeDay},
|
|
|
|
{QStringLiteral("{DT_HOUR}"), PlaceholderType::DateTimeHour},
|
|
|
|
{QStringLiteral("{DT_MINUTE}"), PlaceholderType::DateTimeMinute},
|
|
|
|
{QStringLiteral("{DT_SECOND}"), PlaceholderType::DateTimeSecond},
|
|
|
|
{QStringLiteral("{DT_UTC_SIMPLE}"), PlaceholderType::DateTimeUtcSimple},
|
|
|
|
{QStringLiteral("{DT_UTC_YEAR}"), PlaceholderType::DateTimeUtcYear},
|
|
|
|
{QStringLiteral("{DT_UTC_MONTH}"), PlaceholderType::DateTimeUtcMonth},
|
|
|
|
{QStringLiteral("{DT_UTC_DAY}"), PlaceholderType::DateTimeUtcDay},
|
|
|
|
{QStringLiteral("{DT_UTC_HOUR}"), PlaceholderType::DateTimeUtcHour},
|
|
|
|
{QStringLiteral("{DT_UTC_MINUTE}"), PlaceholderType::DateTimeUtcMinute},
|
2020-05-20 22:44:38 -04:00
|
|
|
{QStringLiteral("{DT_UTC_SECOND}"), PlaceholderType::DateTimeUtcSecond},
|
|
|
|
{QStringLiteral("{DB_DIR}"), PlaceholderType::DbDir}};
|
2017-10-14 12:23:22 -04:00
|
|
|
|
2017-10-16 03:35:40 -04:00
|
|
|
return placeholders.value(placeholder.toUpper(), PlaceholderType::Unknown);
|
2012-07-12 16:33:20 -04:00
|
|
|
}
|
2017-06-29 13:54:49 -04:00
|
|
|
|
|
|
|
QString Entry::resolveUrl(const QString& url) const
|
|
|
|
{
|
2017-07-13 17:10:15 -04:00
|
|
|
QString newUrl = url;
|
2017-06-29 13:54:49 -04:00
|
|
|
|
2018-09-20 23:49:56 -04:00
|
|
|
QRegExp fileRegEx("^([a-z]:)?[\\\\/]", Qt::CaseInsensitive, QRegExp::RegExp2);
|
|
|
|
if (fileRegEx.indexIn(newUrl) != -1) {
|
|
|
|
// Match possible file paths without the scheme and convert it to a file URL
|
|
|
|
newUrl = QDir::fromNativeSeparators(newUrl);
|
|
|
|
newUrl = QUrl::fromLocalFile(newUrl).toString();
|
|
|
|
} else if (newUrl.startsWith("cmd://")) {
|
2017-10-03 13:11:00 -04:00
|
|
|
QStringList cmdList = newUrl.split(" ");
|
2018-03-31 16:01:30 -04:00
|
|
|
for (int i = 1; i < cmdList.size(); ++i) {
|
2017-10-03 13:11:00 -04:00
|
|
|
// Don't pass arguments to the resolveUrl function (they look like URL's)
|
|
|
|
if (!cmdList[i].startsWith("-") && !cmdList[i].startsWith("/")) {
|
|
|
|
return resolveUrl(cmdList[i].remove(QRegExp("'|\"")));
|
2017-07-13 17:10:15 -04:00
|
|
|
}
|
2017-07-01 08:51:20 -04:00
|
|
|
}
|
2017-10-03 13:11:00 -04:00
|
|
|
|
|
|
|
// No URL in this command
|
|
|
|
return QString("");
|
2017-07-01 08:51:20 -04:00
|
|
|
}
|
2017-10-03 13:11:00 -04:00
|
|
|
|
2018-09-20 23:49:56 -04:00
|
|
|
if (!newUrl.isEmpty() && !newUrl.contains("://")) {
|
|
|
|
// URL doesn't have a protocol, add https by default
|
|
|
|
newUrl.prepend("https://");
|
|
|
|
}
|
|
|
|
|
2017-10-03 13:11:00 -04:00
|
|
|
// Validate the URL
|
|
|
|
QUrl tempUrl = QUrl(newUrl);
|
2018-10-31 23:27:38 -04:00
|
|
|
if (tempUrl.isValid()
|
|
|
|
&& (tempUrl.scheme() == "http" || tempUrl.scheme() == "https" || tempUrl.scheme() == "file")) {
|
2017-10-03 13:11:00 -04:00
|
|
|
return tempUrl.url();
|
|
|
|
}
|
|
|
|
|
|
|
|
// No valid http URL's found
|
2017-07-01 08:51:20 -04:00
|
|
|
return QString("");
|
2017-09-29 14:10:54 -04:00
|
|
|
}
|
2018-09-30 08:45:06 -04:00
|
|
|
|
|
|
|
bool EntryData::operator==(const EntryData& other) const
|
|
|
|
{
|
|
|
|
return equals(other, CompareItemDefault);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool EntryData::operator!=(const EntryData& other) const
|
|
|
|
{
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool EntryData::equals(const EntryData& other, CompareItemOptions options) const
|
|
|
|
{
|
|
|
|
if (::compare(iconNumber, other.iconNumber, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(customIcon, other.customIcon, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(foregroundColor, other.foregroundColor, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(backgroundColor, other.backgroundColor, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(overrideUrl, other.overrideUrl, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(tags, other.tags, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(autoTypeEnabled, other.autoTypeEnabled, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(autoTypeObfuscation, other.autoTypeObfuscation, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(defaultAutoTypeSequence, other.defaultAutoTypeSequence, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!timeInfo.equals(other.timeInfo, options)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!totpSettings.isNull() && !other.totpSettings.isNull()) {
|
|
|
|
// Both have TOTP settings, compare them
|
|
|
|
if (::compare(totpSettings->key, other.totpSettings->key, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(totpSettings->digits, other.totpSettings->digits, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (::compare(totpSettings->step, other.totpSettings->step, options) != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (totpSettings.isNull() != other.totpSettings.isNull()) {
|
|
|
|
// The existance of TOTP has changed between these entries
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|