mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-02-14 05:31:34 -05:00
325 lines
8.2 KiB
C++
325 lines
8.2 KiB
C++
/*
|
|
* Copyright (C) 2012 Felix Geyer <debfx@fobos.de>
|
|
* Copyright (C) 2017 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/>.
|
|
*/
|
|
|
|
#include "EntryAttributes.h"
|
|
|
|
#include "core/Global.h"
|
|
|
|
#include <QRegularExpression>
|
|
#include <QUuid>
|
|
|
|
const QString EntryAttributes::TitleKey = "Title";
|
|
const QString EntryAttributes::UserNameKey = "UserName";
|
|
const QString EntryAttributes::PasswordKey = "Password";
|
|
const QString EntryAttributes::URLKey = "URL";
|
|
const QString EntryAttributes::NotesKey = "Notes";
|
|
const QStringList EntryAttributes::DefaultAttributes(QStringList()
|
|
<< TitleKey << UserNameKey << PasswordKey << URLKey << NotesKey);
|
|
|
|
const QString EntryAttributes::WantedFieldGroupName = "WantedField";
|
|
const QString EntryAttributes::SearchInGroupName = "SearchIn";
|
|
const QString EntryAttributes::SearchTextGroupName = "SearchText";
|
|
|
|
const QString EntryAttributes::RememberCmdExecAttr = "_EXEC_CMD";
|
|
|
|
EntryAttributes::EntryAttributes(QObject* parent)
|
|
: ModifiableObject(parent)
|
|
{
|
|
clear();
|
|
}
|
|
|
|
QList<QString> EntryAttributes::keys() const
|
|
{
|
|
return m_attributes.keys();
|
|
}
|
|
|
|
bool EntryAttributes::hasKey(const QString& key) const
|
|
{
|
|
return m_attributes.contains(key);
|
|
}
|
|
|
|
QList<QString> EntryAttributes::customKeys() const
|
|
{
|
|
QList<QString> customKeys;
|
|
const QList<QString> keyList = keys();
|
|
for (const QString& key : keyList) {
|
|
if (!isDefaultAttribute(key)) {
|
|
customKeys.append(key);
|
|
}
|
|
}
|
|
return customKeys;
|
|
}
|
|
|
|
QString EntryAttributes::value(const QString& key) const
|
|
{
|
|
return m_attributes.value(key);
|
|
}
|
|
|
|
QList<QString> EntryAttributes::values(const QList<QString>& keys) const
|
|
{
|
|
QList<QString> values;
|
|
for (const QString& key : keys) {
|
|
values.append(m_attributes.value(key));
|
|
}
|
|
return values;
|
|
}
|
|
|
|
bool EntryAttributes::contains(const QString& key) const
|
|
{
|
|
return m_attributes.contains(key);
|
|
}
|
|
|
|
bool EntryAttributes::containsValue(const QString& value) const
|
|
{
|
|
return asConst(m_attributes).values().contains(value);
|
|
}
|
|
|
|
bool EntryAttributes::isProtected(const QString& key) const
|
|
{
|
|
return m_protectedAttributes.contains(key);
|
|
}
|
|
|
|
bool EntryAttributes::isReference(const QString& key) const
|
|
{
|
|
if (!m_attributes.contains(key)) {
|
|
Q_ASSERT(false);
|
|
return false;
|
|
}
|
|
|
|
const QString data = value(key);
|
|
return matchReference(data).hasMatch();
|
|
}
|
|
|
|
void EntryAttributes::set(const QString& key, const QString& value, bool protect)
|
|
{
|
|
bool shouldEmitModified = false;
|
|
|
|
bool addAttribute = !m_attributes.contains(key);
|
|
bool changeValue = !addAttribute && (m_attributes.value(key) != value);
|
|
bool defaultAttribute = isDefaultAttribute(key);
|
|
|
|
if (addAttribute && !defaultAttribute) {
|
|
emit aboutToBeAdded(key);
|
|
}
|
|
|
|
if (addAttribute || changeValue) {
|
|
m_attributes.insert(key, value);
|
|
shouldEmitModified = true;
|
|
}
|
|
|
|
if (protect) {
|
|
if (!m_protectedAttributes.contains(key)) {
|
|
shouldEmitModified = true;
|
|
}
|
|
m_protectedAttributes.insert(key);
|
|
} else if (m_protectedAttributes.remove(key)) {
|
|
shouldEmitModified = true;
|
|
}
|
|
|
|
if (shouldEmitModified) {
|
|
emitModified();
|
|
}
|
|
|
|
if (defaultAttribute && changeValue) {
|
|
emit defaultKeyModified();
|
|
} else if (addAttribute) {
|
|
emit added(key);
|
|
} else if (shouldEmitModified) {
|
|
emit customKeyModified(key);
|
|
}
|
|
}
|
|
|
|
void EntryAttributes::remove(const QString& key)
|
|
{
|
|
Q_ASSERT(!isDefaultAttribute(key));
|
|
|
|
if (!m_attributes.contains(key)) {
|
|
return;
|
|
}
|
|
|
|
emit aboutToBeRemoved(key);
|
|
|
|
m_attributes.remove(key);
|
|
m_protectedAttributes.remove(key);
|
|
|
|
emit removed(key);
|
|
emitModified();
|
|
}
|
|
|
|
void EntryAttributes::rename(const QString& oldKey, const QString& newKey)
|
|
{
|
|
Q_ASSERT(!isDefaultAttribute(oldKey));
|
|
Q_ASSERT(!isDefaultAttribute(newKey));
|
|
|
|
if (!m_attributes.contains(oldKey)) {
|
|
Q_ASSERT(false);
|
|
return;
|
|
}
|
|
|
|
if (m_attributes.contains(newKey)) {
|
|
Q_ASSERT(false);
|
|
return;
|
|
}
|
|
|
|
QString data = value(oldKey);
|
|
bool protect = isProtected(oldKey);
|
|
|
|
emit aboutToRename(oldKey, newKey);
|
|
|
|
m_attributes.remove(oldKey);
|
|
m_attributes.insert(newKey, data);
|
|
if (protect) {
|
|
m_protectedAttributes.remove(oldKey);
|
|
m_protectedAttributes.insert(newKey);
|
|
}
|
|
|
|
emitModified();
|
|
emit renamed(oldKey, newKey);
|
|
}
|
|
|
|
void EntryAttributes::copyCustomKeysFrom(const EntryAttributes* other)
|
|
{
|
|
if (!areCustomKeysDifferent(other)) {
|
|
return;
|
|
}
|
|
|
|
emit aboutToBeReset();
|
|
|
|
// remove all non-default keys
|
|
const QList<QString> keyList = keys();
|
|
for (const QString& key : keyList) {
|
|
if (!isDefaultAttribute(key)) {
|
|
m_attributes.remove(key);
|
|
m_protectedAttributes.remove(key);
|
|
}
|
|
}
|
|
|
|
const QList<QString> otherKeyList = other->keys();
|
|
for (const QString& key : otherKeyList) {
|
|
if (!isDefaultAttribute(key)) {
|
|
m_attributes.insert(key, other->value(key));
|
|
if (other->isProtected(key)) {
|
|
m_protectedAttributes.insert(key);
|
|
}
|
|
}
|
|
}
|
|
|
|
emit reset();
|
|
emitModified();
|
|
}
|
|
|
|
bool EntryAttributes::areCustomKeysDifferent(const EntryAttributes* other)
|
|
{
|
|
// check if they are equal ignoring the order of the keys
|
|
if (keys().toSet() != other->keys().toSet()) {
|
|
return true;
|
|
}
|
|
|
|
const QList<QString> keyList = keys();
|
|
for (const QString& key : keyList) {
|
|
if (isDefaultAttribute(key)) {
|
|
continue;
|
|
}
|
|
|
|
if (isProtected(key) != other->isProtected(key) || value(key) != other->value(key)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void EntryAttributes::copyDataFrom(const EntryAttributes* other)
|
|
{
|
|
if (*this != *other) {
|
|
emit aboutToBeReset();
|
|
|
|
m_attributes = other->m_attributes;
|
|
m_protectedAttributes = other->m_protectedAttributes;
|
|
|
|
emit reset();
|
|
emitModified();
|
|
}
|
|
}
|
|
|
|
QUuid EntryAttributes::referenceUuid(const QString& key) const
|
|
{
|
|
if (!m_attributes.contains(key)) {
|
|
Q_ASSERT(false);
|
|
return {};
|
|
}
|
|
|
|
auto match = matchReference(value(key));
|
|
if (match.hasMatch()) {
|
|
const QString uuid = match.captured("SearchText");
|
|
if (!uuid.isEmpty()) {
|
|
return QUuid::fromRfc4122(QByteArray::fromHex(uuid.toLatin1()));
|
|
}
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
bool EntryAttributes::operator==(const EntryAttributes& other) const
|
|
{
|
|
return (m_attributes == other.m_attributes && m_protectedAttributes == other.m_protectedAttributes);
|
|
}
|
|
|
|
bool EntryAttributes::operator!=(const EntryAttributes& other) const
|
|
{
|
|
return (m_attributes != other.m_attributes || m_protectedAttributes != other.m_protectedAttributes);
|
|
}
|
|
|
|
QRegularExpressionMatch EntryAttributes::matchReference(const QString& text)
|
|
{
|
|
static QRegularExpression referenceRegExp(
|
|
"\\{REF:(?<WantedField>[TUPANI])@(?<SearchIn>[TUPANIO]):(?<SearchText>[^}]+)\\}",
|
|
QRegularExpression::CaseInsensitiveOption);
|
|
|
|
return referenceRegExp.match(text);
|
|
}
|
|
|
|
void EntryAttributes::clear()
|
|
{
|
|
emit aboutToBeReset();
|
|
|
|
m_attributes.clear();
|
|
m_protectedAttributes.clear();
|
|
|
|
for (const QString& key : DefaultAttributes) {
|
|
m_attributes.insert(key, "");
|
|
}
|
|
|
|
emit reset();
|
|
emitModified();
|
|
}
|
|
|
|
int EntryAttributes::attributesSize() const
|
|
{
|
|
int size = 0;
|
|
for (auto it = m_attributes.constBegin(); it != m_attributes.constEnd(); ++it) {
|
|
size += it.key().toUtf8().size() + it.value().toUtf8().size();
|
|
}
|
|
return size;
|
|
}
|
|
|
|
bool EntryAttributes::isDefaultAttribute(const QString& key)
|
|
{
|
|
return DefaultAttributes.contains(key);
|
|
}
|