mirror of https://gitee.com/openkylin/gvfs.git
487 lines
13 KiB
C
487 lines
13 KiB
C
/* GIO - GLib Input, Output and Streaming Library
|
||
*
|
||
* Copyright (C) 2006-2007 Red Hat, Inc.
|
||
*
|
||
* This library is free software; you can redistribute it and/or
|
||
* modify it under the terms of the GNU Lesser General Public
|
||
* License as published by the Free Software Foundation; either
|
||
* version 2 of the License, or (at your option) any later version.
|
||
*
|
||
* This library 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
|
||
* Lesser General Public License for more details.
|
||
*
|
||
* You should have received a copy of the GNU Lesser General
|
||
* Public License along with this library; if not, write to the
|
||
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
||
* Boston, MA 02110-1301, USA.
|
||
*
|
||
* Author: Alexander Larsson <alexl@redhat.com>
|
||
*/
|
||
|
||
#include <config.h>
|
||
|
||
#include <glib.h>
|
||
#include <glib/gi18n.h>
|
||
#include <glib/gstdio.h>
|
||
#include <locale.h>
|
||
#include <stdlib.h>
|
||
#include "metatree.h"
|
||
#include "gvfsdaemonprotocol.h"
|
||
#include "metadata-dbus.h"
|
||
|
||
#ifdef HAVE_GUDEV
|
||
#include <gudev/gudev.h>
|
||
#endif
|
||
|
||
#if MAJOR_IN_MKDEV
|
||
#include <sys/mkdev.h>
|
||
#elif MAJOR_IN_SYSMACROS
|
||
#include <sys/sysmacros.h>
|
||
#endif
|
||
|
||
#define WRITEOUT_TIMEOUT_SECS 60
|
||
#define WRITEOUT_TIMEOUT_SECS_NFS 15
|
||
|
||
typedef struct {
|
||
char *filename;
|
||
MetaTree *tree;
|
||
guint writeout_timeout;
|
||
} TreeInfo;
|
||
|
||
static GHashTable *tree_infos = NULL;
|
||
static GVfsMetadata *skeleton = NULL;
|
||
#ifdef HAVE_GUDEV
|
||
static GUdevClient *gudev_client = NULL;
|
||
#endif
|
||
|
||
static void
|
||
tree_info_free (TreeInfo *info)
|
||
{
|
||
g_free (info->filename);
|
||
meta_tree_unref (info->tree);
|
||
if (info->writeout_timeout)
|
||
g_source_remove (info->writeout_timeout);
|
||
|
||
g_free (info);
|
||
}
|
||
|
||
static gboolean
|
||
writeout_timeout (gpointer data)
|
||
{
|
||
TreeInfo *info = data;
|
||
|
||
meta_tree_flush (info->tree);
|
||
info->writeout_timeout = 0;
|
||
|
||
return FALSE;
|
||
}
|
||
|
||
static void
|
||
tree_info_schedule_writeout (TreeInfo *info)
|
||
{
|
||
gboolean on_nfs;
|
||
|
||
if (info->writeout_timeout == 0)
|
||
{
|
||
on_nfs = meta_tree_is_on_nfs (info->tree);
|
||
info->writeout_timeout =
|
||
g_timeout_add_seconds (on_nfs ? WRITEOUT_TIMEOUT_SECS_NFS : WRITEOUT_TIMEOUT_SECS,
|
||
writeout_timeout, info);
|
||
}
|
||
}
|
||
|
||
static void
|
||
flush_single (const gchar *filename,
|
||
TreeInfo *info,
|
||
gpointer user_data)
|
||
{
|
||
if (info->writeout_timeout != 0)
|
||
{
|
||
g_source_remove (info->writeout_timeout);
|
||
writeout_timeout (info);
|
||
}
|
||
}
|
||
|
||
static void
|
||
flush_all ()
|
||
{
|
||
g_hash_table_foreach (tree_infos, (GHFunc) flush_single, NULL);
|
||
}
|
||
|
||
static TreeInfo *
|
||
tree_info_new (const char *filename)
|
||
{
|
||
TreeInfo *info;
|
||
MetaTree *tree;
|
||
|
||
tree = meta_tree_open (filename, TRUE);
|
||
if (tree == NULL)
|
||
return NULL;
|
||
|
||
info = g_new0 (TreeInfo, 1);
|
||
info->filename = g_strdup (filename);
|
||
info->tree = tree;
|
||
info->writeout_timeout = 0;
|
||
|
||
return info;
|
||
}
|
||
|
||
static TreeInfo *
|
||
tree_info_lookup (const char *filename)
|
||
{
|
||
TreeInfo *info;
|
||
|
||
info = g_hash_table_lookup (tree_infos, filename);
|
||
if (info)
|
||
return info;
|
||
|
||
info = tree_info_new (filename);
|
||
if (info)
|
||
g_hash_table_insert (tree_infos,
|
||
info->filename,
|
||
info);
|
||
return info;
|
||
}
|
||
|
||
static gboolean
|
||
handle_set (GVfsMetadata *object,
|
||
GDBusMethodInvocation *invocation,
|
||
const gchar *arg_treefile,
|
||
const gchar *arg_path,
|
||
GVariant *arg_data,
|
||
GVfsMetadata *daemon)
|
||
{
|
||
TreeInfo *info;
|
||
const gchar *str;
|
||
const gchar **strv;
|
||
const gchar *key;
|
||
GError *error;
|
||
GVariantIter iter;
|
||
GVariant *value;
|
||
|
||
info = tree_info_lookup (arg_treefile);
|
||
if (info == NULL)
|
||
{
|
||
g_dbus_method_invocation_return_error (invocation,
|
||
G_IO_ERROR,
|
||
G_IO_ERROR_NOT_FOUND,
|
||
_("Can’t find metadata file %s"),
|
||
arg_treefile);
|
||
return TRUE;
|
||
}
|
||
|
||
error = NULL;
|
||
|
||
g_variant_iter_init (&iter, arg_data);
|
||
while (g_variant_iter_next (&iter, "{&sv}", &key, &value))
|
||
{
|
||
if (g_variant_is_of_type (value, G_VARIANT_TYPE_STRING_ARRAY))
|
||
{
|
||
/* stringv */
|
||
strv = g_variant_get_strv (value, NULL);
|
||
if (!meta_tree_set_stringv (info->tree, arg_path, key, (gchar **) strv))
|
||
{
|
||
g_set_error_literal (&error, G_IO_ERROR,
|
||
G_IO_ERROR_FAILED,
|
||
_("Unable to set metadata key"));
|
||
}
|
||
g_free (strv);
|
||
}
|
||
else if (g_variant_is_of_type (value, G_VARIANT_TYPE_STRING))
|
||
{
|
||
/* string */
|
||
str = g_variant_get_string (value, NULL);
|
||
if (!meta_tree_set_string (info->tree, arg_path, key, str))
|
||
{
|
||
g_set_error_literal (&error, G_IO_ERROR,
|
||
G_IO_ERROR_FAILED,
|
||
_("Unable to set metadata key"));
|
||
}
|
||
}
|
||
else if (g_variant_is_of_type (value, G_VARIANT_TYPE_BYTE))
|
||
{
|
||
/* Unset */
|
||
if (!meta_tree_unset (info->tree, arg_path, key))
|
||
{
|
||
g_set_error_literal (&error, G_IO_ERROR,
|
||
G_IO_ERROR_FAILED,
|
||
_("Unable to unset metadata key"));
|
||
}
|
||
}
|
||
g_variant_unref (value);
|
||
}
|
||
|
||
tree_info_schedule_writeout (info);
|
||
|
||
if (error)
|
||
{
|
||
g_dbus_method_invocation_return_gerror (invocation, error);
|
||
g_error_free (error);
|
||
}
|
||
else
|
||
{
|
||
gvfs_metadata_complete_set (object, invocation);
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
static gboolean
|
||
handle_remove (GVfsMetadata *object,
|
||
GDBusMethodInvocation *invocation,
|
||
const gchar *arg_treefile,
|
||
const gchar *arg_path,
|
||
GVfsMetadata *daemon)
|
||
{
|
||
TreeInfo *info;
|
||
|
||
info = tree_info_lookup (arg_treefile);
|
||
if (info == NULL)
|
||
{
|
||
g_dbus_method_invocation_return_error (invocation,
|
||
G_IO_ERROR,
|
||
G_IO_ERROR_NOT_FOUND,
|
||
_("Can’t find metadata file %s"),
|
||
arg_treefile);
|
||
return TRUE;
|
||
}
|
||
|
||
if (!meta_tree_remove (info->tree, arg_path))
|
||
{
|
||
g_dbus_method_invocation_return_error_literal (invocation,
|
||
G_IO_ERROR,
|
||
G_IO_ERROR_FAILED,
|
||
_("Unable to remove metadata keys"));
|
||
return TRUE;
|
||
}
|
||
|
||
tree_info_schedule_writeout (info);
|
||
gvfs_metadata_complete_remove (object, invocation);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
static gboolean
|
||
handle_move (GVfsMetadata *object,
|
||
GDBusMethodInvocation *invocation,
|
||
const gchar *arg_treefile,
|
||
const gchar *arg_path,
|
||
const gchar *arg_dest_path,
|
||
GVfsMetadata *daemon)
|
||
{
|
||
TreeInfo *info;
|
||
|
||
info = tree_info_lookup (arg_treefile);
|
||
if (info == NULL)
|
||
{
|
||
g_dbus_method_invocation_return_error (invocation,
|
||
G_IO_ERROR,
|
||
G_IO_ERROR_NOT_FOUND,
|
||
_("Can’t find metadata file %s"),
|
||
arg_treefile);
|
||
return TRUE;
|
||
}
|
||
|
||
/* Overwrites any dest */
|
||
if (!meta_tree_copy (info->tree, arg_path, arg_dest_path))
|
||
{
|
||
g_dbus_method_invocation_return_error_literal (invocation,
|
||
G_IO_ERROR,
|
||
G_IO_ERROR_FAILED,
|
||
_("Unable to move metadata keys"));
|
||
return TRUE;
|
||
}
|
||
|
||
/* Remove source if copy succeeded (ignoring errors) */
|
||
meta_tree_remove (info->tree, arg_path);
|
||
|
||
tree_info_schedule_writeout (info);
|
||
gvfs_metadata_complete_move (object, invocation);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
static gboolean
|
||
handle_get_tree_from_device (GVfsMetadata *object,
|
||
GDBusMethodInvocation *invocation,
|
||
guint arg_major,
|
||
guint arg_minor)
|
||
{
|
||
char *res = NULL;
|
||
|
||
#ifdef HAVE_GUDEV
|
||
GUdevDeviceNumber devnum = makedev (arg_major, arg_minor);
|
||
GUdevDevice *device;
|
||
|
||
if (g_once_init_enter (&gudev_client))
|
||
g_once_init_leave (&gudev_client, g_udev_client_new (NULL));
|
||
|
||
device = g_udev_client_query_by_device_number (gudev_client, G_UDEV_DEVICE_TYPE_BLOCK, devnum);
|
||
if (device != NULL)
|
||
{
|
||
if (g_udev_device_has_property (device, "ID_FS_UUID_ENC"))
|
||
res = g_strconcat ("uuid-", g_udev_device_get_property (device, "ID_FS_UUID_ENC"), NULL);
|
||
else if (g_udev_device_has_property (device, "ID_FS_LABEL_ENC"))
|
||
res = g_strconcat ("label-", g_udev_device_get_property (device, "ID_FS_LABEL_ENC"), NULL);
|
||
|
||
g_clear_object (&device);
|
||
}
|
||
#endif
|
||
|
||
gvfs_metadata_complete_get_tree_from_device (object, invocation, res ? res : "");
|
||
g_free (res);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
static void
|
||
on_name_lost (GDBusConnection *connection,
|
||
const gchar *name,
|
||
gpointer user_data)
|
||
{
|
||
GMainLoop *loop = user_data;
|
||
|
||
/* means that someone has claimed our name (we allow replacement) */
|
||
flush_all ();
|
||
g_main_loop_quit (loop);
|
||
}
|
||
|
||
static void
|
||
on_connection_closed (GDBusConnection *connection,
|
||
gboolean remote_peer_vanished,
|
||
GError *error,
|
||
gpointer user_data)
|
||
{
|
||
GMainLoop *loop = user_data;
|
||
|
||
/* session bus died */
|
||
flush_all ();
|
||
g_main_loop_quit (loop);
|
||
}
|
||
|
||
int
|
||
main (int argc, char *argv[])
|
||
{
|
||
GMainLoop *loop;
|
||
GDBusConnection *conn;
|
||
gboolean replace;
|
||
gboolean show_version;
|
||
GError *error;
|
||
guint name_owner_id;
|
||
GBusNameOwnerFlags flags;
|
||
GOptionContext *context;
|
||
const GOptionEntry options[] = {
|
||
{ "replace", 'r', 0, G_OPTION_ARG_NONE, &replace, N_("Replace old daemon."), NULL },
|
||
{ "version", 0, 0, G_OPTION_ARG_NONE, &show_version, N_("Show program version."), NULL},
|
||
{ NULL }
|
||
};
|
||
|
||
setlocale (LC_ALL, "");
|
||
|
||
bindtextdomain (GETTEXT_PACKAGE, GVFS_LOCALEDIR);
|
||
bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
|
||
textdomain (GETTEXT_PACKAGE);
|
||
|
||
g_set_application_name (_("GVFS Metadata Daemon"));
|
||
context = g_option_context_new ("");
|
||
|
||
g_option_context_set_summary (context, _("Metadata daemon for GVFS"));
|
||
|
||
g_option_context_add_main_entries (context, options, GETTEXT_PACKAGE);
|
||
|
||
replace = FALSE;
|
||
show_version = FALSE;
|
||
name_owner_id = 0;
|
||
|
||
error = NULL;
|
||
if (!g_option_context_parse (context, &argc, &argv, &error))
|
||
{
|
||
/* Translators: the first %s is the application name, */
|
||
/* the second %s is the error message */
|
||
g_printerr (_("%s: %s"), g_get_application_name(), error->message);
|
||
g_printerr ("\n");
|
||
g_printerr (_("Try “%s --help” for more information."),
|
||
g_get_prgname ());
|
||
g_printerr ("\n");
|
||
g_error_free (error);
|
||
g_option_context_free (context);
|
||
return 1;
|
||
}
|
||
|
||
g_option_context_free (context);
|
||
|
||
if (show_version)
|
||
{
|
||
g_print(PACKAGE_STRING "\n");
|
||
return 0;
|
||
}
|
||
|
||
error = NULL;
|
||
conn = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
||
if (!conn)
|
||
{
|
||
g_printerr ("Failed to connect to the D-BUS daemon: %s (%s, %d)\n",
|
||
error->message, g_quark_to_string (error->domain), error->code);
|
||
g_error_free (error);
|
||
return 1;
|
||
}
|
||
|
||
tree_infos = g_hash_table_new_full (g_str_hash,
|
||
g_str_equal,
|
||
NULL,
|
||
(GDestroyNotify)tree_info_free);
|
||
|
||
loop = g_main_loop_new (NULL, FALSE);
|
||
g_dbus_connection_set_exit_on_close (conn, FALSE);
|
||
g_signal_connect (conn, "closed", G_CALLBACK (on_connection_closed), loop);
|
||
|
||
flags = G_BUS_NAME_OWNER_FLAGS_ALLOW_REPLACEMENT;
|
||
if (replace)
|
||
flags |= G_BUS_NAME_OWNER_FLAGS_REPLACE;
|
||
|
||
skeleton = gvfs_metadata_skeleton_new ();
|
||
|
||
g_signal_connect (skeleton, "handle-set", G_CALLBACK (handle_set), skeleton);
|
||
g_signal_connect (skeleton, "handle-remove", G_CALLBACK (handle_remove), skeleton);
|
||
g_signal_connect (skeleton, "handle-move", G_CALLBACK (handle_move), skeleton);
|
||
g_signal_connect (skeleton, "handle-get-tree-from-device", G_CALLBACK (handle_get_tree_from_device), skeleton);
|
||
|
||
error = NULL;
|
||
if (!g_dbus_interface_skeleton_export (G_DBUS_INTERFACE_SKELETON (skeleton), conn,
|
||
G_VFS_DBUS_METADATA_PATH, &error))
|
||
{
|
||
g_printerr ("Error exporting metadata daemon: %s (%s, %d)\n",
|
||
error->message, g_quark_to_string (error->domain), error->code);
|
||
g_error_free (error);
|
||
g_object_unref (conn);
|
||
g_main_loop_unref (loop);
|
||
return 1;
|
||
}
|
||
|
||
name_owner_id = g_bus_own_name_on_connection (conn,
|
||
G_VFS_DBUS_METADATA_NAME,
|
||
flags,
|
||
NULL,
|
||
on_name_lost,
|
||
loop,
|
||
NULL);
|
||
|
||
g_main_loop_run (loop);
|
||
|
||
if (skeleton)
|
||
g_dbus_interface_skeleton_unexport (G_DBUS_INTERFACE_SKELETON (skeleton));
|
||
if (name_owner_id != 0)
|
||
g_bus_unown_name (name_owner_id);
|
||
if (conn)
|
||
g_object_unref (conn);
|
||
if (loop != NULL)
|
||
g_main_loop_unref (loop);
|
||
#ifdef HAVE_GUDEV
|
||
g_clear_object (&gudev_client);
|
||
#endif
|
||
|
||
return 0;
|
||
}
|