[tracker/extraction-improvements: 2/9] libtracker-extract: Add tracker-extract-client API



commit 449c218074d126f6f5846906bbc00b7eb65c61c4
Author: Carlos Garnacho <carlosg gnome org>
Date:   Wed May 4 14:37:44 2011 +0200

    libtracker-extract: Add tracker-extract-client API
    
    This API is meant to replace the code in TrackerMinerFiles to
    deal with the communication with the extractor.

 src/libtracker-extract/Makefile.am              |    3 +
 src/libtracker-extract/tracker-extract-client.c |  429 +++++++++++++++++++++++
 src/libtracker-extract/tracker-extract-client.h |   51 +++
 src/libtracker-extract/tracker-extract.h        |    1 +
 src/miners/fs/tracker-miner-files.c             |  416 ++--------------------
 5 files changed, 513 insertions(+), 387 deletions(-)
---
diff --git a/src/libtracker-extract/Makefile.am b/src/libtracker-extract/Makefile.am
index 5335269..6007011 100644
--- a/src/libtracker-extract/Makefile.am
+++ b/src/libtracker-extract/Makefile.am
@@ -17,6 +17,8 @@ libtracker_extract_ TRACKER_API_VERSION@_la_SOURCES =  \
 	tracker-encoding.c                             \
 	tracker-exif.c                                 \
 	tracker-exif.h                                 \
+	tracker-extract-client.c                       \
+	tracker-extract-client.h                       \
 	tracker-guarantee.c                            \
 	tracker-guarantee.h                            \
 	tracker-iptc.c                                 \
@@ -33,6 +35,7 @@ libtracker_extractinclude_HEADERS =                    \
 	tracker-data.h                                 \
 	tracker-encoding.h                             \
 	tracker-exif.h                                 \
+	tracker-extract-client.h                       \
 	tracker-extract.h                              \
 	tracker-guarantee.h                            \
 	tracker-iptc.h                                 \
diff --git a/src/libtracker-extract/tracker-extract-client.c b/src/libtracker-extract/tracker-extract-client.c
new file mode 100644
index 0000000..ac229a2
--- /dev/null
+++ b/src/libtracker-extract/tracker-extract-client.c
@@ -0,0 +1,429 @@
+/*
+ * Copyright (C) 2010, Nokia <ivan frade nokia com>
+ *
+ * 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.1 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.
+ */
+
+#include "config.h"
+#include "tracker-extract-client.h"
+
+#include <string.h>
+#include <gio/gunixfdlist.h>
+#include <gio/gunixinputstream.h>
+
+/* Size of buffers used when sending data over a pipe, using DBus FD passing */
+#define DBUS_PIPE_BUFFER_SIZE      65536
+
+#define DBUS_SERVICE_EXTRACT       "org.freedesktop.Tracker1.Extract"
+#define DBUS_PATH_EXTRACT          "/org/freedesktop/Tracker1/Extract"
+#define DBUS_INTERFACE_EXTRACT     "org.freedesktop.Tracker1.Extract"
+
+static GDBusConnection *connection = NULL;
+
+struct TrackerExtractInfo {
+	gchar *preupdate;
+	gchar *update;
+	gchar *where;
+};
+
+typedef void (* SendAndSpliceCallback) (void     *buffer,
+                                        gssize    buffer_size,
+                                        GError   *error, /* Don't free */
+                                        gpointer  user_data);
+
+typedef struct {
+	GInputStream *unix_input_stream;
+	GInputStream *buffered_input_stream;
+	GOutputStream *output_stream;
+	SendAndSpliceCallback callback;
+	GCancellable *cancellable;
+	GSimpleAsyncResult *res;
+	gboolean splice_finished;
+	gboolean dbus_finished;
+	GError *error;
+} SendAndSpliceData;
+
+static TrackerExtractInfo *
+tracker_extract_info_new (const gchar *preupdate,
+                          const gchar *update,
+                          const gchar *where)
+{
+	TrackerExtractInfo *info;
+
+	info = g_slice_new0 (TrackerExtractInfo);
+	info->preupdate = g_strdup (preupdate);
+	info->update = g_strdup (update);
+	info->where = g_strdup (where);
+
+	return info;
+}
+
+static void
+tracker_extract_info_free (TrackerExtractInfo *info)
+{
+	g_free (info->preupdate);
+	g_free (info->update);
+	g_free (info->where);
+
+	g_slice_free (TrackerExtractInfo, info);
+}
+
+static SendAndSpliceData *
+send_and_splice_data_new (GInputStream          *unix_input_stream,
+                          GInputStream          *buffered_input_stream,
+                          GOutputStream         *output_stream,
+                          GCancellable          *cancellable,
+                          SendAndSpliceCallback  callback,
+                          GSimpleAsyncResult    *res)
+{
+	SendAndSpliceData *data;
+
+	data = g_slice_new0 (SendAndSpliceData);
+	data->unix_input_stream = unix_input_stream;
+	data->buffered_input_stream = buffered_input_stream;
+	data->output_stream = output_stream;
+
+	if (cancellable) {
+		data->cancellable = g_object_ref (cancellable);
+	}
+
+	data->callback = callback;
+	data->res = g_object_ref (res);
+
+	return data;
+}
+
+static void
+send_and_splice_data_free (SendAndSpliceData *data)
+{
+	g_object_unref (data->unix_input_stream);
+	g_object_unref (data->buffered_input_stream);
+	g_object_unref (data->output_stream);
+
+	if (data->cancellable) {
+		g_object_unref (data->cancellable);
+	}
+
+	if (data->error) {
+		g_error_free (data->error);
+	}
+
+	if (data->res) {
+		g_object_unref (data->res);
+	}
+
+	g_slice_free (SendAndSpliceData, data);
+}
+
+static void
+dbus_send_and_splice_async_finish (SendAndSpliceData *data)
+{
+	if (!data->error) {
+		(* data->callback) (g_memory_output_stream_get_data (G_MEMORY_OUTPUT_STREAM (data->output_stream)),
+		                    g_memory_output_stream_get_data_size (G_MEMORY_OUTPUT_STREAM (data->output_stream)),
+		                    NULL,
+		                    data->res);
+	} else {
+		(* data->callback) (NULL, -1, data->error, data->res);
+	}
+
+	send_and_splice_data_free (data);
+}
+
+static void
+send_and_splice_splice_callback (GObject      *source,
+                                 GAsyncResult *result,
+                                 gpointer      user_data)
+{
+	SendAndSpliceData *data = user_data;
+	GError *error = NULL;
+
+	g_output_stream_splice_finish (G_OUTPUT_STREAM (source), result, &error);
+
+	if (error) {
+		if (!data->error) {
+			data->error = error;
+		} else {
+			g_error_free (error);
+		}
+	}
+
+	data->splice_finished = TRUE;
+
+	if (data->dbus_finished) {
+		dbus_send_and_splice_async_finish (data);
+	}
+}
+
+static void
+send_and_splice_dbus_callback (GObject      *source,
+                               GAsyncResult *result,
+                               gpointer      user_data)
+{
+	SendAndSpliceData *data = user_data;
+	GDBusMessage *reply;
+	GError *error = NULL;
+
+	reply = g_dbus_connection_send_message_with_reply_finish (G_DBUS_CONNECTION (source),
+	                                                          result, &error);
+
+	if (reply) {
+		if (g_dbus_message_get_message_type (reply) == G_DBUS_MESSAGE_TYPE_ERROR) {
+			g_dbus_message_to_gerror (reply, &error);
+		}
+
+		g_object_unref (reply);
+	}
+
+	if (error) {
+		if (!data->error) {
+			data->error = error;
+		} else {
+			g_error_free (error);
+		}
+	}
+
+	data->dbus_finished = TRUE;
+
+	if (data->splice_finished) {
+		dbus_send_and_splice_async_finish (data);
+	}
+}
+
+static void
+dbus_send_and_splice_async (GDBusConnection       *connection,
+                            GDBusMessage          *message,
+                            int                    fd,
+                            GCancellable          *cancellable,
+                            SendAndSpliceCallback  callback,
+                            GSimpleAsyncResult    *res)
+{
+	SendAndSpliceData *data;
+	GInputStream *unix_input_stream;
+	GInputStream *buffered_input_stream;
+	GOutputStream *output_stream;
+
+	unix_input_stream = g_unix_input_stream_new (fd, TRUE);
+	buffered_input_stream = g_buffered_input_stream_new_sized (unix_input_stream,
+	                                                           DBUS_PIPE_BUFFER_SIZE);
+	output_stream = g_memory_output_stream_new (NULL, 0, g_realloc, g_free);
+
+	data = send_and_splice_data_new (unix_input_stream,
+	                                 buffered_input_stream,
+	                                 output_stream,
+	                                 cancellable,
+	                                 callback,
+	                                 res);
+
+	g_dbus_connection_send_message_with_reply (connection,
+	                                           message,
+	                                           G_DBUS_SEND_MESSAGE_FLAGS_NONE,
+	                                           -1,
+	                                           NULL,
+	                                           cancellable,
+	                                           send_and_splice_dbus_callback,
+	                                           data);
+
+	g_output_stream_splice_async (data->output_stream,
+	                              data->buffered_input_stream,
+	                              G_OUTPUT_STREAM_SPLICE_CLOSE_SOURCE |
+	                              G_OUTPUT_STREAM_SPLICE_CLOSE_TARGET,
+	                              0,
+	                              cancellable,
+	                              send_and_splice_splice_callback,
+	                              data);
+}
+
+static void
+get_metadata_fast_cb (void     *buffer,
+                      gssize    buffer_size,
+                      GError   *error,
+                      gpointer  user_data)
+{
+	GSimpleAsyncResult *res;
+
+	res = user_data;
+
+	if (G_UNLIKELY (error)) {
+		g_simple_async_result_set_from_error (res, error);
+	} else {
+		const gchar *preupdate, *sparql, *where, *end;
+		TrackerExtractInfo *info;
+		gsize len;
+
+		preupdate = sparql = where = NULL;
+		end = (gchar *) buffer + buffer_size;
+
+		if (buffer) {
+			preupdate = buffer;
+			len = strlen (preupdate);
+
+			if (preupdate + len < end) {
+				buffer_size -= len;
+				sparql = preupdate + len + 1;
+				len = strlen (sparql);
+
+				if (sparql + len < end) {
+					where = sparql + len + 1;
+				}
+			}
+		}
+
+		info = tracker_extract_info_new (preupdate, sparql, where);
+		g_simple_async_result_set_op_res_gpointer (res, info,
+		                                           (GDestroyNotify) tracker_extract_info_free);
+	}
+
+	g_simple_async_result_complete_in_idle (res);
+}
+
+static void
+get_metadata_fast_async (GDBusConnection    *connection,
+                         const gchar        *uri,
+                         const gchar        *mime_type,
+                         GCancellable       *cancellable,
+                         GSimpleAsyncResult *res)
+{
+	GDBusMessage *message;
+	GUnixFDList *fd_list;
+	int pipefd[2];
+
+	if (pipe (pipefd) < 0) {
+		g_critical ("Coudln't open pipe");
+		/* FIXME: Report async error */
+		return;
+	}
+
+	message = g_dbus_message_new_method_call (DBUS_SERVICE_EXTRACT,
+	                                          DBUS_PATH_EXTRACT,
+	                                          DBUS_INTERFACE_EXTRACT,
+	                                          "GetMetadataFast");
+
+	fd_list = g_unix_fd_list_new ();
+
+	g_dbus_message_set_body (message,
+	                         g_variant_new ("(ssh)",
+	                                        uri,
+	                                        mime_type,
+	                                        g_unix_fd_list_append (fd_list,
+	                                                               pipefd[1],
+	                                                               NULL)));
+	g_dbus_message_set_unix_fd_list (message, fd_list);
+
+	/* We need to close the fd as g_unix_fd_list_append duplicates the fd */
+
+	close (pipefd[1]);
+
+	g_object_unref (fd_list);
+
+	dbus_send_and_splice_async (connection,
+	                            message,
+	                            pipefd[0],
+	                            cancellable,
+	                            get_metadata_fast_cb,
+	                            res);
+	g_object_unref (message);
+}
+
+void
+tracker_extract_client_get_metadata (GFile               *file,
+                                     const gchar         *mime_type,
+                                     GCancellable        *cancellable,
+                                     GAsyncReadyCallback  callback,
+                                     gpointer             user_data)
+{
+	GSimpleAsyncResult *res;
+	GError *error = NULL;
+	gchar *uri;
+
+	g_return_if_fail (G_IS_FILE (file));
+	g_return_if_fail (mime_type != NULL);
+	g_return_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable));
+	g_return_if_fail (callback != NULL);
+
+	if (G_UNLIKELY (!connection)) {
+		connection = g_bus_get_sync (G_BUS_TYPE_SESSION, cancellable, &error);
+
+		if (error) {
+			g_simple_async_report_gerror_in_idle (G_OBJECT (file), callback, user_data, error);
+			g_error_free (error);
+			return;
+		}
+	}
+
+	uri = g_file_get_uri (file);
+
+	res = g_simple_async_result_new (G_OBJECT (file), callback, user_data, NULL);
+	g_simple_async_result_set_handle_cancellation (res, TRUE);
+
+	get_metadata_fast_async (connection, uri, mime_type, cancellable, res);
+}
+
+TrackerExtractInfo *
+tracker_extract_client_get_metadata_finish (GFile         *file,
+                                            GAsyncResult  *res,
+                                            GError       **error)
+{
+	g_return_val_if_fail (G_IS_FILE (file), NULL);
+	g_return_val_if_fail (G_IS_ASYNC_RESULT (res), NULL);
+	g_return_val_if_fail (!error || !*error, NULL);
+
+	if (g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error)) {
+		return NULL;
+	}
+
+	return g_simple_async_result_get_op_res_gpointer (G_SIMPLE_ASYNC_RESULT (res));
+}
+
+G_CONST_RETURN gchar *
+tracker_extract_info_get_preupdate (TrackerExtractInfo *info)
+{
+	return info->preupdate;
+}
+
+G_CONST_RETURN gchar *
+tracker_extract_info_get_update (TrackerExtractInfo *info)
+{
+	return info->update;
+}
+
+G_CONST_RETURN gchar *
+tracker_extract_info_get_where_clause (TrackerExtractInfo *info)
+{
+	return info->where;
+}
+
+void
+tracker_extract_client_cancel_for_prefix (GFile *prefix)
+{
+	GDBusMessage *message;
+	gchar *uris[2];
+
+	uris[0] = g_file_get_uri (prefix);
+	uris[1] = NULL;
+
+	message = g_dbus_message_new_method_call (DBUS_SERVICE_EXTRACT,
+	                                          DBUS_PATH_EXTRACT,
+	                                          DBUS_INTERFACE_EXTRACT,
+	                                          "CancelTasks");
+
+	g_dbus_message_set_body (message, g_variant_new ("(^as)", uris));
+	g_dbus_connection_send_message (connection, message,
+	                                G_DBUS_SEND_MESSAGE_FLAGS_NONE,
+	                                NULL, NULL);
+
+	g_free (uris[0]);
+}
diff --git a/src/libtracker-extract/tracker-extract-client.h b/src/libtracker-extract/tracker-extract-client.h
new file mode 100644
index 0000000..63b07dc
--- /dev/null
+++ b/src/libtracker-extract/tracker-extract-client.h
@@ -0,0 +1,51 @@
+/*
+ * Copyright (C) 2011, Nokia <ivan frade nokia com>
+ *
+ * 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.1 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.
+ */
+
+#ifndef __TRACKER_EXTRACT_CLIENT_H__
+#define __TRACKER_EXTRACT_CLIENT_H__
+
+#if !defined (__LIBTRACKER_EXTRACT_INSIDE__) && !defined (TRACKER_COMPILATION)
+#error "only <libtracker-extract/tracker-extract.h> must be included directly."
+#endif
+
+#include <gio/gio.h>
+
+G_BEGIN_DECLS
+
+typedef struct TrackerExtractInfo TrackerExtractInfo;
+
+void                 tracker_extract_client_get_metadata        (GFile               *file,
+                                                                 const gchar         *mime_type,
+                                                                 GCancellable        *cancellable,
+                                                                 GAsyncReadyCallback  callback,
+                                                                 gpointer             user_data);
+
+TrackerExtractInfo * tracker_extract_client_get_metadata_finish (GFile               *uri,
+                                                                 GAsyncResult        *res,
+                                                                 GError             **error);
+
+void                 tracker_extract_client_cancel_for_prefix   (GFile               *uri);
+
+G_CONST_RETURN gchar * tracker_extract_info_get_preupdate    (TrackerExtractInfo *info);
+G_CONST_RETURN gchar * tracker_extract_info_get_update       (TrackerExtractInfo *info);
+G_CONST_RETURN gchar * tracker_extract_info_get_where_clause (TrackerExtractInfo *info);
+
+G_END_DECLS
+
+#endif /* __TRACKER_EXTRACT_CLIENT_H__ */
diff --git a/src/libtracker-extract/tracker-extract.h b/src/libtracker-extract/tracker-extract.h
index e66f2b2..8463b5d 100644
--- a/src/libtracker-extract/tracker-extract.h
+++ b/src/libtracker-extract/tracker-extract.h
@@ -27,6 +27,7 @@
 #include "tracker-data.h"
 #include "tracker-encoding.h"
 #include "tracker-exif.h"
+#include "tracker-extract-client.h"
 #include "tracker-module-manager.h"
 #include "tracker-guarantee.h"
 #include "tracker-iptc.h"
diff --git a/src/miners/fs/tracker-miner-files.c b/src/miners/fs/tracker-miner-files.c
index c8b3cde..dab81a9 100644
--- a/src/miners/fs/tracker-miner-files.c
+++ b/src/miners/fs/tracker-miner-files.c
@@ -43,25 +43,16 @@
 #include <libtracker-data/tracker-db-manager.h>
 
 #include <libtracker-extract/tracker-module-manager.h>
+#include <libtracker-extract/tracker-extract-client.h>
 
 #include "tracker-power.h"
 #include "tracker-miner-files.h"
 #include "tracker-config.h"
 #include "tracker-marshal.h"
 
-/* Size of buffers used when sending data over a pipe, using DBus FD passing */
-#define DBUS_PIPE_BUFFER_SIZE      65536
-
-#define DBUS_SERVICE_EXTRACT       "org.freedesktop.Tracker1.Extract"
-#define DBUS_PATH_EXTRACT          "/org/freedesktop/Tracker1/Extract"
-#define DBUS_INTERFACE_EXTRACT     "org.freedesktop.Tracker1.Extract"
-
 #define DISK_SPACE_CHECK_FREQUENCY 10
 #define SECONDS_PER_DAY 86400
 
-/* Default DBus timeout to be used in requests to extractor (milliseconds) */
-#define EXTRACTOR_DBUS_TIMEOUT 60000
-
 #define TRACKER_MINER_FILES_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), TRACKER_TYPE_MINER_FILES, TrackerMinerFilesPrivate))
 
 static GQuark miner_files_error_quark = 0;
@@ -75,34 +66,6 @@ struct ProcessFileData {
 	GFile *file;
 };
 
-typedef void (*fast_async_cb) (const gchar *preupdate,
-                               const gchar *sparql,
-                               const gchar *where,
-                               GError      *error,
-                               gpointer     user_data);
-
-typedef void (*TrackerDBusSendAndSpliceCallback) (void     *buffer,
-                                                  gssize    buffer_size,
-                                                  GError   *error, /* Don't free */
-                                                  gpointer  user_data);
-
-typedef struct {
-	GInputStream *unix_input_stream;
-	GInputStream *buffered_input_stream;
-	GOutputStream *output_stream;
-	TrackerDBusSendAndSpliceCallback callback;
-	GCancellable *cancellable;
-	gpointer user_data;
-	gboolean splice_finished;
-	gboolean dbus_finished;
-	GError *error;
-} SendAndSpliceData;
-
-typedef struct {
-	fast_async_cb callback;
-	gpointer user_data;
-} FastAsyncData;
-
 struct TrackerMinerFilesPrivate {
 	TrackerConfig *config;
 	TrackerStorage *storage;
@@ -244,8 +207,6 @@ static void        miner_files_in_removable_media_remove_by_date  (TrackerMinerF
 static void        miner_files_add_removable_or_optical_directory (TrackerMinerFiles *mf,
                                                                    const gchar       *mount_path,
                                                                    const gchar       *uuid);
-static void        extractor_cancel_tasks                         (GDBusConnection   *connection,
-                                                                   GFile             *prefix);
 
 static GInitableIface* miner_files_initable_parent_iface;
 
@@ -1150,7 +1111,7 @@ mount_point_removed_cb (TrackerStorage *storage,
 	mount_point_file = g_file_new_for_path (mount_point);
 
 	/* Notify extractor about cancellation of all tasks under the mount point */
-	extractor_cancel_tasks (miner->private->connection, mount_point_file);
+	tracker_extract_client_cancel_for_prefix (mount_point_file);
 
 	/* Tell TrackerMinerFS to skip monitoring everything under the mount
 	 *  point (in case there was no pre-unmount notification) */
@@ -1947,20 +1908,31 @@ process_file_data_free (ProcessFileData *data)
 }
 
 static void
-extractor_get_embedded_metadata_cb (const gchar *preupdate,
-                                    const gchar *sparql,
-                                    const gchar *where,
-                                    GError      *error,
-                                    gpointer     user_data)
+extractor_get_embedded_metadata_cb (GObject      *object,
+                                    GAsyncResult *res,
+                                    gpointer      user_data)
 {
 	ProcessFileData *data = user_data;
+	const gchar *preupdate, *sparql, *where;
+	TrackerExtractInfo *info;
+	GError *error = NULL;
 	const gchar *uuid;
 
-	if (error) {
-		/* Something bad happened, notify about the error */
-		tracker_miner_fs_file_notify (TRACKER_MINER_FS (data->miner), data->file, error);
-		process_file_data_free (data);
-		return;
+	if (res) {
+		info = tracker_extract_client_get_metadata_finish (G_FILE (object), res, &error);
+
+		if (error) {
+			/* Something bad happened, notify about the error */
+			tracker_miner_fs_file_notify (TRACKER_MINER_FS (data->miner), data->file, error);
+			process_file_data_free (data);
+			return;
+		}
+
+		preupdate = tracker_extract_info_get_preupdate (info);
+		sparql = tracker_extract_info_get_update (info);
+		where = tracker_extract_info_get_where_clause (info);
+	} else {
+		preupdate = sparql = where = NULL;
 	}
 
 	if (sparql && *sparql) {
@@ -2039,340 +2011,6 @@ extractor_get_embedded_metadata_cb (const gchar *preupdate,
 	process_file_data_free (data);
 }
 
-static SendAndSpliceData *
-send_and_splice_data_new (GInputStream                     *unix_input_stream,
-                          GInputStream                     *buffered_input_stream,
-                          GOutputStream                    *output_stream,
-                          GCancellable                     *cancellable,
-                          TrackerDBusSendAndSpliceCallback  callback,
-                          gpointer                          user_data)
-{
-	SendAndSpliceData *data;
-
-	data = g_slice_new0 (SendAndSpliceData);
-	data->unix_input_stream = unix_input_stream;
-	data->buffered_input_stream = buffered_input_stream;
-	data->output_stream = output_stream;
-	if (cancellable) {
-		data->cancellable = g_object_ref (cancellable);
-	}
-	data->callback = callback;
-	data->user_data = user_data;
-
-	return data;
-}
-
-static void
-send_and_splice_data_free (SendAndSpliceData *data)
-{
-	g_object_unref (data->unix_input_stream);
-	g_object_unref (data->buffered_input_stream);
-	g_object_unref (data->output_stream);
-	if (data->cancellable) {
-		g_object_unref (data->cancellable);
-	}
-	if (data->error) {
-		g_error_free (data->error);
-	}
-	g_slice_free (SendAndSpliceData, data);
-}
-
-static void
-dbus_send_and_splice_async_finish (SendAndSpliceData *data)
-{
-	if (!data->error) {
-		(* data->callback) (g_memory_output_stream_get_data (G_MEMORY_OUTPUT_STREAM (data->output_stream)),
-		                    g_memory_output_stream_get_data_size (G_MEMORY_OUTPUT_STREAM (data->output_stream)),
-		                    NULL,
-		                    data->user_data);
-	} else {
-		(* data->callback) (NULL, -1, data->error, data->user_data);
-	}
-
-	send_and_splice_data_free (data);
-}
-
-static void
-send_and_splice_splice_callback (GObject      *source,
-                                 GAsyncResult *result,
-                                 gpointer      user_data)
-{
-	SendAndSpliceData *data = user_data;
-	GError *error = NULL;
-
-	g_output_stream_splice_finish (G_OUTPUT_STREAM (source), result, &error);
-
-	if (error) {
-		if (!data->error) {
-			data->error = error;
-		} else {
-			g_error_free (error);
-		}
-	}
-
-	data->splice_finished = TRUE;
-
-	if (data->dbus_finished) {
-		dbus_send_and_splice_async_finish (data);
-	}
-}
-
-static void
-send_and_splice_dbus_callback (GObject      *source,
-                               GAsyncResult *result,
-                               gpointer      user_data)
-{
-	SendAndSpliceData *data = user_data;
-	GDBusMessage *reply;
-	GError *error = NULL;
-
-	reply = g_dbus_connection_send_message_with_reply_finish (G_DBUS_CONNECTION (source),
-	                                                          result, &error);
-
-	if (reply) {
-		if (g_dbus_message_get_message_type (reply) == G_DBUS_MESSAGE_TYPE_ERROR) {
-			g_dbus_message_to_gerror (reply, &error);
-		}
-
-		g_object_unref (reply);
-	}
-
-	if (error) {
-		if (!data->error) {
-			data->error = error;
-		} else {
-			g_error_free (error);
-		}
-	}
-
-	data->dbus_finished = TRUE;
-
-	if (data->splice_finished) {
-		dbus_send_and_splice_async_finish (data);
-	}
-}
-
-static gboolean
-dbus_send_and_splice_async (GDBusConnection                  *connection,
-                            GDBusMessage                     *message,
-                            int                               fd,
-                            GCancellable                     *cancellable,
-                            TrackerDBusSendAndSpliceCallback  callback,
-                            gpointer                          user_data)
-{
-	SendAndSpliceData *data;
-	GInputStream *unix_input_stream;
-	GInputStream *buffered_input_stream;
-	GOutputStream *output_stream;
-
-	g_return_val_if_fail (connection != NULL, FALSE);
-	g_return_val_if_fail (message != NULL, FALSE);
-	g_return_val_if_fail (fd > 0, FALSE);
-	g_return_val_if_fail (callback != NULL, FALSE);
-
-	unix_input_stream = g_unix_input_stream_new (fd, TRUE);
-	buffered_input_stream = g_buffered_input_stream_new_sized (unix_input_stream,
-	                                                           DBUS_PIPE_BUFFER_SIZE);
-	output_stream = g_memory_output_stream_new (NULL, 0, g_realloc, g_free);
-
-	data = send_and_splice_data_new (unix_input_stream,
-	                                 buffered_input_stream,
-	                                 output_stream,
-	                                 cancellable,
-	                                 callback,
-	                                 user_data);
-
-	g_dbus_connection_send_message_with_reply (connection,
-	                                           message,
-	                                           G_DBUS_SEND_MESSAGE_FLAGS_NONE,
-	                                           -1,
-	                                           NULL,
-	                                           cancellable,
-	                                           send_and_splice_dbus_callback,
-	                                           data);
-
-	g_output_stream_splice_async (data->output_stream,
-	                              data->buffered_input_stream,
-	                              G_OUTPUT_STREAM_SPLICE_CLOSE_SOURCE |
-	                              G_OUTPUT_STREAM_SPLICE_CLOSE_TARGET,
-	                              0,
-	                              data->cancellable,
-	                              send_and_splice_splice_callback,
-	                              data);
-
-	return TRUE;
-}
-
-static FastAsyncData*
-fast_async_data_new (fast_async_cb  callback,
-                     gpointer       user_data)
-{
-	FastAsyncData *data;
-
-	data = g_slice_new0 (FastAsyncData);
-
-	data->callback = callback;
-	data->user_data = user_data;
-
-	return data;
-}
-
-static void
-fast_async_data_free (FastAsyncData *data)
-{
-	g_slice_free (FastAsyncData, data);
-}
-
-static void
-get_metadata_fast_cb (void     *buffer,
-                      gssize    buffer_size,
-                      GError   *error,
-                      gpointer  user_data)
-{
-	FastAsyncData *data;
-	ProcessFileData *process_data;
-	gboolean valid_input = FALSE;
-	gboolean free_error = FALSE;
-	const gchar *preupdate = NULL;
-	const gchar *sparql = NULL;
-	const gchar *where = NULL;
-
-	data = user_data;
-	process_data = data->user_data;
-
-	if (!error && buffer_size) {
-		gsize preupdate_length, sparql_length, where_length;
-
-		preupdate = buffer;
-		preupdate_length = strnlen (preupdate, buffer_size - 1);
-		if (preupdate[preupdate_length] == '\0') {
-			/* sparql is stored just after preupdate in the original buffer */
-			sparql = preupdate + preupdate_length + 1;
-			sparql_length = strnlen (sparql, preupdate + buffer_size - 1 - sparql);
-			if (sparql[sparql_length] == '\0') {
-				/* where is stored just after sparql in the original buffer */
-				where = sparql + sparql_length + 1;
-				where_length = strnlen (where, preupdate + buffer_size - 1 - where);
-				if (where[where_length] == '\0') {
-					valid_input = TRUE;
-				}
-			}
-		}
-
-		if (!valid_input) {
-			error = g_error_new_literal (miner_files_error_quark,
-			                             0,
-			                             "Invalid data received from GetMetadataFast");
-			free_error = TRUE;
-		}
-	}
-
-	if (G_UNLIKELY (error)) {
-		(* data->callback) (NULL, NULL, NULL, error, process_data);
-
-		if (free_error) {
-			g_error_free (error);
-		}
-	} else {
-		(* data->callback) (preupdate, sparql, where, NULL, data->user_data);
-	}
-
-	fast_async_data_free (data);
-}
-
-static void
-get_metadata_fast_async (GDBusConnection *connection,
-                         const gchar     *uri,
-                         const gchar     *mime_type,
-                         GCancellable    *cancellable,
-                         fast_async_cb    callback,
-                         ProcessFileData *user_data)
-{
-	GDBusMessage *message;
-	GUnixFDList *fd_list;
-	FastAsyncData *data;
-	int pipefd[2];
-
-	g_return_if_fail (connection);
-	g_return_if_fail (uri);
-	g_return_if_fail (mime_type);
-	g_return_if_fail (callback);
-
-	if (pipe (pipefd) < 0) {
-		g_critical ("Coudln't open pipe");
-		return;
-	}
-
-	message = g_dbus_message_new_method_call (DBUS_SERVICE_EXTRACT,
-	                                          DBUS_PATH_EXTRACT,
-	                                          DBUS_INTERFACE_EXTRACT,
-	                                          "GetMetadataFast");
-
-	fd_list = g_unix_fd_list_new ();
-
-	g_dbus_message_set_body (message, g_variant_new ("(ssh)",
-	                                                 uri,
-	                                                 mime_type,
-	                                                 g_unix_fd_list_append (fd_list,
-	                                                                        pipefd[1],
-	                                                                        NULL)));
-	g_dbus_message_set_unix_fd_list (message, fd_list);
-
-	/* We need to close the fd as g_unix_fd_list_append duplicates the fd */
-
-	close (pipefd[1]);
-
-	g_object_unref (fd_list);
-
-	data = fast_async_data_new (callback,
-	                            user_data);
-
-	dbus_send_and_splice_async (connection,
-	                            message,
-	                            pipefd[0],
-	                            cancellable,
-	                            get_metadata_fast_cb,
-	                            data);
-
-	g_object_unref (message);
-}
-
-static void
-extractor_get_embedded_metadata (ProcessFileData *data,
-                                 const gchar     *uri,
-                                 const gchar     *mime_type)
-{
-	get_metadata_fast_async (data->miner->private->connection,
-	                         uri,
-	                         mime_type,
-	                         data->cancellable,
-	                         extractor_get_embedded_metadata_cb,
-	                         data);
-}
-
-static void
-extractor_cancel_tasks (GDBusConnection *connection,
-                        GFile           *prefix)
-{
-	GDBusMessage *message;
-	gchar *uris[2];
-
-	uris[0] = g_file_get_uri (prefix);
-	uris[1] = NULL;
-
-	message = g_dbus_message_new_method_call (DBUS_SERVICE_EXTRACT,
-	                                          DBUS_PATH_EXTRACT,
-	                                          DBUS_INTERFACE_EXTRACT,
-	                                          "CancelTasks");
-
-	g_dbus_message_set_body (message, g_variant_new ("(^as)", uris));
-	g_dbus_connection_send_message (connection, message,
-	                                G_DBUS_SEND_MESSAGE_FLAGS_NONE,
-	                                NULL, NULL);
-
-	g_free (uris[0]);
-}
-
 static void
 process_file_cb (GObject      *object,
                  GAsyncResult *result,
@@ -2465,11 +2103,15 @@ process_file_cb (GObject      *object,
 
 	if (tracker_extract_module_manager_mimetype_is_handled (mime_type)) {
 		/* Next step, if handled by the extractor, get embedded metadata */
-		extractor_get_embedded_metadata (data, uri, mime_type);
+		tracker_extract_client_get_metadata (data->file,
+		                                     mime_type,
+		                                     data->cancellable,
+		                                     extractor_get_embedded_metadata_cb,
+		                                     data);
 	} else {
 		/* Otherwise, don't request embedded metadata extraction. */
 		g_debug ("Avoiding embedded metadata request for uri '%s'", uri);
-		extractor_get_embedded_metadata_cb (NULL, NULL, NULL, NULL, user_data);
+		extractor_get_embedded_metadata_cb (NULL, NULL, user_data);
 	}
 
 	g_object_unref (file_info);



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]