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;
|
|||
|
}
|