Blame gtk/gtksearchenginebeagle.c

Packit 98cdb6
/*
Packit 98cdb6
 * Copyright (C) 2005 Novell, Inc.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Lesser General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Lesser General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Lesser General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 *
Packit 98cdb6
 * Author: Anders Carlsson <andersca@imendio.com>
Packit 98cdb6
 *
Packit 98cdb6
 * Based on nautilus-search-engine-beagle.c
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "config.h"
Packit 98cdb6
#include <gmodule.h>
Packit 98cdb6
#include "gtksearchenginebeagle.h"
Packit 98cdb6
Packit 98cdb6
#if 0
Packit 98cdb6
#include <beagle/beagle.h>
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
/* We dlopen() all the following from libbeagle at runtime */
Packit 98cdb6
Packit 98cdb6
typedef struct _BeagleHit BeagleHit;
Packit 98cdb6
typedef struct _BeagleQuery BeagleQuery;
Packit 98cdb6
typedef struct _BeagleClient BeagleClient;
Packit 98cdb6
typedef struct _BeagleRequest BeagleRequest;
Packit 98cdb6
typedef struct _BeagleFinishedResponse BeagleFinishedResponse;
Packit 98cdb6
typedef struct _BeagleHitsAddedResponse BeagleHitsAddedResponse;
Packit 98cdb6
typedef struct _BeagleHitsSubtractedResponse BeagleHitsSubtractedResponse;
Packit 98cdb6
typedef struct _BeagleQueryPartProperty BeagleQueryPartProperty;
Packit 98cdb6
typedef struct _BeagleQueryPart BeagleQueryPart;
Packit 98cdb6
Packit 98cdb6
#define BEAGLE_HIT(x) ((BeagleHit *)(x))
Packit 98cdb6
#define BEAGLE_REQUEST(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), beagle_request_get_type(), BeagleRequest))
Packit 98cdb6
#define BEAGLE_QUERY_PART(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), beagle_query_part_get_type(), BeagleQueryPart))
Packit 98cdb6
Packit 98cdb6
typedef enum 
Packit 98cdb6
{
Packit 98cdb6
  BEAGLE_QUERY_PART_LOGIC_REQUIRED   = 1,
Packit 98cdb6
  BEAGLE_QUERY_PART_LOGIC_PROHIBITED = 2
Packit 98cdb6
} BeagleQueryPartLogic;
Packit 98cdb6
Packit 98cdb6
typedef enum 
Packit 98cdb6
  {
Packit 98cdb6
    BEAGLE_PROPERTY_TYPE_UNKNOWN = 0,
Packit 98cdb6
    BEAGLE_PROPERTY_TYPE_TEXT    = 1,
Packit 98cdb6
    BEAGLE_PROPERTY_TYPE_KEYWORD = 2,
Packit 98cdb6
    BEAGLE_PROPERTY_TYPE_DATE    = 3,
Packit 98cdb6
    BEAGLE_PROPERTY_TYPE_LAST    = 4
Packit 98cdb6
} BeaglePropertyType;
Packit 98cdb6
Packit 98cdb6
/* *static* wrapper function pointers */
Packit 98cdb6
static gboolean (*beagle_client_send_request_async) (BeagleClient   *client,
Packit 98cdb6
						     BeagleRequest  *request,
Packit 98cdb6
						     GError        **err) = NULL;
Packit 98cdb6
static const char *(*beagle_hit_get_uri) (BeagleHit *hit) = NULL;
Packit 98cdb6
static GSList *(*beagle_hits_added_response_get_hits) (BeagleHitsAddedResponse *response) = NULL;
Packit 98cdb6
static GSList *(*beagle_hits_subtracted_response_get_uris) (BeagleHitsSubtractedResponse *response) = NULL;
Packit 98cdb6
static BeagleQuery *(*beagle_query_new) (void) = NULL;
Packit 98cdb6
static void (*beagle_query_add_text) (BeagleQuery     *query,
Packit 98cdb6
				      const char      *str) = NULL;
Packit 98cdb6
static void (*beagle_query_set_max_hits) (BeagleQuery *query,
Packit 98cdb6
					  gint         max_hits) = NULL;
Packit 98cdb6
static BeagleQueryPartProperty *(*beagle_query_part_property_new) (void) = NULL;
Packit 98cdb6
static void (*beagle_query_part_set_logic) (BeagleQueryPart      *part, 
Packit 98cdb6
					    BeagleQueryPartLogic  logic) = NULL;
Packit 98cdb6
static void (*beagle_query_part_property_set_key) (BeagleQueryPartProperty *part, 
Packit 98cdb6
						   const char              *key) = NULL;
Packit 98cdb6
static void (*beagle_query_part_property_set_value) (BeagleQueryPartProperty *part, 
Packit 98cdb6
						     const char *             value) = NULL;
Packit 98cdb6
static void (*beagle_query_part_property_set_property_type) (BeagleQueryPartProperty *part, 
Packit 98cdb6
							     BeaglePropertyType       prop_type) = NULL;
Packit 98cdb6
static void (*beagle_query_add_part) (BeagleQuery     *query, 
Packit 98cdb6
				      BeagleQueryPart *part) = NULL;
Packit 98cdb6
static GType (*beagle_request_get_type) (void) = NULL;
Packit 98cdb6
static GType (*beagle_query_part_get_type) (void) = NULL;
Packit 98cdb6
static gboolean (*beagle_util_daemon_is_running) (void) = NULL;
Packit 98cdb6
static BeagleClient *(*beagle_client_new) (const char *client_name) = NULL;
Packit 98cdb6
Packit 98cdb6
static struct BeagleDlMapping
Packit 98cdb6
{
Packit 98cdb6
  const char *fn_name;
Packit 98cdb6
  gpointer *fn_ptr_ref;
Packit 98cdb6
} beagle_dl_mapping[] =
Packit 98cdb6
{
Packit 98cdb6
#define MAP(a) { #a, (gpointer *)&a }
Packit 98cdb6
  MAP (beagle_client_send_request_async),
Packit 98cdb6
  MAP (beagle_hit_get_uri),
Packit 98cdb6
  MAP (beagle_hits_added_response_get_hits),
Packit 98cdb6
  MAP (beagle_hits_subtracted_response_get_uris),
Packit 98cdb6
  MAP (beagle_query_new),
Packit 98cdb6
  MAP (beagle_query_add_text),
Packit 98cdb6
  MAP (beagle_query_set_max_hits),
Packit 98cdb6
  MAP (beagle_query_part_property_new),
Packit 98cdb6
  MAP (beagle_query_part_set_logic),
Packit 98cdb6
  MAP (beagle_query_part_property_set_key),
Packit 98cdb6
  MAP (beagle_query_part_property_set_value),
Packit 98cdb6
  MAP (beagle_query_part_property_set_property_type),
Packit 98cdb6
  MAP (beagle_query_add_part),
Packit 98cdb6
  MAP (beagle_request_get_type),
Packit 98cdb6
  MAP (beagle_query_part_get_type),
Packit 98cdb6
  MAP (beagle_util_daemon_is_running),
Packit 98cdb6
  MAP (beagle_client_new)
Packit 98cdb6
#undef MAP
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
static void 
Packit 98cdb6
open_libbeagle (void)
Packit 98cdb6
{
Packit 98cdb6
  static gboolean done = FALSE;
Packit 98cdb6
Packit 98cdb6
  if (!done)
Packit 98cdb6
    {
Packit 98cdb6
      int i;
Packit 98cdb6
      GModule *beagle;
Packit 98cdb6
      
Packit 98cdb6
      done = TRUE;
Packit 98cdb6
 
Packit 98cdb6
      beagle = g_module_open ("libbeagle.so.1", G_MODULE_BIND_LAZY | G_MODULE_BIND_LOCAL);
Packit 98cdb6
      if (!beagle)
Packit 98cdb6
        beagle = g_module_open ("libbeagle.so.0", G_MODULE_BIND_LAZY | G_MODULE_BIND_LOCAL);
Packit 98cdb6
Packit 98cdb6
      if (!beagle)
Packit 98cdb6
	return;
Packit 98cdb6
      
Packit 98cdb6
      for (i = 0; i < G_N_ELEMENTS (beagle_dl_mapping); i++)
Packit 98cdb6
	{
Packit 98cdb6
	  if (!g_module_symbol (beagle, beagle_dl_mapping[i].fn_name,
Packit 98cdb6
				beagle_dl_mapping[i].fn_ptr_ref))
Packit 98cdb6
	    {
Packit 98cdb6
	      g_warning ("Missing symbol '%s' in libbeagle\n",
Packit 98cdb6
			 beagle_dl_mapping[i].fn_name);
Packit 98cdb6
	      g_module_close (beagle);
Packit 98cdb6
Packit 98cdb6
	      for (i = 0; i < G_N_ELEMENTS (beagle_dl_mapping); i++)
Packit 98cdb6
		beagle_dl_mapping[i].fn_ptr_ref = NULL;
Packit 98cdb6
Packit 98cdb6
	      return;
Packit 98cdb6
	    }
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
struct _GtkSearchEngineBeaglePrivate 
Packit 98cdb6
{
Packit 98cdb6
  BeagleClient *client;
Packit 98cdb6
  GtkQuery *query;
Packit 98cdb6
Packit 98cdb6
  BeagleQuery *current_query;
Packit 98cdb6
  char *current_query_uri_prefix;
Packit 98cdb6
  gboolean query_finished;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
G_DEFINE_TYPE (GtkSearchEngineBeagle, _gtk_search_engine_beagle, GTK_TYPE_SEARCH_ENGINE);
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
finalize (GObject *object)
Packit 98cdb6
{
Packit 98cdb6
  GtkSearchEngineBeagle *beagle;
Packit 98cdb6
  
Packit 98cdb6
  beagle = GTK_SEARCH_ENGINE_BEAGLE (object);
Packit 98cdb6
  
Packit 98cdb6
  if (beagle->priv->current_query) 
Packit 98cdb6
    {
Packit 98cdb6
      g_object_unref (beagle->priv->current_query);
Packit 98cdb6
      beagle->priv->current_query = NULL;
Packit 98cdb6
      g_free (beagle->priv->current_query_uri_prefix);
Packit 98cdb6
      beagle->priv->current_query_uri_prefix = NULL;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (beagle->priv->query) 
Packit 98cdb6
    {
Packit 98cdb6
      g_object_unref (beagle->priv->query);
Packit 98cdb6
      beagle->priv->query = NULL;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (beagle->priv->client) 
Packit 98cdb6
    {
Packit 98cdb6
      g_object_unref (beagle->priv->client);
Packit 98cdb6
      beagle->priv->client = NULL;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  G_OBJECT_CLASS (_gtk_search_engine_beagle_parent_class)->finalize (object);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
beagle_hits_added (BeagleQuery             *query, 
Packit 98cdb6
		   BeagleHitsAddedResponse *response, 
Packit 98cdb6
		   GtkSearchEngineBeagle   *engine)
Packit 98cdb6
{
Packit 98cdb6
  GSList *hits, *list;
Packit 98cdb6
  GList *hit_uris;
Packit 98cdb6
  const gchar *uri;
Packit 98cdb6
  
Packit 98cdb6
  hit_uris = NULL;
Packit 98cdb6
  
Packit 98cdb6
  hits = beagle_hits_added_response_get_hits (response);
Packit 98cdb6
  
Packit 98cdb6
  for (list = hits; list != NULL; list = list->next) 
Packit 98cdb6
    {
Packit 98cdb6
      BeagleHit *hit = BEAGLE_HIT (list->data);
Packit 98cdb6
      
Packit 98cdb6
      uri = beagle_hit_get_uri (hit);
Packit 98cdb6
      
Packit 98cdb6
      if (engine->priv->current_query_uri_prefix &&
Packit 98cdb6
	  !g_str_has_prefix (uri, engine->priv->current_query_uri_prefix)) 
Packit 98cdb6
	continue;
Packit 98cdb6
		
Packit 98cdb6
      hit_uris = g_list_prepend (hit_uris, (char *)uri);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  _gtk_search_engine_hits_added (GTK_SEARCH_ENGINE (engine), hit_uris);
Packit 98cdb6
  g_list_free (hit_uris);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
beagle_hits_subtracted (BeagleQuery                  *query, 
Packit 98cdb6
			BeagleHitsSubtractedResponse *response, 
Packit 98cdb6
			GtkSearchEngineBeagle        *engine)
Packit 98cdb6
{
Packit 98cdb6
  GSList *uris, *list;
Packit 98cdb6
  GList *hit_uris;
Packit 98cdb6
  
Packit 98cdb6
  hit_uris = NULL;
Packit 98cdb6
  
Packit 98cdb6
  uris = beagle_hits_subtracted_response_get_uris (response);
Packit 98cdb6
  
Packit 98cdb6
  for (list = uris; list != NULL; list = list->next) 
Packit 98cdb6
    {
Packit 98cdb6
      hit_uris = g_list_prepend (hit_uris, (char *)list->data);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  _gtk_search_engine_hits_subtracted (GTK_SEARCH_ENGINE (engine), hit_uris);
Packit 98cdb6
  g_list_free (hit_uris);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
beagle_finished (BeagleQuery            *query, 
Packit 98cdb6
		 BeagleFinishedResponse *response,
Packit 98cdb6
		 GtkSearchEngineBeagle  *engine)
Packit 98cdb6
{
Packit 98cdb6
  /* For some reason we keep getting finished events,
Packit 98cdb6
   * only emit finished once */
Packit 98cdb6
  if (engine->priv->query_finished) 
Packit 98cdb6
    return;
Packit 98cdb6
  
Packit 98cdb6
  engine->priv->query_finished = TRUE;
Packit 98cdb6
  _gtk_search_engine_finished (GTK_SEARCH_ENGINE (engine));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
beagle_error (BeagleQuery           *query,
Packit 98cdb6
	      GError                *error,
Packit 98cdb6
	      GtkSearchEngineBeagle *engine)
Packit 98cdb6
{
Packit 98cdb6
  _gtk_search_engine_error (GTK_SEARCH_ENGINE (engine), error->message);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_search_engine_beagle_start (GtkSearchEngine *engine)
Packit 98cdb6
{
Packit 98cdb6
  GtkSearchEngineBeagle *beagle;
Packit 98cdb6
  GError *error;
Packit 98cdb6
  gchar *text;
Packit 98cdb6
  gchar *query;
Packit 98cdb6
Packit 98cdb6
  error = NULL;
Packit 98cdb6
  beagle = GTK_SEARCH_ENGINE_BEAGLE (engine);
Packit 98cdb6
  
Packit 98cdb6
  g_return_if_fail (beagle->priv->query != NULL);
Packit 98cdb6
Packit 98cdb6
  if (beagle->priv->current_query)
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  beagle->priv->query_finished = FALSE;
Packit 98cdb6
  beagle->priv->current_query = beagle_query_new ();
Packit 98cdb6
  g_signal_connect (beagle->priv->current_query,
Packit 98cdb6
		    "hits-added", G_CALLBACK (beagle_hits_added), engine);
Packit 98cdb6
  g_signal_connect (beagle->priv->current_query,
Packit 98cdb6
		    "hits-subtracted", G_CALLBACK (beagle_hits_subtracted), engine);
Packit 98cdb6
  g_signal_connect (beagle->priv->current_query,
Packit 98cdb6
		    "finished", G_CALLBACK (beagle_finished), engine);
Packit 98cdb6
  g_signal_connect (beagle->priv->current_query,
Packit 98cdb6
		    "error", G_CALLBACK (beagle_error), engine);
Packit 98cdb6
  
Packit 98cdb6
  /* We only want files */
Packit 98cdb6
  
Packit 98cdb6
 
Packit 98cdb6
  text = _gtk_query_get_text (beagle->priv->query);
Packit 98cdb6
  query = g_strconcat (text, " type:File", NULL);
Packit 98cdb6
                          
Packit 98cdb6
  beagle_query_set_max_hits (beagle->priv->current_query, 1000);
Packit 98cdb6
  beagle_query_add_text (beagle->priv->current_query, query);
Packit 98cdb6
  
Packit 98cdb6
  beagle->priv->current_query_uri_prefix = _gtk_query_get_location (beagle->priv->query);
Packit 98cdb6
  
Packit 98cdb6
  if (!beagle_client_send_request_async (beagle->priv->client,
Packit 98cdb6
					 BEAGLE_REQUEST (beagle->priv->current_query), &error)) 
Packit 98cdb6
    {
Packit 98cdb6
      _gtk_search_engine_error (engine, error->message);
Packit 98cdb6
      g_error_free (error);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* These must live during the lifetime of the query */
Packit 98cdb6
  g_free (text);
Packit 98cdb6
  g_free (query);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_search_engine_beagle_stop (GtkSearchEngine *engine)
Packit 98cdb6
{
Packit 98cdb6
  GtkSearchEngineBeagle *beagle;
Packit 98cdb6
  
Packit 98cdb6
  beagle = GTK_SEARCH_ENGINE_BEAGLE (engine);
Packit 98cdb6
  
Packit 98cdb6
  if (beagle->priv->current_query) 
Packit 98cdb6
    {
Packit 98cdb6
      g_object_unref (beagle->priv->current_query);
Packit 98cdb6
      beagle->priv->current_query = NULL;
Packit 98cdb6
Packit 98cdb6
      g_free (beagle->priv->current_query_uri_prefix);
Packit 98cdb6
      beagle->priv->current_query_uri_prefix = NULL;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_search_engine_beagle_is_indexed (GtkSearchEngine *engine)
Packit 98cdb6
{
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_search_engine_beagle_set_query (GtkSearchEngine *engine, 
Packit 98cdb6
				    GtkQuery        *query)
Packit 98cdb6
{
Packit 98cdb6
  GtkSearchEngineBeagle *beagle;
Packit 98cdb6
  
Packit 98cdb6
  beagle = GTK_SEARCH_ENGINE_BEAGLE (engine);
Packit 98cdb6
  
Packit 98cdb6
  if (query)
Packit 98cdb6
    g_object_ref (query);
Packit 98cdb6
Packit 98cdb6
  if (beagle->priv->query)
Packit 98cdb6
    g_object_unref (beagle->priv->query);
Packit 98cdb6
Packit 98cdb6
  beagle->priv->query = query;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
_gtk_search_engine_beagle_class_init (GtkSearchEngineBeagleClass *class)
Packit 98cdb6
{
Packit 98cdb6
  GObjectClass *gobject_class;
Packit 98cdb6
  GtkSearchEngineClass *engine_class;
Packit 98cdb6
  
Packit 98cdb6
  gobject_class = G_OBJECT_CLASS (class);
Packit 98cdb6
  gobject_class->finalize = finalize;
Packit 98cdb6
  
Packit 98cdb6
  engine_class = GTK_SEARCH_ENGINE_CLASS (class);
Packit 98cdb6
  engine_class->set_query = gtk_search_engine_beagle_set_query;
Packit 98cdb6
  engine_class->start = gtk_search_engine_beagle_start;
Packit 98cdb6
  engine_class->stop = gtk_search_engine_beagle_stop;
Packit 98cdb6
  engine_class->is_indexed = gtk_search_engine_beagle_is_indexed;
Packit 98cdb6
Packit 98cdb6
  g_type_class_add_private (gobject_class, sizeof (GtkSearchEngineBeaglePrivate));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
_gtk_search_engine_beagle_init (GtkSearchEngineBeagle *engine)
Packit 98cdb6
{
Packit 98cdb6
  engine->priv = G_TYPE_INSTANCE_GET_PRIVATE (engine, GTK_TYPE_SEARCH_ENGINE_BEAGLE, GtkSearchEngineBeaglePrivate);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
GtkSearchEngine *
Packit 98cdb6
_gtk_search_engine_beagle_new (void)
Packit 98cdb6
{
Packit 98cdb6
  GtkSearchEngineBeagle *engine;
Packit 98cdb6
  BeagleClient *client;
Packit 98cdb6
Packit 98cdb6
  open_libbeagle ();
Packit 98cdb6
Packit 98cdb6
  if (!beagle_util_daemon_is_running)
Packit 98cdb6
    return NULL;
Packit 98cdb6
Packit 98cdb6
  /* check whether daemon is running as beagle_client_new
Packit 98cdb6
   * doesn't fail when a stale socket file exists */
Packit 98cdb6
  if (!beagle_util_daemon_is_running ()) 
Packit 98cdb6
      return NULL;
Packit 98cdb6
Packit 98cdb6
  client = beagle_client_new (NULL);
Packit 98cdb6
  
Packit 98cdb6
  if (client == NULL)
Packit 98cdb6
    return NULL;
Packit 98cdb6
	
Packit 98cdb6
  engine = g_object_new (GTK_TYPE_SEARCH_ENGINE_BEAGLE, NULL);
Packit 98cdb6
  
Packit 98cdb6
  engine->priv->client = client;
Packit 98cdb6
  
Packit 98cdb6
  return GTK_SEARCH_ENGINE (engine);
Packit 98cdb6
}