/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
/* * Copyright © 2011 Igalia S.L.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include "ephy-history-service.h"
#include "ephy-history-service-private.h"
#include "ephy-history-types.h"
#include "ephy-history-type-builtins.h"
#include "ephy-sqlite-connection.h"
typedef gboolean (*EphyHistoryServiceMethod) (EphyHistoryService *self, gpointer data, gpointer *result);
typedef enum {
/* WRITE */
SET_URL_TITLE,
SET_URL_ZOOM_LEVEL,
SET_URL_HIDDEN,
SET_URL_THUMBNAIL_TIME,
ADD_VISIT,
ADD_VISITS,
DELETE_URLS,
DELETE_HOST,
CLEAR,
/* QUIT */
QUIT,
/* READ */
GET_URL,
GET_HOST_FOR_URL,
QUERY_URLS,
QUERY_VISITS,
GET_HOSTS,
QUERY_HOSTS
} EphyHistoryServiceMessageType;
enum {
VISIT_URL,
URLS_VISITED,
CLEARED,
URL_TITLE_CHANGED,
URL_DELETED,
HOST_DELETED,
LAST_SIGNAL
};
static guint signals[LAST_SIGNAL];
typedef struct _EphyHistoryServiceMessage {
EphyHistoryService *service;
EphyHistoryServiceMessageType type;
gpointer *method_argument;
gboolean success;
gpointer result;
gpointer user_data;
GCancellable *cancellable;
GDestroyNotify method_argument_cleanup;
EphyHistoryJobCallback callback;
} EphyHistoryServiceMessage;
static gpointer run_history_service_thread (EphyHistoryService *self);
static void ephy_history_service_process_message (EphyHistoryService *self, EphyHistoryServiceMessage *message);
static gboolean ephy_history_service_execute_quit (EphyHistoryService *self, gpointer data, gpointer *result);
static void ephy_history_service_quit (EphyHistoryService *self, EphyHistoryJobCallback callback, gpointer user_data);
enum {
PROP_0,
PROP_HISTORY_FILENAME,
};
#define EPHY_HISTORY_SERVICE_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE((o), EPHY_TYPE_HISTORY_SERVICE, EphyHistoryServicePrivate))
G_DEFINE_TYPE (EphyHistoryService, ephy_history_service, G_TYPE_OBJECT);
static void
ephy_history_service_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec)
{
EphyHistoryService *self = EPHY_HISTORY_SERVICE (object);
switch (property_id) {
case PROP_HISTORY_FILENAME:
g_free (self->priv->history_filename);
self->priv->history_filename = g_strdup (g_value_get_string (value));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (self, property_id, pspec);
break;
}
}
static void
ephy_history_service_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec)
{
EphyHistoryService *self = EPHY_HISTORY_SERVICE (object);
switch (property_id) {
case PROP_HISTORY_FILENAME:
g_value_set_string (value, self->priv->history_filename);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static void
ephy_history_service_finalize (GObject *self)
{
EphyHistoryServicePrivate *priv = EPHY_HISTORY_SERVICE (self)->priv;
ephy_history_service_quit (EPHY_HISTORY_SERVICE (self), NULL, NULL);
if (priv->history_thread)
g_thread_join (priv->history_thread);
g_free (priv->history_filename);
G_OBJECT_CLASS (ephy_history_service_parent_class)->finalize (self);
}
static void
ephy_history_service_dispose (GObject *self)
{
EphyHistoryServicePrivate *priv = EPHY_HISTORY_SERVICE (self)->priv;
if (priv->queue_urls_visited_id) {
g_source_remove (priv->queue_urls_visited_id);
priv->queue_urls_visited_id = 0;
}
}
static gboolean
emit_urls_visited (EphyHistoryService *self)
{
g_signal_emit (self, signals[URLS_VISITED], 0);
self->priv->queue_urls_visited_id = 0;
return FALSE;
}
static void
ephy_history_service_queue_urls_visited (EphyHistoryService *self)
{
if (self->priv->queue_urls_visited_id)
return;
self->priv->queue_urls_visited_id =
g_idle_add_full (G_PRIORITY_LOW, (GSourceFunc)emit_urls_visited, self, NULL);
}
static gboolean
impl_visit_url (EphyHistoryService *self, const char *url, EphyHistoryPageVisitType visit_type)
{
EphyHistoryPageVisit *visit;
visit = ephy_history_page_visit_new (url,
time (NULL),
visit_type);
ephy_history_service_add_visit (self,
visit, NULL, NULL, NULL);
ephy_history_page_visit_free (visit);
ephy_history_service_queue_urls_visited (self);
return FALSE;
}
static void
ephy_history_service_class_init (EphyHistoryServiceClass *klass)
{
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
gobject_class->finalize = ephy_history_service_finalize;
gobject_class->dispose = ephy_history_service_dispose;
gobject_class->get_property = ephy_history_service_get_property;
gobject_class->set_property = ephy_history_service_set_property;
klass->visit_url = impl_visit_url;
signals[VISIT_URL] =
g_signal_new ("visit-url",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyHistoryServiceClass, visit_url),
g_signal_accumulator_true_handled, NULL,
g_cclosure_marshal_generic,
G_TYPE_BOOLEAN,
2,
G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE,
EPHY_TYPE_HISTORY_PAGE_VISIT_TYPE);
/**
* EphyHistoryService::urls-visited:
* @service: the #EphyHistoryService that received the signal
*
* The ::urls-visited signal is emitted after one or more visits to
* URLS have taken place. This signal is intended for use-cases when
* precise information of the actual URLS visited is not important and
* there is only interest in the fact that there have been changes in
* the history. For more precise information, you can use ::visit-url
**/
signals[URLS_VISITED] =
g_signal_new ("urls-visited",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
0, NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE,
0);
signals[CLEARED] =
g_signal_new ("cleared",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
0, NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE,
0);
signals[URL_TITLE_CHANGED] =
g_signal_new ("url-title-changed",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
0, NULL, NULL,
g_cclosure_marshal_generic,
G_TYPE_NONE,
2,
G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE,
G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE);
signals[URL_DELETED] =
g_signal_new ("url-deleted",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
0, NULL, NULL,
g_cclosure_marshal_VOID__STRING,
G_TYPE_NONE,
1,
G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE);
signals[HOST_DELETED] =
g_signal_new ("host-deleted",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
0, NULL, NULL,
g_cclosure_marshal_VOID__STRING,
G_TYPE_NONE,
1,
G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE);
g_object_class_install_property (gobject_class,
PROP_HISTORY_FILENAME,
g_param_spec_string ("history-filename",
"History filename",
"The filename of the SQLite file holding containing history",
NULL,
G_PARAM_CONSTRUCT_ONLY | G_PARAM_WRITABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB));
g_type_class_add_private (gobject_class, sizeof (EphyHistoryServicePrivate));
}
static void
ephy_history_service_init (EphyHistoryService *self)
{
self->priv = EPHY_HISTORY_SERVICE_GET_PRIVATE (self);
self->priv->history_thread = g_thread_new ("EphyHistoryService", (GThreadFunc) run_history_service_thread, self);
self->priv->queue = g_async_queue_new ();
}
EphyHistoryService *
ephy_history_service_new (const char *history_filename)
{
return EPHY_HISTORY_SERVICE (g_object_new (EPHY_TYPE_HISTORY_SERVICE,
"history-filename", history_filename,
NULL));
}
static gint
sort_messages (EphyHistoryServiceMessage* a, EphyHistoryServiceMessage* b, gpointer user_data)
{
return a->type > b->type ? 1 : a->type == b->type ? 0 : -1;
}
static void
ephy_history_service_send_message (EphyHistoryService *self, EphyHistoryServiceMessage *message)
{
EphyHistoryServicePrivate *priv = self->priv;
g_async_queue_push_sorted (priv->queue, message, (GCompareDataFunc)sort_messages, NULL);
}
static void
ephy_history_service_commit (EphyHistoryService *self)
{
EphyHistoryServicePrivate *priv = self->priv;
GError *error = NULL;
g_assert (priv->history_thread == g_thread_self ());
if (NULL == priv->history_database)
return;
ephy_sqlite_connection_commit_transaction (priv->history_database, &error);
if (NULL != error) {
g_error ("Could not commit idle history database transaction: %s", error->message);
g_error_free (error);
}
ephy_sqlite_connection_begin_transaction (priv->history_database, &error);
if (NULL != error) {
g_error ("Could not start long-running history database transaction: %s", error->message);
g_error_free (error);
}
self->priv->scheduled_to_commit = FALSE;
}
static void
ephy_history_service_enable_foreign_keys (EphyHistoryService *self)
{
EphyHistoryServicePrivate *priv = EPHY_HISTORY_SERVICE (self)->priv;
GError *error = NULL;
if (NULL == priv->history_database)
return;
ephy_sqlite_connection_execute (priv->history_database,
"PRAGMA foreign_keys = ON", &error);
if (error) {
g_error ("Could not enable foreign keys pragma: %s", error->message);
g_error_free (error);
}
}
static gboolean
ephy_history_service_open_database_connections (EphyHistoryService *self)
{
EphyHistoryServicePrivate *priv = EPHY_HISTORY_SERVICE (self)->priv;
GError *error = NULL;
g_assert (priv->history_thread == g_thread_self ());
priv->history_database = ephy_sqlite_connection_new ();
ephy_sqlite_connection_open (priv->history_database, priv->history_filename, &error);
if (error) {
g_object_unref (priv->history_database);
priv->history_database = NULL;
g_error ("Could not open history database at %s: %s", priv->history_filename, error->message);
g_error_free (error);
return FALSE;
}
ephy_history_service_enable_foreign_keys (self);
ephy_sqlite_connection_begin_transaction (priv->history_database, &error);
if (error) {
g_error ("Could not begin long running transaction in history database: %s", error->message);
g_error_free (error);
return FALSE;
}
if ((ephy_history_service_initialize_hosts_table (self) == FALSE) ||
(ephy_history_service_initialize_urls_table (self) == FALSE) ||
(ephy_history_service_initialize_visits_table (self) == FALSE))
return FALSE;
return TRUE;
}
static void
ephy_history_service_close_database_connections (EphyHistoryService *self)
{
EphyHistoryServicePrivate *priv = EPHY_HISTORY_SERVICE (self)->priv;
g_assert (priv->history_thread == g_thread_self ());
ephy_sqlite_connection_close (priv->history_database);
g_object_unref (priv->history_database);
priv->history_database = NULL;
}
static void
ephy_history_service_clear_all (EphyHistoryService *self)
{
EphyHistoryServicePrivate *priv = EPHY_HISTORY_SERVICE (self)->priv;
GError *error = NULL;
if (NULL == priv->history_database)
return;
ephy_sqlite_connection_execute (priv->history_database,
"DELETE FROM hosts;", &error);
if (error) {
g_error ("Couldn't clear history database: %s", error->message);
g_error_free(error);
}
}
static gboolean
ephy_history_service_is_scheduled_to_quit (EphyHistoryService *self)
{
return self->priv->scheduled_to_quit;
}
static gboolean
ephy_history_service_is_scheduled_to_commit (EphyHistoryService *self)
{
return self->priv->scheduled_to_commit;
}
void
ephy_history_service_schedule_commit (EphyHistoryService *self)
{
self->priv->scheduled_to_commit = TRUE;
}
static gboolean
ephy_history_service_execute_quit (EphyHistoryService *self, gpointer data, gpointer *result)
{
EphyHistoryServicePrivate *priv = EPHY_HISTORY_SERVICE (self)->priv;
g_assert (priv->history_thread == g_thread_self ());
if (ephy_history_service_is_scheduled_to_commit (self))
ephy_history_service_commit (self);
g_async_queue_unref (priv->queue);
self->priv->scheduled_to_quit = TRUE;
return FALSE;
}
static gpointer
run_history_service_thread (EphyHistoryService *self)
{
EphyHistoryServicePrivate *priv = self->priv;
EphyHistoryServiceMessage *message;
g_assert (priv->history_thread == g_thread_self ());
if (ephy_history_service_open_database_connections (self) == FALSE)
return NULL;
do {
message = g_async_queue_try_pop (priv->queue);
if (!message) {
/* Schedule commit if needed. */
if (ephy_history_service_is_scheduled_to_commit (self))
ephy_history_service_commit (self);
/* Block the thread until there's data in the queue. */
message = g_async_queue_pop (priv->queue);
}
/* Process item. */
ephy_history_service_process_message (self, message);
} while (!ephy_history_service_is_scheduled_to_quit (self));
ephy_history_service_close_database_connections (self);
return NULL;
}
static EphyHistoryServiceMessage *
ephy_history_service_message_new (EphyHistoryService *service,
EphyHistoryServiceMessageType type,
gpointer method_argument,
GDestroyNotify method_argument_cleanup,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message = g_slice_alloc0 (sizeof (EphyHistoryServiceMessage));
message->service = service;
message->type = type;
message->method_argument = method_argument;
message->method_argument_cleanup = method_argument_cleanup;
message->cancellable = cancellable ? g_object_ref (cancellable) : NULL;
message->callback = callback;
message->user_data = user_data;
return message;
}
static void
ephy_history_service_message_free (EphyHistoryServiceMessage *message)
{
if (message->method_argument_cleanup)
message->method_argument_cleanup (message->method_argument);
if (message->cancellable)
g_object_unref (message->cancellable);
g_slice_free1 (sizeof (EphyHistoryServiceMessage), message);
}
static gboolean
ephy_history_service_execute_job_callback (gpointer data)
{
EphyHistoryServiceMessage *message = (EphyHistoryServiceMessage*) data;
g_assert (message->callback || message->type == CLEAR);
if (g_cancellable_is_cancelled (message->cancellable)) {
ephy_history_service_message_free (message);
return FALSE;
}
if (message->callback)
message->callback (message->service, message->success, message->result, message->user_data);
if (message->type == CLEAR)
g_signal_emit (message->service, signals[CLEARED], 0);
ephy_history_service_message_free (message);
return FALSE;
}
typedef struct {
EphyHistoryService *service;
gpointer user_data;
GDestroyNotify destroy_func;
} SignalEmissionContext;
static void
signal_emission_context_free (SignalEmissionContext *ctx)
{
g_object_unref (ctx->service);
if (ctx->destroy_func && ctx->user_data)
ctx->destroy_func (ctx->user_data);
g_slice_free (SignalEmissionContext, ctx);
}
static SignalEmissionContext *
signal_emission_context_new (EphyHistoryService *service,
gpointer user_data,
GDestroyNotify destroy_func)
{
SignalEmissionContext *ctx = g_slice_new0 (SignalEmissionContext);
ctx->service = g_object_ref (service);
ctx->user_data = user_data;
ctx->destroy_func = destroy_func;
return ctx;
}
static gboolean
ephy_history_service_execute_add_visit_helper (EphyHistoryService *self, EphyHistoryPageVisit *visit)
{
if (visit->url->host == NULL)
visit->url->host = ephy_history_service_get_host_row_from_url (self, visit->url->url);
else if (visit->url->host->id == -1) {
/* This will happen when we migrate the old history to the new
* format. We need to store a zoom level for a not-yet-created
* host, so we'll end up here. Ugly, but it works. */
double zoom_level = visit->url->host->zoom_level;
ephy_history_host_free (visit->url->host);
visit->url->host = ephy_history_service_get_host_row_from_url (self, visit->url->url);
visit->url->host->zoom_level = zoom_level;
}
visit->url->host->visit_count++;
ephy_history_service_update_host_row (self, visit->url->host);
/* A NULL return here means that the URL does not yet exist in the database */
if (NULL == ephy_history_service_get_url_row (self, visit->url->url, visit->url)) {
visit->url->last_visit_time = visit->visit_time;
visit->url->visit_count = 1;
ephy_history_service_add_url_row (self, visit->url);
if (visit->url->id == -1) {
g_error ("Adding visit failed after failed URL addition.");
return FALSE;
}
} else {
visit->url->visit_count++;
if (visit->visit_time > visit->url->last_visit_time)
visit->url->last_visit_time = visit->visit_time;
ephy_history_service_update_url_row (self, visit->url);
}
ephy_history_service_add_visit_row (self, visit);
return visit->id != -1;
}
static gboolean
ephy_history_service_execute_add_visit (EphyHistoryService *self, EphyHistoryPageVisit *visit, gpointer *result)
{
gboolean success;
g_assert (self->priv->history_thread == g_thread_self ());
success = ephy_history_service_execute_add_visit_helper (self, visit);
return success;
}
static gboolean
ephy_history_service_execute_add_visits (EphyHistoryService *self, GList *visits, gpointer *result)
{
gboolean success = TRUE;
g_assert (self->priv->history_thread == g_thread_self ());
while (visits) {
success = success && ephy_history_service_execute_add_visit_helper (self, (EphyHistoryPageVisit *) visits->data);
visits = visits->next;
}
ephy_history_service_schedule_commit (self);
return success;
}
static gboolean
ephy_history_service_execute_find_visits (EphyHistoryService *self, EphyHistoryQuery *query, gpointer *result)
{
GList *visits = ephy_history_service_find_visit_rows (self, query);
GList *current = visits;
/* FIXME: We don't have a good way to tell the difference between failures and empty returns */
while (current) {
EphyHistoryPageVisit *visit = (EphyHistoryPageVisit *) current->data;
if (NULL == ephy_history_service_get_url_row (self, NULL, visit->url)) {
ephy_history_page_visit_list_free (visits);
g_error ("Tried to process an orphaned page visit");
return FALSE;
}
current = current->next;
}
*result = visits;
return TRUE;
}
static gboolean
ephy_history_service_execute_get_hosts (EphyHistoryService *self,
gpointer pointer, gpointer *results)
{
GList *hosts;
hosts = ephy_history_service_get_all_hosts (self);
*results = hosts;
return TRUE;
}
static gboolean
ephy_history_service_execute_query_hosts (EphyHistoryService *self,
EphyHistoryQuery *query, gpointer *results)
{
GList *hosts;
hosts = ephy_history_service_find_host_rows (self, query);
*results = hosts;
return TRUE;
}
void
ephy_history_service_add_visit (EphyHistoryService *self, EphyHistoryPageVisit *visit, GCancellable *cancellable, EphyHistoryJobCallback callback, gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (visit != NULL);
message = ephy_history_service_message_new (self, ADD_VISIT,
ephy_history_page_visit_copy (visit),
(GDestroyNotify) ephy_history_page_visit_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
void
ephy_history_service_add_visits (EphyHistoryService *self, GList *visits, GCancellable *cancellable, EphyHistoryJobCallback callback, gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (visits != NULL);
message = ephy_history_service_message_new (self, ADD_VISITS,
ephy_history_page_visit_list_copy (visits),
(GDestroyNotify) ephy_history_page_visit_list_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
void
ephy_history_service_find_visits_in_time (EphyHistoryService *self, gint64 from, gint64 to, GCancellable *cancellable, EphyHistoryJobCallback callback, gpointer user_data)
{
EphyHistoryQuery *query;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
query = ephy_history_query_new ();
query->from = from;
query->to = to;
ephy_history_service_query_visits (self, query, cancellable, callback, user_data);
ephy_history_query_free (query);
}
void
ephy_history_service_query_visits (EphyHistoryService *self, EphyHistoryQuery *query, GCancellable *cancellable, EphyHistoryJobCallback callback, gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (query != NULL);
message = ephy_history_service_message_new (self, QUERY_VISITS,
ephy_history_query_copy (query), (GDestroyNotify) ephy_history_query_free, cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
ephy_history_service_execute_query_urls (EphyHistoryService *self, EphyHistoryQuery *query, gpointer *result)
{
GList *urls = ephy_history_service_find_url_rows (self, query);
*result = urls;
return TRUE;
}
void
ephy_history_service_query_urls (EphyHistoryService *self, EphyHistoryQuery *query, GCancellable *cancellable, EphyHistoryJobCallback callback, gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (query != NULL);
message = ephy_history_service_message_new (self, QUERY_URLS,
ephy_history_query_copy (query), (GDestroyNotify) ephy_history_query_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
void
ephy_history_service_get_hosts (EphyHistoryService *self,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
message = ephy_history_service_message_new (self, GET_HOSTS,
NULL, NULL, cancellable,
callback, user_data);
ephy_history_service_send_message (self, message);
}
void
ephy_history_service_query_hosts (EphyHistoryService *self,
EphyHistoryQuery *query,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
message = ephy_history_service_message_new (self, QUERY_HOSTS,
ephy_history_query_copy (query),
(GDestroyNotify) ephy_history_query_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
set_url_title_signal_emit (SignalEmissionContext *ctx)
{
EphyHistoryURL *url = (EphyHistoryURL *)ctx->user_data;
g_signal_emit (ctx->service, signals[URL_TITLE_CHANGED], 0, url->url, url->title);
return FALSE;
}
static gboolean
ephy_history_service_execute_set_url_title (EphyHistoryService *self,
EphyHistoryURL *url,
gpointer *result)
{
char *title = g_strdup (url->title);
if (NULL == ephy_history_service_get_url_row (self, NULL, url)) {
/* The URL is not yet in the database, so we can't update it.. */
g_free (title);
return FALSE;
} else {
SignalEmissionContext *ctx;
g_free (url->title);
url->title = title;
ephy_history_service_update_url_row (self, url);
ephy_history_service_schedule_commit (self);
ctx = signal_emission_context_new (self,
ephy_history_url_copy (url),
(GDestroyNotify)ephy_history_url_free);
g_idle_add_full (G_PRIORITY_DEFAULT_IDLE,
(GSourceFunc)set_url_title_signal_emit,
ctx, (GDestroyNotify)signal_emission_context_free);
return TRUE;
}
}
void
ephy_history_service_set_url_title (EphyHistoryService *self,
const char *orig_url,
const char *title,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryURL *url;
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (orig_url != NULL);
url = ephy_history_url_new (orig_url, title, 0, 0, 0);
message = ephy_history_service_message_new (self, SET_URL_TITLE,
url, (GDestroyNotify) ephy_history_url_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
ephy_history_service_execute_set_url_zoom_level (EphyHistoryService *self,
GVariant *variant,
gpointer *result)
{
char *url_string;
double zoom_level;
EphyHistoryHost *host;
g_variant_get (variant, "(sd)", &url_string, &zoom_level);
host = ephy_history_service_get_host_row_from_url (self, url_string);
g_free (url_string);
g_return_val_if_fail (host != NULL, FALSE);
host->zoom_level = zoom_level;
ephy_history_service_update_host_row (self, host);
ephy_history_service_schedule_commit (self);
return TRUE;
}
void
ephy_history_service_set_url_zoom_level (EphyHistoryService *self,
const char *url,
const double zoom_level,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
GVariant *variant;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (url != NULL);
variant = g_variant_new ("(sd)", url, zoom_level);
message = ephy_history_service_message_new (self, SET_URL_ZOOM_LEVEL,
variant, (GDestroyNotify)g_variant_unref,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
ephy_history_service_execute_set_url_hidden (EphyHistoryService *self,
EphyHistoryURL *url,
gpointer *result)
{
gboolean hidden;
hidden = url->hidden;
if (NULL == ephy_history_service_get_url_row (self, NULL, url)) {
/* The URL is not yet in the database, so we can't update it.. */
return FALSE;
} else {
url->hidden = hidden;
ephy_history_service_update_url_row (self, url);
ephy_history_service_schedule_commit (self);
return TRUE;
}
}
void
ephy_history_service_set_url_hidden (EphyHistoryService *self,
const char *orig_url,
gboolean hidden,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
EphyHistoryURL *url;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (orig_url != NULL);
url = ephy_history_url_new (orig_url, NULL, 0, 0, 0);
url->hidden = hidden;
message = ephy_history_service_message_new (self, SET_URL_HIDDEN,
url, (GDestroyNotify)ephy_history_url_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
ephy_history_service_execute_set_url_thumbnail_time (EphyHistoryService *self,
EphyHistoryURL *url,
gpointer *result)
{
int thumbnail_time;
thumbnail_time = url->thumbnail_time;
if (NULL == ephy_history_service_get_url_row (self, NULL, url))
return FALSE;
else {
url->thumbnail_time = thumbnail_time;
ephy_history_service_update_url_row (self, url);
ephy_history_service_schedule_commit (self);
return TRUE;
}
}
void
ephy_history_service_set_url_thumbnail_time (EphyHistoryService *self,
const char *orig_url,
int thumbnail_time,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryURL *url;
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (orig_url != NULL);
url = ephy_history_url_new (orig_url, NULL, 0, 0, 0);
url->thumbnail_time = thumbnail_time;
message = ephy_history_service_message_new (self, SET_URL_THUMBNAIL_TIME,
url, (GDestroyNotify) ephy_history_url_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
ephy_history_service_execute_get_url (EphyHistoryService *self,
const gchar *orig_url,
gpointer *result)
{
EphyHistoryURL *url;
url = ephy_history_service_get_url_row (self, orig_url, NULL);
*result = url;
return url != NULL;
}
void
ephy_history_service_get_url (EphyHistoryService *self,
const char *url,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (url != NULL);
message = ephy_history_service_message_new (self, GET_URL,
g_strdup (url), g_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
ephy_history_service_execute_get_host_for_url (EphyHistoryService *self,
const gchar *url,
gpointer *result)
{
EphyHistoryHost *host;
host = ephy_history_service_get_host_row_from_url (self, url);
g_return_val_if_fail (host != NULL, FALSE);
*result = host;
return host != NULL;
}
void
ephy_history_service_get_host_for_url (EphyHistoryService *self,
const char *url,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (url != NULL);
message = ephy_history_service_message_new (self, GET_HOST_FOR_URL,
g_strdup (url), g_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static gboolean
delete_urls_signal_emit (SignalEmissionContext *ctx)
{
char *url = (char *)ctx->user_data;
g_signal_emit (ctx->service, signals[URL_DELETED], 0, url);
return FALSE;
}
static gboolean
ephy_history_service_execute_delete_urls (EphyHistoryService *self,
GList *urls,
gpointer *result)
{
GList *l;
EphyHistoryURL *url;
SignalEmissionContext *ctx;
for (l = urls; l != NULL; l = l->next) {
url = l->data;
ephy_history_service_delete_url (self, url);
ctx = signal_emission_context_new (self, g_strdup (url->url),
(GDestroyNotify) g_free);
g_idle_add_full (G_PRIORITY_DEFAULT_IDLE,
(GSourceFunc)delete_urls_signal_emit,
ctx,
(GDestroyNotify)signal_emission_context_free);
}
ephy_history_service_delete_orphan_hosts (self);
ephy_history_service_schedule_commit (self);
return TRUE;
}
static gboolean
delete_host_signal_emit (SignalEmissionContext *ctx)
{
char *host = (char *)ctx->user_data;
g_signal_emit (ctx->service, signals[HOST_DELETED], 0, host);
return FALSE;
}
static gboolean
ephy_history_service_execute_delete_host (EphyHistoryService *self,
EphyHistoryHost *host,
EphyHistoryJobCallback callback,
gpointer user_data)
{
SignalEmissionContext *ctx;
ephy_history_service_delete_host_row (self, host);
ephy_history_service_schedule_commit (self);
ctx = signal_emission_context_new (self, g_strdup (host->url),
(GDestroyNotify)g_free);
g_idle_add_full (G_PRIORITY_DEFAULT_IDLE,
(GSourceFunc) delete_host_signal_emit,
ctx,
(GDestroyNotify) signal_emission_context_free);
return TRUE;
}
static gboolean
ephy_history_service_execute_clear (EphyHistoryService *self,
gpointer pointer,
gpointer *result)
{
ephy_history_service_clear_all (self);
ephy_history_service_schedule_commit (self);
return TRUE;
}
void
ephy_history_service_delete_urls (EphyHistoryService *self,
GList *urls,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (urls != NULL);
message = ephy_history_service_message_new (self, DELETE_URLS,
ephy_history_url_list_copy (urls), (GDestroyNotify)ephy_history_url_list_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
void
ephy_history_service_delete_host (EphyHistoryService *self,
EphyHistoryHost *host,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message =
ephy_history_service_message_new (self, DELETE_HOST,
ephy_history_host_copy (host), (GDestroyNotify)ephy_history_host_free,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
void
ephy_history_service_clear (EphyHistoryService *self,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
message = ephy_history_service_message_new (self, CLEAR,
NULL, NULL,
cancellable, callback, user_data);
ephy_history_service_send_message (self, message);
}
static void
ephy_history_service_quit (EphyHistoryService *self,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryServiceMessage *message =
ephy_history_service_message_new (self, QUIT,
NULL, NULL, NULL,
callback, user_data);
ephy_history_service_send_message (self, message);
}
static EphyHistoryServiceMethod methods[] = {
(EphyHistoryServiceMethod)ephy_history_service_execute_set_url_title,
(EphyHistoryServiceMethod)ephy_history_service_execute_set_url_zoom_level,
(EphyHistoryServiceMethod)ephy_history_service_execute_set_url_hidden,
(EphyHistoryServiceMethod)ephy_history_service_execute_set_url_thumbnail_time,
(EphyHistoryServiceMethod)ephy_history_service_execute_add_visit,
(EphyHistoryServiceMethod)ephy_history_service_execute_add_visits,
(EphyHistoryServiceMethod)ephy_history_service_execute_delete_urls,
(EphyHistoryServiceMethod)ephy_history_service_execute_delete_host,
(EphyHistoryServiceMethod)ephy_history_service_execute_clear,
(EphyHistoryServiceMethod)ephy_history_service_execute_quit,
(EphyHistoryServiceMethod)ephy_history_service_execute_get_url,
(EphyHistoryServiceMethod)ephy_history_service_execute_get_host_for_url,
(EphyHistoryServiceMethod)ephy_history_service_execute_query_urls,
(EphyHistoryServiceMethod)ephy_history_service_execute_find_visits,
(EphyHistoryServiceMethod)ephy_history_service_execute_get_hosts,
(EphyHistoryServiceMethod)ephy_history_service_execute_query_hosts
};
static gboolean
ephy_history_service_message_is_write (EphyHistoryServiceMessage *message)
{
return message->type < QUIT;
}
static void
ephy_history_service_process_message (EphyHistoryService *self,
EphyHistoryServiceMessage *message)
{
EphyHistoryServiceMethod method;
g_assert (self->priv->history_thread == g_thread_self ());
if (g_cancellable_is_cancelled (message->cancellable) &&
!ephy_history_service_message_is_write (message)) {
ephy_history_service_message_free (message);
return;
}
method = methods[message->type];
message->result = NULL;
message->success = method (message->service, message->method_argument, &message->result);
if (message->callback || message->type == CLEAR)
g_idle_add ((GSourceFunc)ephy_history_service_execute_job_callback, message);
else
ephy_history_service_message_free (message);
return;
}
/* Public API. */
void
ephy_history_service_find_urls (EphyHistoryService *self,
gint64 from, gint64 to,
guint limit, gint host,
GList *substring_list,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryQuery *query;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
query = ephy_history_query_new ();
query->from = from;
query->to = to;
query->substring_list = substring_list;
query->sort_type = EPHY_HISTORY_SORT_MV;
query->host = host;
if (limit != 0)
query->limit = limit;
ephy_history_service_query_urls (self,
query, cancellable,
callback, user_data);
ephy_history_query_free (query);
}
void
ephy_history_service_visit_url (EphyHistoryService *self,
const char *url,
EphyHistoryPageVisitType visit_type)
{
gboolean result;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
g_return_if_fail (url != NULL);
g_signal_emit (self, signals[VISIT_URL], 0, url, visit_type, &result);
}
void
ephy_history_service_find_hosts (EphyHistoryService *self,
gint64 from, gint64 to,
GCancellable *cancellable,
EphyHistoryJobCallback callback,
gpointer user_data)
{
EphyHistoryQuery *query;
g_return_if_fail (EPHY_IS_HISTORY_SERVICE (self));
query = ephy_history_query_new ();
query->from = from;
query->to = to;
ephy_history_service_query_hosts (self, query,
cancellable, callback, user_data);
ephy_history_query_free (query);
}