/*
* Copyright (C) 2016 Alberts Muktupāvels
*
* 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 3 of the License, or
* (at your option) any later version.
*
* 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 .
*/
#include
#include "gf-sn-watcher-v0.h"
struct _GfSnWatcherV0
{
GfSnWatcherV0GenSkeleton parent;
guint bus_name_id;
GSList *hosts;
GSList *items;
};
typedef enum
{
GF_WATCH_TYPE_HOST,
GF_WATCH_TYPE_ITEM
} GfWatchType;
typedef struct
{
GfSnWatcherV0 *v0;
GfWatchType type;
gchar *service;
gchar *bus_name;
gchar *object_path;
guint watch_id;
} GfWatch;
static void gf_sn_watcher_v0_gen_init (GfSnWatcherV0GenIface *iface);
G_DEFINE_TYPE_WITH_CODE (GfSnWatcherV0, gf_sn_watcher_v0, GF_TYPE_SN_WATCHER_V0_GEN_SKELETON,
G_IMPLEMENT_INTERFACE (GF_TYPE_SN_WATCHER_V0_GEN, gf_sn_watcher_v0_gen_init))
static void
update_registered_items (GfSnWatcherV0 *v0)
{
GVariantBuilder builder;
GSList *l;
GVariant *variant;
const gchar **items;
g_variant_builder_init (&builder, G_VARIANT_TYPE ("as"));
for (l = v0->items; l != NULL; l = g_slist_next (l))
{
GfWatch *watch;
gchar *item;
watch = (GfWatch *) l->data;
item = g_strdup_printf ("%s%s", watch->bus_name, watch->object_path);
g_variant_builder_add (&builder, "s", item);
g_free (item);
}
variant = g_variant_builder_end (&builder);
items = g_variant_get_strv (variant, NULL);
gf_sn_watcher_v0_gen_set_registered_items (GF_SN_WATCHER_V0_GEN (v0), items);
g_variant_unref (variant);
g_free (items);
}
static void
gf_watch_free (gpointer data)
{
GfWatch *watch;
watch = (GfWatch *) data;
if (watch->watch_id > 0)
g_bus_unwatch_name (watch->watch_id);
g_free (watch->service);
g_free (watch->bus_name);
g_free (watch->object_path);
g_free (watch);
}
static void
name_vanished_cb (GDBusConnection *connection,
const char *name,
gpointer user_data)
{
GfWatch *watch;
GfSnWatcherV0 *v0;
GfSnWatcherV0Gen *gen;
watch = (GfWatch *) user_data;
v0 = watch->v0;
gen = GF_SN_WATCHER_V0_GEN (v0);
if (watch->type == GF_WATCH_TYPE_HOST)
{
v0->hosts = g_slist_remove (v0->hosts, watch);
if (v0->hosts == NULL)
{
gf_sn_watcher_v0_gen_set_is_host_registered (gen, FALSE);
gf_sn_watcher_v0_gen_emit_host_registered (gen);
}
}
else if (watch->type == GF_WATCH_TYPE_ITEM)
{
gchar *tmp;
v0->items = g_slist_remove (v0->items, watch);
update_registered_items (v0);
tmp = g_strdup_printf ("%s%s", watch->bus_name, watch->object_path);
gf_sn_watcher_v0_gen_emit_item_unregistered (gen, tmp);
g_free (tmp);
}
else
{
g_assert_not_reached ();
}
gf_watch_free (watch);
}
static GfWatch *
gf_watch_new (GfSnWatcherV0 *v0,
GfWatchType type,
const gchar *service,
const gchar *bus_name,
const gchar *object_path)
{
GfWatch *watch;
watch = g_new0 (GfWatch, 1);
watch->v0 = v0;
watch->type = type;
watch->service = g_strdup (service);
watch->bus_name = g_strdup (bus_name);
watch->object_path = g_strdup (object_path);
watch->watch_id = g_bus_watch_name (G_BUS_TYPE_SESSION, bus_name,
G_BUS_NAME_WATCHER_FLAGS_NONE, NULL,
name_vanished_cb, watch, NULL);
return watch;
}
static GfWatch *
gf_watch_find (GSList *list,
const gchar *bus_name,
const gchar *object_path)
{
GSList *l;
for (l = list; l != NULL; l = g_slist_next (l))
{
GfWatch *watch;
watch = (GfWatch *) l->data;
if (g_strcmp0 (watch->bus_name, bus_name) == 0 &&
g_strcmp0 (watch->object_path, object_path) == 0)
{
return watch;
}
}
return NULL;
}
static gboolean
gf_sn_watcher_v0_handle_register_host (GfSnWatcherV0Gen *object,
GDBusMethodInvocation *invocation,
const gchar *service)
{
GfSnWatcherV0 *v0;
const gchar *bus_name;
const gchar *object_path;
GfWatch *watch;
v0 = GF_SN_WATCHER_V0 (object);
if (*service == '/')
{
bus_name = g_dbus_method_invocation_get_sender (invocation);
object_path = service;
}
else
{
bus_name = service;
object_path = "/StatusNotifierHost";
}
if (g_dbus_is_name (bus_name) == FALSE)
{
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
G_DBUS_ERROR_INVALID_ARGS,
"D-Bus bus name '%s' is not valid",
bus_name);
return TRUE;
}
watch = gf_watch_find (v0->hosts, bus_name, object_path);
if (watch != NULL)
{
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
G_DBUS_ERROR_INVALID_ARGS,
"Status Notifier Host with bus name '%s' and object path '%s' is already registered",
bus_name, object_path);
return TRUE;
}
watch = gf_watch_new (v0, GF_WATCH_TYPE_HOST, service, bus_name, object_path);
v0->hosts = g_slist_prepend (v0->hosts, watch);
if (!gf_sn_watcher_v0_gen_get_is_host_registered (object))
{
gf_sn_watcher_v0_gen_set_is_host_registered (object, TRUE);
gf_sn_watcher_v0_gen_emit_host_registered (object);
}
gf_sn_watcher_v0_gen_complete_register_host (object, invocation);
return TRUE;
}
static gboolean
gf_sn_watcher_v0_handle_register_item (GfSnWatcherV0Gen *object,
GDBusMethodInvocation *invocation,
const gchar *service)
{
GfSnWatcherV0 *v0;
const gchar *bus_name;
const gchar *object_path;
GfWatch *watch;
gchar *tmp;
v0 = GF_SN_WATCHER_V0 (object);
if (*service == '/')
{
bus_name = g_dbus_method_invocation_get_sender (invocation);
object_path = service;
}
else
{
bus_name = service;
object_path = "/StatusNotifierItem";
}
if (g_dbus_is_name (bus_name) == FALSE)
{
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
G_DBUS_ERROR_INVALID_ARGS,
"D-Bus bus name '%s' is not valid",
bus_name);
return TRUE;
}
watch = gf_watch_find (v0->items, bus_name, object_path);
if (watch != NULL)
{
/* the specification doesn't explicitly state what should happen when
* trying to register the same item again, so it would make sense to
* forbid it. Unfortunately libappindicator tries re-registering pretty
* often, and even falls back to System Tray if it fails.
* So in practice we need to be forgiving and pretend it's OK. */
#if 0
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR,
G_DBUS_ERROR_INVALID_ARGS,
"Status Notifier Item with bus name '%s' and object path '%s' is already registered",
bus_name, object_path);
#else
g_warning ("Status Notifier Item with bus name '%s' and object path '%s' is already registered",
bus_name, object_path);
/* FIXME: is it OK to simply ignore the request instead of removing the
* old one and adding the new one? I don't see the problem as
* they are identical, but...? */
gf_sn_watcher_v0_gen_complete_register_item (object, invocation);
#endif
return TRUE;
}
watch = gf_watch_new (v0, GF_WATCH_TYPE_ITEM, service, bus_name, object_path);
v0->items = g_slist_prepend (v0->items, watch);
update_registered_items (v0);
tmp = g_strdup_printf ("%s%s", bus_name, object_path);
gf_sn_watcher_v0_gen_emit_item_registered (object, tmp);
g_free (tmp);
gf_sn_watcher_v0_gen_complete_register_item (object, invocation);
return TRUE;
}
static void
gf_sn_watcher_v0_gen_init (GfSnWatcherV0GenIface *iface)
{
iface->handle_register_host = gf_sn_watcher_v0_handle_register_host;
iface->handle_register_item = gf_sn_watcher_v0_handle_register_item;
}
static void
bus_acquired_cb (GDBusConnection *connection,
const gchar *name,
gpointer user_data)
{
GfSnWatcherV0 *v0;
GDBusInterfaceSkeleton *skeleton;
GError *error;
v0 = GF_SN_WATCHER_V0 (user_data);
skeleton = G_DBUS_INTERFACE_SKELETON (v0);
error = NULL;
g_dbus_interface_skeleton_export (skeleton, connection,
"/StatusNotifierWatcher", &error);
if (error != NULL)
{
g_warning ("%s", error->message);
g_error_free (error);
return;
}
}
static void
gf_sn_watcher_v0_dispose (GObject *object)
{
GfSnWatcherV0 *v0;
v0 = GF_SN_WATCHER_V0 (object);
if (v0->bus_name_id > 0)
{
g_bus_unown_name (v0->bus_name_id);
v0->bus_name_id = 0;
}
if (v0->hosts != NULL)
{
g_slist_free_full (v0->hosts, gf_watch_free);
v0->hosts = NULL;
}
if (v0->items != NULL)
{
g_slist_free_full (v0->items, gf_watch_free);
v0->items = NULL;
}
G_OBJECT_CLASS (gf_sn_watcher_v0_parent_class)->dispose (object);
}
static void
gf_sn_watcher_v0_class_init (GfSnWatcherV0Class *v0_class)
{
GObjectClass *object_class;
object_class = G_OBJECT_CLASS (v0_class);
object_class->dispose = gf_sn_watcher_v0_dispose;
}
static void
gf_sn_watcher_v0_init (GfSnWatcherV0 *v0)
{
GBusNameOwnerFlags flags;
flags = G_BUS_NAME_OWNER_FLAGS_ALLOW_REPLACEMENT |
G_BUS_NAME_OWNER_FLAGS_REPLACE;
v0->bus_name_id = g_bus_own_name (G_BUS_TYPE_SESSION,
"org.kde.StatusNotifierWatcher", flags,
bus_acquired_cb, NULL, NULL, v0, NULL);
}
GfSnWatcherV0 *
gf_sn_watcher_v0_new (void)
{
return g_object_new (GF_TYPE_SN_WATCHER_V0, NULL);
}