mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2026-03-11 08:54:48 +00:00
This plugin implements the Secret Storage specification version 0.2. While running KeePassXC, it acts as a Secret Service server, registered on DBus, so clients like seahorse, python-secretstorage, or other implementations can connect and access the exposed database in KeePassXC. Squashed commits: - Initial code - Add SessionAdaptor and fix build - The skeletons for all dbus objects are in place - Implement collection creation and deletion - Emit collectionChanged signal - Implement app-wise settings page - Implement error message on GUI - Implement settings - Fix uuid to dbus path - Implement app level settings - Add freedesktop logo - Implement database settings page - Change database settings to a treeview - Move all settings read/write to one place - Rename SecretServiceOptionsPage to SettingsWidgetFdoSecrets - Fix selected group can not be saved if the user hasn't click on the item - Show selected group per database in app settings - Disable editing of various readonly widgets - Remove unused warning about non exposed database - Fix method signature on dbus adaptors - Fix type derived from DBusObject not recognized as QDBusContext - Resolve a few TODOs around error handling - Remove const when passing DBus exposed objects - Move dismiss to PromptBase - Implement per collection locking/unlocking - Fix const correctness on Item::setSecret - Implement SecretService::getSecrets - Rework the signal connections around collections. - Remove generateId from DBusObject - Per spec, use encoded label as DBus object path for collections - Fix some corner cases around collection name changes - Implement alias - Fix wrong alias dbus path - Implement encryption per spec - Cleanup SessionCipher - Implement searchItems for SecretService - Use Tools::uuidToHex - Implement Item attributes and delete - Implement createItem - Always check if the database is unlocked before perform any operation - Add missing ReadAlias/SetAlias on service - Reorganize and fix OpenSession always returning empty output - Overhaul error handling - Make sure default alias is always present - Remove collection aliases early in doDelete - Handles all content types, fix setProperties not working - Fix sometimes there is an extraneous leading zero when converting from MPI - Fix session encryption negotiation - Do not expose recycle bin - Protect against the methods not called from DBus - Also emit collectionChanged signal when lock state changes - Show notification when entry secret is requested - Add a README file - Actually close session when client disconnects - Gracefully return alternative label when collection is locked - Reorganize, rename secretservice to fdosecrets - Fix issues reported by clazy - Unify UI strings and fix icon - Implement a setting to skip confirmation when deleting entries from DBus - Remove some unused debugging log - Simply ignore errors when DBus context is not available. QtDBus won't set QDBusContext when deliver property get/set, and there is no way to get a QDBusMessage in property getter/setter. - Simplify GcryptMPI using std::unique_ptr and add unit test - Format code in fdosecrets - Move DBusReturnImpl to details namespace - Fix crash when locking a database: don't modify exposedGroup setting in customData when database is deleted - Make sure Collection::searchItems works, whether it's locked or not - Fix FdoSecrets::Collection becomes empty after a database reload - Fix crash when looping while modifying the list
178 lines
4.9 KiB
C++
178 lines
4.9 KiB
C++
/*
|
|
* Copyright (C) 2018 Aetf <aetf@unlimitedcodeworks.xyz>
|
|
*
|
|
* 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/>.
|
|
*/
|
|
|
|
#ifndef KEEPASSXC_FDOSECRETS_DBUSOBJECT_H
|
|
#define KEEPASSXC_FDOSECRETS_DBUSOBJECT_H
|
|
|
|
#include "fdosecrets/objects/DBusReturn.h"
|
|
#include "fdosecrets/objects/DBusTypes.h"
|
|
|
|
#include <QDBusConnection>
|
|
#include <QDBusConnectionInterface>
|
|
#include <QDBusContext>
|
|
#include <QDBusObjectPath>
|
|
#include <QDebug>
|
|
#include <QList>
|
|
#include <QMetaProperty>
|
|
#include <QObject>
|
|
#include <QScopedPointer>
|
|
|
|
class QDBusAbstractAdaptor;
|
|
|
|
namespace FdoSecrets
|
|
{
|
|
class Service;
|
|
|
|
/**
|
|
* @brief A common base class for all dbus-exposed objects
|
|
*/
|
|
class DBusObject : public QObject, public QDBusContext
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
explicit DBusObject(DBusObject* parent = nullptr);
|
|
|
|
const QDBusObjectPath& objectPath() const
|
|
{
|
|
return m_objectPath;
|
|
}
|
|
|
|
protected:
|
|
void registerWithPath(const QString& path, QDBusAbstractAdaptor* adaptor);
|
|
|
|
void unregisterCurrentPath()
|
|
{
|
|
QDBusConnection::sessionBus().unregisterObject(m_objectPath.path());
|
|
m_dbusAdaptor = nullptr;
|
|
m_objectPath.setPath(QStringLiteral("/"));
|
|
}
|
|
|
|
QString callingPeer() const
|
|
{
|
|
Q_ASSERT(calledFromDBus());
|
|
return message().service();
|
|
}
|
|
|
|
uint callingPeerPid() const
|
|
{
|
|
return connection().interface()->servicePid(callingPeer());
|
|
}
|
|
|
|
QString callingPeerName() const;
|
|
|
|
template <typename Adaptor> Adaptor& dbusAdaptor() const
|
|
{
|
|
return *static_cast<Adaptor*>(m_dbusAdaptor);
|
|
}
|
|
|
|
DBusObject* p() const
|
|
{
|
|
return qobject_cast<DBusObject*>(parent());
|
|
}
|
|
|
|
private:
|
|
/**
|
|
* Derived class should not directly use sendErrorReply.
|
|
* Instead, use raiseError
|
|
*/
|
|
using QDBusContext::sendErrorReply;
|
|
|
|
template <typename U> friend class DBusReturn;
|
|
|
|
private:
|
|
QDBusAbstractAdaptor* m_dbusAdaptor;
|
|
QDBusObjectPath m_objectPath;
|
|
};
|
|
|
|
/**
|
|
* Return the object path of the pointed DBusObject, or "/" if the pointer is null
|
|
* @tparam T
|
|
* @param object
|
|
* @return
|
|
*/
|
|
template <typename T> QDBusObjectPath objectPathSafe(T* object)
|
|
{
|
|
if (object) {
|
|
return object->objectPath();
|
|
}
|
|
return QDBusObjectPath(QStringLiteral("/"));
|
|
}
|
|
|
|
/**
|
|
* Convert a list of DBusObjects to object path
|
|
* @tparam T
|
|
* @param objects
|
|
* @return
|
|
*/
|
|
template <typename T> QList<QDBusObjectPath> objectsToPath(QList<T*> objects)
|
|
{
|
|
QList<QDBusObjectPath> res;
|
|
res.reserve(objects.size());
|
|
for (auto object : objects) {
|
|
res.append(objectPathSafe(object));
|
|
}
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* Convert an object path to a pointer of the object
|
|
* @tparam T
|
|
* @param path
|
|
* @return the pointer of the object, or nullptr if path is "/"
|
|
*/
|
|
template <typename T> T* pathToObject(const QDBusObjectPath& path)
|
|
{
|
|
if (path.path() == QStringLiteral("/")) {
|
|
return nullptr;
|
|
}
|
|
return qobject_cast<T*>(QDBusConnection::sessionBus().objectRegisteredAt(path.path()));
|
|
}
|
|
|
|
/**
|
|
* Convert a list of object paths to a list of objects.
|
|
* "/" paths (i.e. nullptrs) will be skipped in the resulting list
|
|
* @tparam T
|
|
* @param paths
|
|
* @return
|
|
*/
|
|
template <typename T> QList<T*> pathsToObject(const QList<QDBusObjectPath>& paths)
|
|
{
|
|
QList<T*> res;
|
|
res.reserve(paths.size());
|
|
for (const auto& path : paths) {
|
|
auto object = pathToObject<T>(path);
|
|
if (object) {
|
|
res.append(object);
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* Encode the string value to a DBus object path safe representation,
|
|
* using a schema similar to URI encoding, but with percentage(%) replaced with
|
|
* underscore(_). All characters except [A-Za-z0-9] are encoded. For non-ascii
|
|
* characters, UTF-8 encoding is first applied and each of the resulting byte
|
|
* value is encoded.
|
|
* @param value
|
|
* @return encoded string
|
|
*/
|
|
QString encodePath(const QString& value);
|
|
|
|
} // namespace FdoSecrets
|
|
|
|
#endif // KEEPASSXC_FDOSECRETS_DBUSOBJECT_H
|