Use org.freedesktop.DBus.Monitoring to monitor notifications

Plasma notification widget also uses the interface, and sometimes
notification ids can become out of sync between the two DBus adaptors.

BUG: 447385
FIXED-IN: 23.08
This commit is contained in:
Fushan Wen 2023-03-13 02:20:12 +00:00
parent f9404b83ed
commit 4523ba5882
4 changed files with 181 additions and 76 deletions

View file

@ -85,6 +85,7 @@ else()
find_package(Qt${QT_MAJOR_VERSION} REQUIRED COMPONENTS WaylandClient) find_package(Qt${QT_MAJOR_VERSION} REQUIRED COMPONENTS WaylandClient)
find_package(WaylandProtocols REQUIRED) find_package(WaylandProtocols REQUIRED)
pkg_check_modules(XkbCommon IMPORTED_TARGET xkbcommon) pkg_check_modules(XkbCommon IMPORTED_TARGET xkbcommon)
pkg_check_modules(GIO REQUIRED IMPORTED_TARGET gio-2.0)
endif() endif()
find_package(KF5PeopleVCard) find_package(KF5PeopleVCard)

View file

@ -30,6 +30,7 @@ target_link_libraries(kdeconnect_sendnotifications
Qt::Gui Qt::Gui
KF5::IconThemes KF5::IconThemes
KF5::ConfigCore KF5::ConfigCore
PkgConfig::GIO
) )
####################################### #######################################

View file

@ -5,13 +5,17 @@
*/ */
#include "notificationslistener.h" #include "notificationslistener.h"
#include <unordered_map>
#include <KConfig> #include <KConfig>
#include <KConfigGroup> #include <KConfigGroup>
#include <QDBusArgument> #include <QDBusArgument>
#include <QDBusInterface> #include <QDBusInterface>
#include <QImage> #include <QImage>
#include <QScopeGuard>
#include <QStandardPaths> #include <QStandardPaths>
#include <QtDebug> #include <QtDebug>
#include <kiconloader.h> #include <kiconloader.h>
#include <kicontheme.h> #include <kicontheme.h>
@ -28,20 +32,26 @@
#include "qtcompat_p.h" #include "qtcompat_p.h"
NotificationsListener::NotificationsListener(KdeConnectPlugin *aPlugin) NotificationsListener::NotificationsListener(KdeConnectPlugin *aPlugin)
: QDBusAbstractAdaptor(aPlugin) : QObject(aPlugin)
, m_plugin(aPlugin) , m_plugin(aPlugin)
{ {
qRegisterMetaTypeStreamOperators<NotifyingApplication>("NotifyingApplication"); qRegisterMetaTypeStreamOperators<NotifyingApplication>("NotifyingApplication");
bool ret = QDBusConnection::sessionBus().registerObject(QStringLiteral("/org/freedesktop/Notifications"), this, QDBusConnection::ExportScriptableContents); GError *error = nullptr;
if (!ret) m_gdbusConnection = g_bus_get_sync(G_BUS_TYPE_SESSION, nullptr, &error);
qCWarning(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Error registering notifications listener for device" << m_plugin->device()->name() << ":" g_assert_no_error(error);
<< QDBusConnection::sessionBus().lastError(); m_gdbusFilterId = g_dbus_connection_add_filter(m_gdbusConnection, NotificationsListener::onMessageFiltered, this, nullptr);
else
qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Registered notifications listener for device" << m_plugin->device()->name();
QDBusInterface iface(QStringLiteral("org.freedesktop.DBus"), QStringLiteral("/org/freedesktop/DBus"), QStringLiteral("org.freedesktop.DBus")); g_autoptr(GDBusMessage) msg =
iface.call(QStringLiteral("AddMatch"), QStringLiteral("interface='org.freedesktop.Notifications',member='Notify',type='method_call',eavesdrop='true'")); g_dbus_message_new_method_call("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus.Monitoring", "BecomeMonitor");
GVariantBuilder *arrayBuilder = g_variant_builder_new(G_VARIANT_TYPE("as"));
g_variant_builder_add(arrayBuilder, "s", "interface='org.freedesktop.Notifications'");
g_variant_builder_add(arrayBuilder, "s", "member='Notify'");
g_dbus_message_set_body(msg, g_variant_new("(asu)", arrayBuilder, 0u));
g_dbus_connection_send_message(m_gdbusConnection, msg, GDBusSendMessageFlags::G_DBUS_SEND_MESSAGE_FLAGS_NONE, nullptr, &error);
g_assert_no_error(error);
setTranslatedAppName(); setTranslatedAppName();
loadApplications(); loadApplications();
@ -52,10 +62,8 @@ NotificationsListener::NotificationsListener(KdeConnectPlugin *aPlugin)
NotificationsListener::~NotificationsListener() NotificationsListener::~NotificationsListener()
{ {
qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Destroying NotificationsListener"; qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Destroying NotificationsListener";
QDBusInterface iface(QStringLiteral("org.freedesktop.DBus"), QStringLiteral("/org/freedesktop/DBus"), QStringLiteral("org.freedesktop.DBus")); g_dbus_connection_remove_filter(m_gdbusConnection, m_gdbusFilterId);
QDBusMessage res = iface.call(QStringLiteral("RemoveMatch"), g_object_unref(m_gdbusConnection);
QStringLiteral("interface='org.freedesktop.Notifications',member='Notify',type='method_call',eavesdrop='true'"));
QDBusConnection::sessionBus().unregisterObject(QStringLiteral("/org/freedesktop/Notifications"));
} }
void NotificationsListener::setTranslatedAppName() void NotificationsListener::setTranslatedAppName()
@ -86,7 +94,7 @@ void NotificationsListener::loadApplications()
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Loaded" << applications.size() << " applications"; // qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Loaded" << applications.size() << " applications";
} }
bool NotificationsListener::parseImageDataArgument(const QVariant &argument, bool NotificationsListener::parseImageDataArgument(GVariant *argument,
int &width, int &width,
int &height, int &height,
int &rowStride, int &rowStride,
@ -95,16 +103,58 @@ bool NotificationsListener::parseImageDataArgument(const QVariant &argument,
bool &hasAlpha, bool &hasAlpha,
QByteArray &imageData) const QByteArray &imageData) const
{ {
if (!argument.canConvert<QDBusArgument>()) if (g_variant_n_children(argument) != 7) {
return false; return false;
const QDBusArgument dbusArg = argument.value<QDBusArgument>(); }
dbusArg.beginStructure();
dbusArg >> width >> height >> rowStride >> hasAlpha >> bitsPerSample >> channels >> imageData; g_autoptr(GVariant) variant;
dbusArg.endStructure();
variant = g_variant_get_child_value(argument, 0);
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
return false;
}
width = g_variant_get_int32(variant);
variant = g_variant_get_child_value(argument, 1);
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
return false;
}
height = g_variant_get_int32(variant);
variant = g_variant_get_child_value(argument, 2);
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
return false;
}
rowStride = g_variant_get_int32(variant);
variant = g_variant_get_child_value(argument, 3);
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_BOOLEAN)) {
return false;
}
hasAlpha = g_variant_get_boolean(variant);
variant = g_variant_get_child_value(argument, 4);
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
return false;
}
bitsPerSample = g_variant_get_int32(variant);
variant = g_variant_get_child_value(argument, 5);
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
return false;
}
channels = g_variant_get_int32(variant);
variant = g_variant_get_child_value(argument, 6);
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_ARRAY)) {
return false;
}
imageData = g_variant_get_bytestring(variant);
return true; return true;
} }
QSharedPointer<QIODevice> NotificationsListener::iconForImageData(const QVariant &argument) const QSharedPointer<QIODevice> NotificationsListener::iconForImageData(GVariant *argument) const
{ {
int width, height, rowStride, bitsPerSample, channels; int width, height, rowStride, bitsPerSample, channels;
bool hasAlpha; bool hasAlpha;
@ -133,7 +183,7 @@ QSharedPointer<QIODevice> NotificationsListener::iconForImageData(const QVariant
return buffer; return buffer;
} }
QSharedPointer<QIODevice> NotificationsListener::iconForIconName(const QString &iconName) const QSharedPointer<QIODevice> NotificationsListener::iconForIconName(const QString &iconName)
{ {
int size = KIconLoader::SizeEnormous; // use big size to allow for good int size = KIconLoader::SizeEnormous; // use big size to allow for good
// quality on high-DPI mobile devices // quality on high-DPI mobile devices
@ -154,65 +204,110 @@ QSharedPointer<QIODevice> NotificationsListener::iconForIconName(const QString &
return QSharedPointer<QIODevice>(); return QSharedPointer<QIODevice>();
} }
uint NotificationsListener::Notify(const QString &appName, GDBusMessage *NotificationsListener::onMessageFiltered(GDBusConnection *, GDBusMessage *msg, int, void *parent)
uint replacesId,
const QString &appIcon,
const QString &summary,
const QString &body,
const QStringList &actions,
const QVariantMap &hints,
int timeout)
{ {
static int id = 0; static unsigned id = 0;
Q_UNUSED(actions); if (!msg) {
return msg;
}
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Got notification appName=" << appName << "replacesId=" << replacesId << "appIcon=" << appIcon << const gchar *interface = g_dbus_message_get_interface(msg);
// "summary=" << summary << "body=" << body << "actions=" << actions << "hints=" << hints << "timeout=" << timeout; if (!interface || strcmp(interface, "org.freedesktop.Notifications")) {
// The first message will be from org.freedesktop.DBus.Monitoring
return msg;
}
const gchar *member = g_dbus_message_get_member(msg);
if (!member || strcmp(member, "Notify")) {
// Even if member is set, the monitor will still notify messages from other members.
return nullptr;
}
g_autoptr(GVariant) bodyVariant = g_dbus_message_get_body(msg);
Q_ASSERT(bodyVariant);
// Data order and types: https://specifications.freedesktop.org/notification-spec/notification-spec-latest.html
g_autoptr(GVariant) variant = g_variant_get_child_value(bodyVariant, 0);
const QString appName = QString::fromUtf8(g_variant_get_string(variant, nullptr));
// skip our own notifications // skip our own notifications
if (appName == m_translatedAppName) auto listener = static_cast<NotificationsListener *>(parent);
return 0;
if (appName == listener->m_translatedAppName) {
return nullptr;
}
variant = g_variant_get_child_value(bodyVariant, 2);
const QString appIcon = QString::fromUtf8(g_variant_get_string(variant, nullptr));
NotifyingApplication app; NotifyingApplication app;
if (!m_applications.contains(appName)) { if (!listener->m_applications.contains(appName)) {
// new application -> add to config // new application -> add to config
app.name = appName; app.name = appName;
app.icon = appIcon; app.icon = appIcon;
app.active = true; app.active = true;
app.blacklistExpression = QRegularExpression(); app.blacklistExpression = QRegularExpression();
m_applications.insert(app.name, app); listener->m_applications.insert(app.name, app);
// update config: // update config:
QVariantList list; QVariantList list;
for (const auto &a : qAsConst(m_applications)) for (const auto &a : std::as_const(listener->m_applications))
list << QVariant::fromValue<NotifyingApplication>(a); list << QVariant::fromValue<NotifyingApplication>(a);
m_plugin->config()->setList(QStringLiteral("applications"), list); listener->m_plugin->config()->setList(QStringLiteral("applications"), list);
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Added new application to config:" << app; // qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Added new application to config:" << app;
} else { } else {
app = m_applications.value(appName); app = listener->m_applications.value(appName);
} }
if (!app.active) if (!app.active) {
return 0; return nullptr;
}
if (timeout > 0 && m_plugin->config()->getBool(QStringLiteral("generalPersistent"), false)) variant = g_variant_get_child_value(bodyVariant, 7);
return 0; const auto timeout = g_variant_get_int32(variant);
if (timeout > 0 && listener->m_plugin->config()->getBool(QStringLiteral("generalPersistent"), false)) {
return nullptr;
}
variant = g_variant_get_child_value(bodyVariant, 6);
std::unordered_map<QString, GVariant *> hints;
GVariantIter *iter;
const gchar *key;
GVariant *value = nullptr;
g_variant_get(variant, "a{sv}", &iter);
while (g_variant_iter_loop(iter, "{sv}", &key, &value)) {
hints.emplace(QString::fromUtf8(key), value);
}
g_variant_iter_free(iter);
QScopeGuard cleanupHints([&hints] {
for (auto &[_, hint] : hints) {
g_variant_unref(hint);
}
});
int urgency = -1; int urgency = -1;
if (hints.contains(QStringLiteral("urgency"))) { if (auto it = hints.find(QStringLiteral("urgency")); it != hints.end()) {
bool ok; if (g_variant_is_of_type(it->second, G_VARIANT_TYPE_BYTE)) {
urgency = hints[QStringLiteral("urgency")].toInt(&ok); urgency = g_variant_get_byte(it->second);
if (!ok) }
urgency = -1;
} }
if (urgency > -1 && urgency < m_plugin->config()->getInt(QStringLiteral("generalUrgency"), 0)) if (urgency > -1 && urgency < listener->m_plugin->config()->getInt(QStringLiteral("generalUrgency"), 0))
return 0; return nullptr;
QString ticker = summary; variant = g_variant_get_child_value(bodyVariant, 3);
if (!body.isEmpty() && m_plugin->config()->getBool(QStringLiteral("generalIncludeBody"), true)) QString ticker = QString::fromUtf8(g_variant_get_string(variant, nullptr));
variant = g_variant_get_child_value(bodyVariant, 4);
const QString body = QString::fromUtf8(g_variant_get_string(variant, nullptr));
if (!body.isEmpty() && listener->m_plugin->config()->getBool(QStringLiteral("generalIncludeBody"), true)) {
ticker += QStringLiteral(": ") + body; ticker += QStringLiteral(": ") + body;
}
if (app.blacklistExpression.isValid() && !app.blacklistExpression.pattern().isEmpty() && app.blacklistExpression.match(ticker).hasMatch()) if (app.blacklistExpression.isValid() && !app.blacklistExpression.pattern().isEmpty() && app.blacklistExpression.match(ticker).hasMatch()) {
return 0; return nullptr;
}
variant = g_variant_get_child_value(bodyVariant, 1);
const unsigned replacesId = g_variant_get_uint32(variant);
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Sending notification from" << appName << ":" <<ticker << "; appIcon=" << appIcon; // qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Sending notification from" << appName << ":" <<ticker << "; appIcon=" << appIcon;
NetworkPacket np(PACKET_TYPE_NOTIFICATION, NetworkPacket np(PACKET_TYPE_NOTIFICATION,
@ -224,28 +319,33 @@ uint NotificationsListener::Notify(const QString &appName,
// clearability is pointless // clearability is pointless
// sync any icon data? // sync any icon data?
if (m_plugin->config()->getBool(QStringLiteral("generalSynchronizeIcons"), true)) { if (listener->m_plugin->config()->getBool(QStringLiteral("generalSynchronizeIcons"), true)) {
QSharedPointer<QIODevice> iconSource; QSharedPointer<QIODevice> iconSource;
// try different image sources according to priorities in notifications- // try different image sources according to priorities in notifications-
// spec version 1.2: // spec version 1.2:
if (hints.contains(QStringLiteral("image-data"))) if (auto it = hints.find(QStringLiteral("image-data")); it != hints.end()) {
iconSource = iconForImageData(hints[QStringLiteral("image-data")]); iconSource = listener->iconForImageData(it->second);
else if (hints.contains(QStringLiteral("image_data"))) // 1.1 backward compatibility } else if (auto it = hints.find(QStringLiteral("image_data")); it != hints.end()) { // 1.1 backward compatibility
iconSource = iconForImageData(hints[QStringLiteral("image_data")]); iconSource = listener->iconForImageData(it->second);
else if (hints.contains(QStringLiteral("image-path"))) } else if (auto it = hints.find(QStringLiteral("image-path")); it != hints.end()) {
iconSource = iconForIconName(hints[QStringLiteral("image-path")].toString()); iconSource = iconForIconName(QString::fromUtf8(g_variant_get_string(it->second, nullptr)));
else if (hints.contains(QStringLiteral("image_path"))) // 1.1 backward compatibility } else if (auto it = hints.find(QStringLiteral("image_path")); it != hints.end()) { // 1.1 backward compatibility
iconSource = iconForIconName(hints[QStringLiteral("image_path")].toString()); iconSource = iconForIconName(QString::fromUtf8(g_variant_get_string(it->second, nullptr)));
else if (!appIcon.isEmpty()) } else if (!appIcon.isEmpty()) {
iconSource = iconForIconName(appIcon); iconSource = iconForIconName(appIcon);
else if (hints.contains(QStringLiteral("icon_data"))) // < 1.1 backward compatibility } else if (auto it = hints.find(QStringLiteral("icon_data")); it != hints.end()) { // < 1.1 backward compatibility
iconSource = iconForImageData(hints[QStringLiteral("icon_data")]); iconSource = listener->iconForImageData(it->second);
}
if (iconSource) if (iconSource)
np.setPayload(iconSource, iconSource->size()); np.setPayload(iconSource, iconSource->size());
} }
m_plugin->sendPacket(np); listener->m_plugin->sendPacket(np);
return (replacesId > 0 ? replacesId : id); qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Got notification appName=" << appName << "replacesId=" << replacesId << "appIcon=" << appIcon
<< "summary=" << ticker << "body=" << body << "hints=" << hints.size() << "urgency=" << urgency
<< "timeout=" << timeout;
return nullptr;
} }

View file

@ -13,14 +13,15 @@
#include <QFile> #include <QFile>
#include <core/device.h> #include <core/device.h>
#include <gio/gio.h>
class KdeConnectPlugin; class KdeConnectPlugin;
class Notification; class Notification;
struct NotifyingApplication; struct NotifyingApplication;
class NotificationsListener : public QDBusAbstractAdaptor class NotificationsListener : public QObject
{ {
Q_OBJECT Q_OBJECT
Q_CLASSINFO("D-Bus Interface", "org.freedesktop.Notifications")
public: public:
explicit NotificationsListener(KdeConnectPlugin *aPlugin); explicit NotificationsListener(KdeConnectPlugin *aPlugin);
@ -32,7 +33,7 @@ protected:
// virtual helper function to make testing possible (QDBusArgument can not // virtual helper function to make testing possible (QDBusArgument can not
// be injected without making a DBUS-call): // be injected without making a DBUS-call):
virtual bool parseImageDataArgument(const QVariant &argument, virtual bool parseImageDataArgument(GVariant *argument,
int &width, int &width,
int &height, int &height,
int &rowStride, int &rowStride,
@ -40,11 +41,10 @@ protected:
int &channels, int &channels,
bool &hasAlpha, bool &hasAlpha,
QByteArray &imageData) const; QByteArray &imageData) const;
QSharedPointer<QIODevice> iconForImageData(const QVariant &argument) const; QSharedPointer<QIODevice> iconForImageData(GVariant *argument) const;
QSharedPointer<QIODevice> iconForIconName(const QString &iconName) const; static QSharedPointer<QIODevice> iconForIconName(const QString &iconName);
public Q_SLOTS: static GDBusMessage *onMessageFiltered(GDBusConnection *connection, GDBusMessage *msg, int incoming, void *parent);
Q_SCRIPTABLE uint Notify(const QString &, uint, const QString &, const QString &, const QString &, const QStringList &, const QVariantMap &, int);
private Q_SLOTS: private Q_SLOTS:
void loadApplications(); void loadApplications();
@ -52,6 +52,9 @@ private Q_SLOTS:
private: private:
void setTranslatedAppName(); void setTranslatedAppName();
QString m_translatedAppName; QString m_translatedAppName;
GDBusConnection *m_gdbusConnection = nullptr;
unsigned m_gdbusFilterId = 0;
}; };
#endif // NOTIFICATIONSLISTENER_H #endif // NOTIFICATIONSLISTENER_H