Blame lib/bluetooth-settings-obexpush.c

Packit Service fc05fa
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 4; tab-width: 4 -*- */
Packit Service fc05fa
Packit Service fc05fa
/*
Packit Service fc05fa
 *  Copyright (C) 2004-2008 Red Hat, Inc.
Packit Service fc05fa
 *  Copyright (C) 2013 Intel Corporation.
Packit Service fc05fa
 *
Packit Service fc05fa
 *  Nautilus is free software; you can redistribute it and/or
Packit Service fc05fa
 *  modify it under the terms of the GNU General Public License as
Packit Service fc05fa
 *  published by the Free Software Foundation; either version 2 of the
Packit Service fc05fa
 *  License, or (at your option) any later version.
Packit Service fc05fa
 *
Packit Service fc05fa
 *  Nautilus is distributed in the hope that it will be useful,
Packit Service fc05fa
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service fc05fa
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service fc05fa
 *  General Public License for more details.
Packit Service fc05fa
 *
Packit Service fc05fa
 *  You should have received a copy of the GNU General Public License
Packit Service fc05fa
 *  along with this program; if not, write to the Free Software
Packit Service fc05fa
 *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
Packit Service fc05fa
 *
Packit Service fc05fa
 *  Authors: Bastien Nocera <hadess@hadess.net>
Packit Service fc05fa
 *  Authors: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
Packit Service fc05fa
 *
Packit Service fc05fa
 */
Packit Service fc05fa
Packit Service fc05fa
#include "config.h"
Packit Service fc05fa
Packit Service fc05fa
#include <glib.h>
Packit Service fc05fa
#include <glib/gstdio.h>
Packit Service fc05fa
#include <glib/gi18n-lib.h>
Packit Service fc05fa
#include <gio/gio.h>
Packit Service fc05fa
#include <gtk/gtk.h>
Packit Service fc05fa
#include <bluetooth-client.h>
Packit Service fc05fa
#include <libnotify/notify.h>
Packit Service fc05fa
#include <canberra-gtk.h>
Packit Service fc05fa
Packit Service fc05fa
#include "bluetooth-settings-obexpush.h"
Packit Service fc05fa
Packit Service fc05fa
#define MANAGER_SERVICE	"org.bluez.obex"
Packit Service fc05fa
#define MANAGER_IFACE	"org.bluez.obex.AgentManager1"
Packit Service fc05fa
#define MANAGER_PATH	"/org/bluez/obex"
Packit Service fc05fa
Packit Service fc05fa
#define AGENT_PATH	"/org/gnome/share/agent"
Packit Service fc05fa
#define AGENT_IFACE	"org.bluez.obex.Agent1"
Packit Service fc05fa
Packit Service fc05fa
#define TRANSFER_IFACE	"org.bluez.obex.Transfer1"
Packit Service fc05fa
#define SESSION_IFACE	"org.bluez.obex.Session1"
Packit Service fc05fa
Packit Service fc05fa
static GDBusNodeInfo *introspection_data = NULL;
Packit Service fc05fa
Packit Service fc05fa
static const gchar introspection_xml[] =
Packit Service fc05fa
"<node name='"AGENT_PATH"'>"
Packit Service fc05fa
"  <interface name='"AGENT_IFACE"'>"
Packit Service fc05fa
"    <method name='Release'>"
Packit Service fc05fa
"    </method>"
Packit Service fc05fa
"    <method name='Cancel'>"
Packit Service fc05fa
"    </method>"
Packit Service fc05fa
"    <method name='AuthorizePush'>"
Packit Service fc05fa
"      <arg name='transfer' type='o' />"
Packit Service fc05fa
"      <arg name='path' type='s' direction='out' />"
Packit Service fc05fa
"    </method>"
Packit Service fc05fa
"  </interface>"
Packit Service fc05fa
"</node>";
Packit Service fc05fa
Packit Service fc05fa
G_DEFINE_TYPE(ObexAgent, obex_agent, G_TYPE_OBJECT)
Packit Service fc05fa
Packit Service fc05fa
static ObexAgent *agent;
Packit Service fc05fa
static BluetoothClient *client;
Packit Service fc05fa
static GCancellable *cancellable;
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
on_close_notification (NotifyNotification *notification)
Packit Service fc05fa
{
Packit Service fc05fa
	g_object_unref (notification);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
notification_launch_action_on_file_cb (NotifyNotification *notification,
Packit Service fc05fa
				       const char *action,
Packit Service fc05fa
				       const char *file_uri)
Packit Service fc05fa
{
Packit Service fc05fa
	g_assert (action != NULL);
Packit Service fc05fa
Packit Service fc05fa
	/* We launch the file viewer for the file */
Packit Service fc05fa
	if (g_str_equal (action, "display") != FALSE) {
Packit Service fc05fa
		GdkDisplay *display;
Packit Service fc05fa
		GAppLaunchContext *ctx;
Packit Service fc05fa
		GTimeVal val;
Packit Service fc05fa
Packit Service fc05fa
		g_get_current_time (&val;;
Packit Service fc05fa
Packit Service fc05fa
		display = gdk_display_get_default ();
Packit Service fc05fa
		ctx = G_APP_LAUNCH_CONTEXT (gdk_display_get_app_launch_context (display));
Packit Service fc05fa
		gdk_app_launch_context_set_timestamp (GDK_APP_LAUNCH_CONTEXT (ctx), val.tv_sec);
Packit Service fc05fa
Packit Service fc05fa
		if (g_app_info_launch_default_for_uri (file_uri, ctx, NULL) == FALSE) {
Packit Service fc05fa
			g_warning ("Failed to launch the file viewer\n");
Packit Service fc05fa
		}
Packit Service fc05fa
Packit Service fc05fa
		g_object_unref (ctx);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	/* we open the Downloads folder */
Packit Service fc05fa
	if (g_str_equal (action, "reveal") != FALSE) {
Packit Service fc05fa
		GDBusConnection *connection = agent->connection;
Packit Service fc05fa
		GVariantBuilder builder;
Packit Service fc05fa
Packit Service fc05fa
		g_variant_builder_init (&builder, G_VARIANT_TYPE ("as"));
Packit Service fc05fa
		g_variant_builder_add (&builder, "s", file_uri);
Packit Service fc05fa
Packit Service fc05fa
		g_dbus_connection_call (connection,
Packit Service fc05fa
					"org.freedesktop.FileManager1",
Packit Service fc05fa
					"/org/freedesktop/FileManager1",
Packit Service fc05fa
					"org.freedesktop.FileManager1",
Packit Service fc05fa
					"ShowItems",
Packit Service fc05fa
					g_variant_new ("(ass)", &builder, ""),
Packit Service fc05fa
					NULL,
Packit Service fc05fa
					G_DBUS_CALL_FLAGS_NONE,
Packit Service fc05fa
					-1,
Packit Service fc05fa
					cancellable,
Packit Service fc05fa
					NULL,
Packit Service fc05fa
					NULL);
Packit Service fc05fa
Packit Service fc05fa
		g_variant_builder_clear (&builder);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	notify_notification_close (notification, NULL);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
show_notification (const char *filename)
Packit Service fc05fa
{
Packit Service fc05fa
	char *file_uri, *notification_text, *display, *mime_type;
Packit Service fc05fa
	NotifyNotification *notification;
Packit Service fc05fa
	ca_context *ctx;
Packit Service fc05fa
	GAppInfo *app;
Packit Service fc05fa
Packit Service fc05fa
	file_uri = g_filename_to_uri (filename, NULL, NULL);
Packit Service fc05fa
	if (file_uri == NULL) {
Packit Service fc05fa
		g_warning ("Could not make a filename from '%s'", filename);
Packit Service fc05fa
		return;
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	display = g_filename_display_basename (filename);
Packit Service fc05fa
	/* Translators: %s is the name of the filename received */
Packit Service fc05fa
	notification_text = g_strdup_printf(_("You received ā€œ%sā€ via Bluetooth"), display);
Packit Service fc05fa
	g_free (display);
Packit Service fc05fa
	notification = notify_notification_new (_("You received a file"),
Packit Service fc05fa
						notification_text,
Packit Service fc05fa
						"bluetooth");
Packit Service fc05fa
Packit Service fc05fa
	notify_notification_set_timeout (notification, NOTIFY_EXPIRES_DEFAULT);
Packit Service fc05fa
	notify_notification_set_hint_string (notification, "desktop-entry", "gnome-bluetooth-panel");
Packit Service fc05fa
Packit Service fc05fa
	mime_type = g_content_type_guess (filename, NULL, 0, NULL);
Packit Service fc05fa
	app = g_app_info_get_default_for_type (mime_type, FALSE);
Packit Service fc05fa
	if (app != NULL) {
Packit Service fc05fa
		g_object_unref (app);
Packit Service fc05fa
		notify_notification_add_action (notification, "display", _("Open File"),
Packit Service fc05fa
						(NotifyActionCallback) notification_launch_action_on_file_cb,
Packit Service fc05fa
						g_strdup (file_uri), (GFreeFunc) g_free);
Packit Service fc05fa
	}
Packit Service fc05fa
	notify_notification_add_action (notification, "reveal", _("Open Containing Folder"),
Packit Service fc05fa
					(NotifyActionCallback) notification_launch_action_on_file_cb,
Packit Service fc05fa
					g_strdup (file_uri), (GFreeFunc) g_free);
Packit Service fc05fa
Packit Service fc05fa
	g_free (file_uri);
Packit Service fc05fa
	
Packit Service fc05fa
	g_signal_connect (G_OBJECT (notification), "closed", G_CALLBACK (on_close_notification), notification);
Packit Service fc05fa
Packit Service fc05fa
	if (!notify_notification_show (notification, NULL)) {
Packit Service fc05fa
		g_warning ("failed to send notification\n");
Packit Service fc05fa
	}
Packit Service fc05fa
	g_free (notification_text);
Packit Service fc05fa
Packit Service fc05fa
	/* Now we do the audio notification */
Packit Service fc05fa
	ctx = ca_gtk_context_get ();
Packit Service fc05fa
	ca_context_play (ctx, 0,
Packit Service fc05fa
			 CA_PROP_EVENT_ID, "complete-download",
Packit Service fc05fa
			 CA_PROP_EVENT_DESCRIPTION, _("File reception complete"),
Packit Service fc05fa
			 NULL);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
reject_transfer (GDBusMethodInvocation *invocation)
Packit Service fc05fa
{
Packit Service fc05fa
	const char *filename;
Packit Service fc05fa
Packit Service fc05fa
	filename = g_object_get_data (G_OBJECT (invocation), "temp-filename");
Packit Service fc05fa
	g_remove (filename);
Packit Service fc05fa
Packit Service fc05fa
	g_dbus_method_invocation_return_dbus_error (invocation,
Packit Service fc05fa
		"org.bluez.obex.Error.Rejected", "Not Authorized");
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
ask_user_transfer_accepted (NotifyNotification *notification,
Packit Service fc05fa
			    char *action,
Packit Service fc05fa
			    GDBusMethodInvocation *invocation)
Packit Service fc05fa
{
Packit Service fc05fa
	gchar *file = g_object_get_data (G_OBJECT (invocation), "temp-filename");
Packit Service fc05fa
Packit Service fc05fa
	g_debug ("Notification: transfer accepted! accepting transfer");
Packit Service fc05fa
Packit Service fc05fa
	g_dbus_method_invocation_return_value (invocation,
Packit Service fc05fa
		g_variant_new ("(s)", file));
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
ask_user_transfer_rejected (NotifyNotification *notification,
Packit Service fc05fa
			    char *action,
Packit Service fc05fa
			    GDBusMethodInvocation *invocation)
Packit Service fc05fa
{
Packit Service fc05fa
	g_debug ("Notification: transfer rejected! rejecting transfer");
Packit Service fc05fa
	reject_transfer (invocation);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
ask_user_on_close (NotifyNotification *notification,
Packit Service fc05fa
		   GDBusMethodInvocation *invocation)
Packit Service fc05fa
{
Packit Service fc05fa
	g_debug ("Notification closed! rejecting transfer");
Packit Service fc05fa
	reject_transfer (invocation);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
ask_user (GDBusMethodInvocation *invocation,
Packit Service fc05fa
	  const char            *filename,
Packit Service fc05fa
	  const char            *name)
Packit Service fc05fa
{
Packit Service fc05fa
	NotifyNotification *notification;
Packit Service fc05fa
	char *summary, *body;
Packit Service fc05fa
Packit Service fc05fa
	summary = g_strdup_printf(_("Bluetooth file transfer from %s"), name);
Packit Service fc05fa
	body = g_filename_display_basename (filename);
Packit Service fc05fa
Packit Service fc05fa
	notification = notify_notification_new (summary, body, "bluetooth");
Packit Service fc05fa
Packit Service fc05fa
	notify_notification_set_urgency (notification, NOTIFY_URGENCY_CRITICAL);
Packit Service fc05fa
	notify_notification_set_timeout (notification, NOTIFY_EXPIRES_NEVER);
Packit Service fc05fa
	notify_notification_set_hint_string (notification, "desktop-entry",
Packit Service fc05fa
					     "gnome-bluetooth-panel");
Packit Service fc05fa
Packit Service fc05fa
	notify_notification_add_action (notification, "cancel", _("Decline"),
Packit Service fc05fa
					(NotifyActionCallback) ask_user_transfer_rejected,
Packit Service fc05fa
					invocation, NULL);
Packit Service fc05fa
	notify_notification_add_action (notification, "receive", _("Accept"),
Packit Service fc05fa
					(NotifyActionCallback) ask_user_transfer_accepted,
Packit Service fc05fa
					invocation, NULL);
Packit Service fc05fa
Packit Service fc05fa
	/* We want to reject the transfer if the user closes the notification
Packit Service fc05fa
	 * without accepting or rejecting it, so we connect to it. However
Packit Service fc05fa
	 * if the user clicks on one of the actions, the callback for the
Packit Service fc05fa
	 * action will be invoked, and then the notification will be closed
Packit Service fc05fa
	 * and the callback for :closed will be called. So we disconnect
Packit Service fc05fa
	 * from :closed if the invocation object goes away (which will happen
Packit Service fc05fa
	 * after the handler of either action accepts or rejects the transfer).
Packit Service fc05fa
	 */
Packit Service fc05fa
	g_signal_connect_object (G_OBJECT (notification), "closed",
Packit Service fc05fa
		G_CALLBACK (ask_user_on_close), invocation, 0);
Packit Service fc05fa
Packit Service fc05fa
	if (!notify_notification_show (notification, NULL))
Packit Service fc05fa
		g_warning ("failed to send notification\n");
Packit Service fc05fa
Packit Service fc05fa
	g_free (summary);
Packit Service fc05fa
	g_free (body);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static gboolean
Packit Service fc05fa
get_paired_for_address (const char  *adapter,
Packit Service fc05fa
			const char  *device,
Packit Service fc05fa
			char       **name)
Packit Service fc05fa
{
Packit Service fc05fa
	GtkTreeModel *model;
Packit Service fc05fa
	GtkTreeIter parent;
Packit Service fc05fa
	gboolean next;
Packit Service fc05fa
	gboolean ret = FALSE;
Packit Service fc05fa
	char *addr;
Packit Service fc05fa
Packit Service fc05fa
	model = bluetooth_client_get_model (client);
Packit Service fc05fa
Packit Service fc05fa
	for (next = gtk_tree_model_get_iter_first (model, &parent);
Packit Service fc05fa
	     next;
Packit Service fc05fa
	     next = gtk_tree_model_iter_next (model, &parent)) {
Packit Service fc05fa
		gtk_tree_model_get (model, &parent,
Packit Service fc05fa
			BLUETOOTH_COLUMN_ADDRESS, &addr,
Packit Service fc05fa
			-1);
Packit Service fc05fa
Packit Service fc05fa
		if (g_strcmp0 (addr, adapter) == 0) {
Packit Service fc05fa
			GtkTreeIter child;
Packit Service fc05fa
			char *dev_addr;
Packit Service fc05fa
Packit Service fc05fa
			for (next = gtk_tree_model_iter_children (model, &child, &parent);
Packit Service fc05fa
			     next;
Packit Service fc05fa
			     next = gtk_tree_model_iter_next (model, &child)) {
Packit Service fc05fa
				gboolean paired;
Packit Service fc05fa
				char *alias;
Packit Service fc05fa
Packit Service fc05fa
				gtk_tree_model_get (model, &child,
Packit Service fc05fa
					BLUETOOTH_COLUMN_ADDRESS, &dev_addr,
Packit Service fc05fa
					BLUETOOTH_COLUMN_PAIRED, &paired,
Packit Service fc05fa
					BLUETOOTH_COLUMN_ALIAS, &alias,
Packit Service fc05fa
					-1);
Packit Service fc05fa
				if (g_strcmp0 (dev_addr, device) == 0) {
Packit Service fc05fa
					ret = paired;
Packit Service fc05fa
					*name = alias;
Packit Service fc05fa
					next = FALSE;
Packit Service fc05fa
				} else {
Packit Service fc05fa
					g_free (alias);
Packit Service fc05fa
				}
Packit Service fc05fa
				g_free (dev_addr);
Packit Service fc05fa
			}
Packit Service fc05fa
		}
Packit Service fc05fa
		g_free (addr);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	g_object_unref (model);
Packit Service fc05fa
	return ret;
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
on_check_bonded_or_ask_session_acquired (GObject *object,
Packit Service fc05fa
					 GAsyncResult *res,
Packit Service fc05fa
					 gpointer user_data)
Packit Service fc05fa
{
Packit Service fc05fa
	GDBusMethodInvocation *invocation = user_data;
Packit Service fc05fa
	GDBusProxy *session;
Packit Service fc05fa
	GError *error = NULL;
Packit Service fc05fa
	GVariant *v;
Packit Service fc05fa
	char *device, *adapter, *name;
Packit Service fc05fa
	gboolean paired;
Packit Service fc05fa
Packit Service fc05fa
	session = g_dbus_proxy_new_for_bus_finish (res, &error);
Packit Service fc05fa
Packit Service fc05fa
	if (!session) {
Packit Service fc05fa
		g_debug ("Failed to create a proxy for the session: %s", error->message);
Packit Service fc05fa
		g_clear_error (&error);
Packit Service fc05fa
		goto out;
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	device = NULL;
Packit Service fc05fa
	adapter = NULL;
Packit Service fc05fa
Packit Service fc05fa
	/* obexd puts the remote device in Destination and our adapter
Packit Service fc05fa
	 * in Source */
Packit Service fc05fa
	v = g_dbus_proxy_get_cached_property (session, "Destination");
Packit Service fc05fa
	if (v) {
Packit Service fc05fa
		device = g_variant_dup_string (v, NULL);
Packit Service fc05fa
		g_variant_unref (v);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	v = g_dbus_proxy_get_cached_property (session, "Source");
Packit Service fc05fa
	if (v) {
Packit Service fc05fa
		adapter = g_variant_dup_string (v, NULL);
Packit Service fc05fa
		g_variant_unref (v);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	g_object_unref (session);
Packit Service fc05fa
Packit Service fc05fa
	if (!device || !adapter) {
Packit Service fc05fa
		g_debug ("Could not get remote device for the transfer");
Packit Service fc05fa
		g_free (device);
Packit Service fc05fa
		g_free (adapter);
Packit Service fc05fa
		goto out;
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	name = NULL;
Packit Service fc05fa
	paired = get_paired_for_address (adapter, device, &name);
Packit Service fc05fa
	g_free (device);
Packit Service fc05fa
	g_free (adapter);
Packit Service fc05fa
Packit Service fc05fa
	if (paired) {
Packit Service fc05fa
		g_debug ("Remote device '%s' is paired, auto-accepting the transfer", name);
Packit Service fc05fa
		g_dbus_method_invocation_return_value (invocation,
Packit Service fc05fa
						       g_variant_new ("(s)", g_object_get_data (G_OBJECT (invocation), "temp-filename")));
Packit Service fc05fa
		g_free (name);
Packit Service fc05fa
		return;
Packit Service fc05fa
	} else {
Packit Service fc05fa
		ask_user (invocation,
Packit Service fc05fa
			  g_object_get_data (G_OBJECT (invocation), "filename"),
Packit Service fc05fa
			  name ? name : device);
Packit Service fc05fa
		g_free (name);
Packit Service fc05fa
		return;
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
out:
Packit Service fc05fa
	g_debug ("Rejecting transfer");
Packit Service fc05fa
	reject_transfer (invocation);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
check_if_bonded_or_ask (GDBusProxy *transfer,
Packit Service fc05fa
			GDBusMethodInvocation *invocation)
Packit Service fc05fa
{
Packit Service fc05fa
	GVariant *v;
Packit Service fc05fa
	const gchar *session = NULL;
Packit Service fc05fa
Packit Service fc05fa
	v = g_dbus_proxy_get_cached_property (transfer, "Session");
Packit Service fc05fa
Packit Service fc05fa
	if (v) {
Packit Service fc05fa
		session = g_variant_get_string (v, NULL);
Packit Service fc05fa
		g_dbus_proxy_new_for_bus (G_BUS_TYPE_SESSION,
Packit Service fc05fa
					  G_DBUS_PROXY_FLAGS_NONE,
Packit Service fc05fa
					  NULL,
Packit Service fc05fa
					  MANAGER_SERVICE,
Packit Service fc05fa
					  session,
Packit Service fc05fa
					  SESSION_IFACE,
Packit Service fc05fa
					  cancellable,
Packit Service fc05fa
					  on_check_bonded_or_ask_session_acquired,
Packit Service fc05fa
					  invocation);
Packit Service fc05fa
		g_variant_unref (v);
Packit Service fc05fa
	} else {
Packit Service fc05fa
		g_debug ("Could not get session path for the transfer, "
Packit Service fc05fa
			 "rejecting the transfer");
Packit Service fc05fa
		reject_transfer (invocation);
Packit Service fc05fa
	}
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
obex_agent_release (GError **error)
Packit Service fc05fa
{
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
obex_agent_cancel (GError **error)
Packit Service fc05fa
{
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
/* From the old embed/mozilla/MozDownload.cpp */
Packit Service fc05fa
static const char*
Packit Service fc05fa
file_is_compressed (const char *filename)
Packit Service fc05fa
{
Packit Service fc05fa
  int i;
Packit Service fc05fa
  static const char * const compression[] = {".gz", ".bz2", ".Z", ".lz", ".xz", NULL};
Packit Service fc05fa
Packit Service fc05fa
  for (i = 0; compression[i] != NULL; i++) {
Packit Service fc05fa
    if (g_str_has_suffix (filename, compression[i]))
Packit Service fc05fa
      return compression[i];
Packit Service fc05fa
  }
Packit Service fc05fa
Packit Service fc05fa
  return NULL;
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static const char*
Packit Service fc05fa
parse_extension (const char *filename)
Packit Service fc05fa
{
Packit Service fc05fa
  const char *compression;
Packit Service fc05fa
  const char *last_separator;
Packit Service fc05fa
Packit Service fc05fa
  compression = file_is_compressed (filename);
Packit Service fc05fa
Packit Service fc05fa
  /* if the file is compressed we might have a double extension */
Packit Service fc05fa
  if (compression != NULL) {
Packit Service fc05fa
    int i;
Packit Service fc05fa
    static const char * const extensions[] = {"tar", "ps", "xcf", "dvi", "txt", "text", NULL};
Packit Service fc05fa
Packit Service fc05fa
    for (i = 0; extensions[i] != NULL; i++) {
Packit Service fc05fa
      char *suffix;
Packit Service fc05fa
      suffix = g_strdup_printf (".%s%s", extensions[i], compression);
Packit Service fc05fa
Packit Service fc05fa
      if (g_str_has_suffix (filename, suffix)) {
Packit Service fc05fa
        char *p;
Packit Service fc05fa
Packit Service fc05fa
        p = g_strrstr (filename, suffix);
Packit Service fc05fa
        g_free (suffix);
Packit Service fc05fa
Packit Service fc05fa
        return p;
Packit Service fc05fa
      }
Packit Service fc05fa
Packit Service fc05fa
      g_free (suffix);
Packit Service fc05fa
    }
Packit Service fc05fa
  }
Packit Service fc05fa
Packit Service fc05fa
  /* no compression, just look for the last dot in the filename */
Packit Service fc05fa
  last_separator = strrchr (filename, G_DIR_SEPARATOR);
Packit Service fc05fa
  return strrchr ((last_separator) ? last_separator : filename, '.');
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
char *
Packit Service fc05fa
lookup_download_dir (void)
Packit Service fc05fa
{
Packit Service fc05fa
	const char *special_dir;
Packit Service fc05fa
	char *dir;
Packit Service fc05fa
Packit Service fc05fa
	special_dir = g_get_user_special_dir (G_USER_DIRECTORY_DOWNLOAD);
Packit Service fc05fa
	if (special_dir != NULL) {
Packit Service fc05fa
		g_mkdir_with_parents (special_dir, 0755);
Packit Service fc05fa
		return g_strdup (special_dir);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	dir = g_build_filename (g_get_home_dir (), "Downloads", NULL);
Packit Service fc05fa
	g_mkdir_with_parents (dir, 0755);
Packit Service fc05fa
	return dir;
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static char *
Packit Service fc05fa
move_temp_filename (GObject *object)
Packit Service fc05fa
{
Packit Service fc05fa
	const char *orig_filename;
Packit Service fc05fa
	char *dest_filename, *dest_dir;
Packit Service fc05fa
	GFile *src, *dest;
Packit Service fc05fa
	GError *error = NULL;
Packit Service fc05fa
	gboolean res;
Packit Service fc05fa
Packit Service fc05fa
	orig_filename = g_object_get_data (object, "temp-filename");
Packit Service fc05fa
	src = g_file_new_for_path (orig_filename);
Packit Service fc05fa
Packit Service fc05fa
	dest_dir = lookup_download_dir ();
Packit Service fc05fa
	dest_filename = g_build_filename (dest_dir, g_object_get_data (object, "filename"), NULL);
Packit Service fc05fa
	g_free (dest_dir);
Packit Service fc05fa
	dest = g_file_new_for_path (dest_filename);
Packit Service fc05fa
Packit Service fc05fa
	res = g_file_move (src, dest,
Packit Service fc05fa
			   G_FILE_COPY_NONE, NULL,
Packit Service fc05fa
			   NULL, NULL, &error);
Packit Service fc05fa
Packit Service fc05fa
	/* This is sync, but the files will be on the same partition already
Packit Service fc05fa
	 * (~/.cache/obexd to ~/Downloads) */
Packit Service fc05fa
	if (!res && g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS)) {
Packit Service fc05fa
		guint i = 1;
Packit Service fc05fa
		const char *dot_pos;
Packit Service fc05fa
		gssize position;
Packit Service fc05fa
		char *serial = NULL;
Packit Service fc05fa
		GString *tmp_filename;
Packit Service fc05fa
Packit Service fc05fa
		dot_pos = parse_extension (dest_filename);
Packit Service fc05fa
		if (dot_pos)
Packit Service fc05fa
			position = dot_pos - dest_filename;
Packit Service fc05fa
		else
Packit Service fc05fa
			position = strlen (dest_filename);
Packit Service fc05fa
Packit Service fc05fa
		tmp_filename = g_string_new (NULL);
Packit Service fc05fa
		g_string_assign (tmp_filename, dest_filename);
Packit Service fc05fa
Packit Service fc05fa
		while (!res && g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS)) {
Packit Service fc05fa
			g_debug ("Couldn't move file to %s", tmp_filename->str);
Packit Service fc05fa
Packit Service fc05fa
			g_clear_error (&error);
Packit Service fc05fa
			g_object_unref (dest);
Packit Service fc05fa
Packit Service fc05fa
			serial = g_strdup_printf ("(%d)", i++);
Packit Service fc05fa
Packit Service fc05fa
			g_string_assign (tmp_filename, dest_filename);
Packit Service fc05fa
			g_string_insert (tmp_filename, position, serial);
Packit Service fc05fa
Packit Service fc05fa
			g_free (serial);
Packit Service fc05fa
Packit Service fc05fa
			dest = g_file_new_for_path (tmp_filename->str);
Packit Service fc05fa
			res = g_file_move (src, dest,
Packit Service fc05fa
					   G_FILE_COPY_NONE, NULL,
Packit Service fc05fa
					   NULL, NULL, &error);
Packit Service fc05fa
		}
Packit Service fc05fa
Packit Service fc05fa
		g_free (dest_filename);
Packit Service fc05fa
		dest_filename = g_strdup (tmp_filename->str);
Packit Service fc05fa
		g_string_free (tmp_filename, TRUE);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	if (!res) {
Packit Service fc05fa
		g_warning ("Failed to move %s to %s: '%s'",
Packit Service fc05fa
			   orig_filename, dest_filename, error->message);
Packit Service fc05fa
		g_error_free (error);
Packit Service fc05fa
	} else {
Packit Service fc05fa
		g_debug ("Moved %s (orig name %s) to %s",
Packit Service fc05fa
			 orig_filename, (char *) g_object_get_data (object, "filename"), dest_filename);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	g_object_unref (src);
Packit Service fc05fa
	g_object_unref (dest);
Packit Service fc05fa
Packit Service fc05fa
	return dest_filename;
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
transfer_property_changed (GDBusProxy *transfer,
Packit Service fc05fa
			   GVariant   *changed_properties,
Packit Service fc05fa
			   GStrv       invalidated_properties,
Packit Service fc05fa
			   gpointer    user_data)
Packit Service fc05fa
{
Packit Service fc05fa
	GVariantIter iter;
Packit Service fc05fa
	const gchar *key;
Packit Service fc05fa
	GVariant *value;
Packit Service fc05fa
	const char *filename;
Packit Service fc05fa
Packit Service fc05fa
	g_debug ("Calling transfer_property_changed()");
Packit Service fc05fa
Packit Service fc05fa
	filename = g_object_get_data (G_OBJECT (transfer), "filename");
Packit Service fc05fa
Packit Service fc05fa
	g_variant_iter_init (&iter, changed_properties);
Packit Service fc05fa
	while (g_variant_iter_next (&iter, "{&sv}", &key, &value)) {
Packit Service fc05fa
		char *str = g_variant_print (value, TRUE);
Packit Service fc05fa
Packit Service fc05fa
		if (g_str_equal (key, "Status")) {
Packit Service fc05fa
			const gchar *status;
Packit Service fc05fa
Packit Service fc05fa
			status = g_variant_get_string (value, NULL);
Packit Service fc05fa
Packit Service fc05fa
			g_debug ("Got status %s = %s for filename %s", status, str, filename);
Packit Service fc05fa
Packit Service fc05fa
			if (g_str_equal (status, "complete")) {
Packit Service fc05fa
				char *path;
Packit Service fc05fa
Packit Service fc05fa
				path = move_temp_filename (G_OBJECT (transfer));
Packit Service fc05fa
				g_debug ("transfer completed, showing a notification");
Packit Service fc05fa
				show_notification (path);
Packit Service fc05fa
				g_free (path);
Packit Service fc05fa
			}
Packit Service fc05fa
Packit Service fc05fa
			/* Done with this transfer */
Packit Service fc05fa
			if (g_str_equal (status, "complete") ||
Packit Service fc05fa
			    g_str_equal (status, "error")) {
Packit Service fc05fa
				g_object_unref (transfer);
Packit Service fc05fa
			}
Packit Service fc05fa
		} else {
Packit Service fc05fa
			g_debug ("Unhandled property changed %s = %s for filename %s", key, str, filename);
Packit Service fc05fa
		}
Packit Service fc05fa
		g_free (str);
Packit Service fc05fa
		g_variant_unref (value);
Packit Service fc05fa
	}
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
obex_agent_authorize_push (GObject *source_object,
Packit Service fc05fa
			   GAsyncResult *res,
Packit Service fc05fa
			   gpointer user_data)
Packit Service fc05fa
{
Packit Service fc05fa
	GDBusProxy *transfer;
Packit Service fc05fa
	GError *error = NULL;
Packit Service fc05fa
	GDBusMethodInvocation *invocation;
Packit Service fc05fa
	GVariant *variant;
Packit Service fc05fa
	const gchar *filename;
Packit Service fc05fa
	char *template;
Packit Service fc05fa
	int fd;
Packit Service fc05fa
Packit Service fc05fa
	transfer = g_dbus_proxy_new_for_bus_finish (res, &error);
Packit Service fc05fa
	if (!transfer) {
Packit Service fc05fa
		g_debug ("obex_agent_authorize_push() failed: %s", error->message);
Packit Service fc05fa
		g_error_free (error);
Packit Service fc05fa
		return;
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	invocation = user_data;
Packit Service fc05fa
	variant = g_dbus_proxy_get_cached_property (transfer, "Name");
Packit Service fc05fa
	filename = g_variant_get_string (variant, NULL);
Packit Service fc05fa
Packit Service fc05fa
	g_debug ("AuthorizePush received");
Packit Service fc05fa
Packit Service fc05fa
	template = g_build_filename (g_get_user_cache_dir (), "obexd", "XXXXXX", NULL);
Packit Service fc05fa
	fd = g_mkstemp (template);
Packit Service fc05fa
	close (fd);
Packit Service fc05fa
Packit Service fc05fa
	g_object_set_data_full (G_OBJECT (transfer), "filename", g_strdup (filename), g_free);
Packit Service fc05fa
	g_object_set_data_full (G_OBJECT (transfer), "temp-filename", g_strdup (template), g_free);
Packit Service fc05fa
Packit Service fc05fa
	g_object_set_data_full (G_OBJECT (invocation), "filename", g_strdup (filename), g_free);
Packit Service fc05fa
	g_object_set_data_full (G_OBJECT (invocation), "temp-filename", g_strdup (template), g_free);
Packit Service fc05fa
Packit Service fc05fa
	g_signal_connect (transfer, "g-properties-changed",
Packit Service fc05fa
			  G_CALLBACK (transfer_property_changed), NULL);
Packit Service fc05fa
Packit Service fc05fa
	/* check_if_bonded_or_ask() will accept or reject the transfer */
Packit Service fc05fa
	check_if_bonded_or_ask (transfer, invocation);
Packit Service fc05fa
	g_variant_unref (variant);
Packit Service fc05fa
	g_free (template);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
handle_method_call (GDBusConnection       *connection,
Packit Service fc05fa
		    const gchar           *sender,
Packit Service fc05fa
		    const gchar           *object_path,
Packit Service fc05fa
		    const gchar           *interface_name,
Packit Service fc05fa
		    const gchar           *method_name,
Packit Service fc05fa
		    GVariant              *parameters,
Packit Service fc05fa
		    GDBusMethodInvocation *invocation,
Packit Service fc05fa
		    gpointer               user_data)
Packit Service fc05fa
{
Packit Service fc05fa
	if (g_str_equal (method_name, "Cancel")) {
Packit Service fc05fa
		obex_agent_cancel (NULL);
Packit Service fc05fa
		g_dbus_method_invocation_return_value (invocation, NULL);
Packit Service fc05fa
	} else if (g_str_equal (method_name, "Release")) {
Packit Service fc05fa
		obex_agent_release (NULL);
Packit Service fc05fa
		g_dbus_method_invocation_return_value (invocation, NULL);
Packit Service fc05fa
	} else if (g_str_equal (method_name, "AuthorizePush")) {
Packit Service fc05fa
		const gchar *transfer;
Packit Service fc05fa
Packit Service fc05fa
		g_variant_get (parameters, "(&o)", &transfer);
Packit Service fc05fa
Packit Service fc05fa
		g_dbus_proxy_new_for_bus (G_BUS_TYPE_SESSION,
Packit Service fc05fa
					  G_DBUS_PROXY_FLAGS_NONE,
Packit Service fc05fa
					  NULL,
Packit Service fc05fa
					  MANAGER_SERVICE,
Packit Service fc05fa
					  transfer,
Packit Service fc05fa
					  TRANSFER_IFACE,
Packit Service fc05fa
					  cancellable,
Packit Service fc05fa
					  obex_agent_authorize_push,
Packit Service fc05fa
					  invocation);
Packit Service fc05fa
	} else {
Packit Service fc05fa
		g_warning ("Unknown method name or unknown parameters: %s",
Packit Service fc05fa
			   method_name);
Packit Service fc05fa
	}
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static const GDBusInterfaceVTable interface_vtable =
Packit Service fc05fa
{
Packit Service fc05fa
  handle_method_call,
Packit Service fc05fa
  NULL,
Packit Service fc05fa
  NULL
Packit Service fc05fa
};
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
obexd_appeared_cb (GDBusConnection *connection,
Packit Service fc05fa
		   const gchar *name,
Packit Service fc05fa
		   const gchar *name_owner,
Packit Service fc05fa
		   gpointer user_data)
Packit Service fc05fa
{
Packit Service fc05fa
	ObexAgent *self = user_data;
Packit Service fc05fa
Packit Service fc05fa
	g_debug ("obexd appeared, registering agent");
Packit Service fc05fa
	g_dbus_connection_call (self->connection,
Packit Service fc05fa
				MANAGER_SERVICE,
Packit Service fc05fa
				MANAGER_PATH,
Packit Service fc05fa
				MANAGER_IFACE,
Packit Service fc05fa
				"RegisterAgent",
Packit Service fc05fa
				g_variant_new ("(o)", AGENT_PATH),
Packit Service fc05fa
				NULL,
Packit Service fc05fa
				G_DBUS_CALL_FLAGS_NONE,
Packit Service fc05fa
				-1,
Packit Service fc05fa
				cancellable,
Packit Service fc05fa
				NULL,
Packit Service fc05fa
				NULL);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
on_bus_acquired (GDBusConnection *connection,
Packit Service fc05fa
		 const gchar     *name,
Packit Service fc05fa
		 gpointer         user_data)
Packit Service fc05fa
{
Packit Service fc05fa
	ObexAgent *self = user_data;
Packit Service fc05fa
Packit Service fc05fa
	/* parse introspection data */
Packit Service fc05fa
	introspection_data = g_dbus_node_info_new_for_xml (introspection_xml,
Packit Service fc05fa
							   NULL);
Packit Service fc05fa
Packit Service fc05fa
	self->connection = connection;
Packit Service fc05fa
	self->object_reg_id = g_dbus_connection_register_object (connection,
Packit Service fc05fa
								 AGENT_PATH,
Packit Service fc05fa
								 introspection_data->interfaces[0],
Packit Service fc05fa
								 &interface_vtable,
Packit Service fc05fa
								 NULL,  /* user_data */
Packit Service fc05fa
								 NULL,  /* user_data_free_func */
Packit Service fc05fa
								 NULL); /* GError** */
Packit Service fc05fa
Packit Service fc05fa
	g_dbus_node_info_unref (introspection_data);
Packit Service fc05fa
Packit Service fc05fa
	g_assert (self->object_reg_id > 0);
Packit Service fc05fa
Packit Service fc05fa
	self->obexd_watch_id = g_bus_watch_name_on_connection (self->connection,
Packit Service fc05fa
							       MANAGER_SERVICE,
Packit Service fc05fa
							       G_BUS_NAME_WATCHER_FLAGS_AUTO_START,
Packit Service fc05fa
							       obexd_appeared_cb,
Packit Service fc05fa
							       NULL,
Packit Service fc05fa
							       self,
Packit Service fc05fa
							       NULL);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
obex_agent_init (ObexAgent *self)
Packit Service fc05fa
{
Packit Service fc05fa
	self->owner_id = g_bus_own_name (G_BUS_TYPE_SESSION,
Packit Service fc05fa
					 AGENT_IFACE,
Packit Service fc05fa
					 G_BUS_NAME_OWNER_FLAGS_ALLOW_REPLACEMENT,
Packit Service fc05fa
					 on_bus_acquired,
Packit Service fc05fa
					 NULL,
Packit Service fc05fa
					 NULL,
Packit Service fc05fa
					 self,
Packit Service fc05fa
					 NULL);
Packit Service fc05fa
Packit Service fc05fa
	client = bluetooth_client_new ();
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
obex_agent_dispose (GObject *obj)
Packit Service fc05fa
{
Packit Service fc05fa
	ObexAgent *self = OBEX_AGENT (obj);
Packit Service fc05fa
Packit Service fc05fa
	if (self->object_reg_id != 0) {
Packit Service fc05fa
		g_dbus_connection_unregister_object (self->connection, self->object_reg_id);
Packit Service fc05fa
		self->object_reg_id = 0;
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	g_bus_unown_name (self->owner_id);
Packit Service fc05fa
	self->owner_id = 0;
Packit Service fc05fa
Packit Service fc05fa
	if (self->obexd_watch_id != 0) {
Packit Service fc05fa
		g_bus_unwatch_name (self->obexd_watch_id);
Packit Service fc05fa
		self->obexd_watch_id = 0;
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	g_clear_object (&client);
Packit Service fc05fa
Packit Service fc05fa
	G_OBJECT_CLASS (obex_agent_parent_class)->dispose (obj);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static void
Packit Service fc05fa
obex_agent_class_init (ObexAgentClass *klass)
Packit Service fc05fa
{
Packit Service fc05fa
	GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit Service fc05fa
Packit Service fc05fa
	gobject_class->dispose = obex_agent_dispose;
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
static ObexAgent *
Packit Service fc05fa
obex_agent_new (void)
Packit Service fc05fa
{
Packit Service fc05fa
	return (ObexAgent *) g_object_new (OBEX_AGENT_TYPE, NULL);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
void
Packit Service fc05fa
obex_agent_down (void)
Packit Service fc05fa
{
Packit Service fc05fa
	if (agent != NULL && agent->connection != NULL) {
Packit Service fc05fa
		g_dbus_connection_call (agent->connection,
Packit Service fc05fa
					MANAGER_SERVICE,
Packit Service fc05fa
					MANAGER_PATH,
Packit Service fc05fa
					MANAGER_IFACE,
Packit Service fc05fa
					"UnregisterAgent",
Packit Service fc05fa
					g_variant_new ("(o)", AGENT_PATH),
Packit Service fc05fa
					NULL,
Packit Service fc05fa
					G_DBUS_CALL_FLAGS_NONE,
Packit Service fc05fa
					-1,
Packit Service fc05fa
					NULL,
Packit Service fc05fa
					NULL,
Packit Service fc05fa
					NULL);
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	if (cancellable != NULL) {
Packit Service fc05fa
		g_cancellable_cancel (cancellable);
Packit Service fc05fa
		g_clear_object (&cancellable);
Packit Service fc05fa
	}
Packit Service fc05fa
	g_clear_object (&agent);
Packit Service fc05fa
	g_clear_object (&client);
Packit Service fc05fa
}
Packit Service fc05fa
Packit Service fc05fa
void
Packit Service fc05fa
obex_agent_up (void)
Packit Service fc05fa
{
Packit Service fc05fa
	if (agent == NULL)
Packit Service fc05fa
		agent = obex_agent_new ();
Packit Service fc05fa
Packit Service fc05fa
	if (!notify_init ("gnome-bluetooth")) {
Packit Service fc05fa
		g_warning("Unable to initialize the notification system");
Packit Service fc05fa
	}
Packit Service fc05fa
Packit Service fc05fa
	g_assert (cancellable == NULL);
Packit Service fc05fa
	cancellable = g_cancellable_new ();
Packit Service fc05fa
}