/*
* Copyright (C) 2018 Canonical Ltd
*
* Nautilus 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 3 of the
* License, or (at your option) any later version.
*
* Nautilus 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; see the file COPYING. If not,
* see .
*
* Author: Marco Trevisan
*/
#include
#include "nautilus-search-hit.h"
#include "nautilus-search-provider.h"
#include "nautilus-search-engine-recent.h"
#include "nautilus-search-engine-private.h"
#include "nautilus-ui-utilities.h"
#define DEBUG_FLAG NAUTILUS_DEBUG_SEARCH
#include "nautilus-debug.h"
#include
#include
#include
#define FILE_ATTRIBS G_FILE_ATTRIBUTE_STANDARD_IS_HIDDEN "," \
G_FILE_ATTRIBUTE_STANDARD_IS_BACKUP "," \
G_FILE_ATTRIBUTE_ACCESS_CAN_READ "," \
G_FILE_ATTRIBUTE_TIME_MODIFIED "," \
G_FILE_ATTRIBUTE_TIME_ACCESS "," \
G_FILE_ATTRIBUTE_TIME_CREATED
struct _NautilusSearchEngineRecent
{
GObject parent_instance;
NautilusQuery *query;
gboolean running;
GCancellable *cancellable;
GtkRecentManager *recent_manager;
guint add_hits_idle_id;
};
static void nautilus_search_provider_init (NautilusSearchProviderInterface *iface);
G_DEFINE_TYPE_WITH_CODE (NautilusSearchEngineRecent,
nautilus_search_engine_recent,
G_TYPE_OBJECT,
G_IMPLEMENT_INTERFACE (NAUTILUS_TYPE_SEARCH_PROVIDER,
nautilus_search_provider_init))
enum
{
PROP_0,
PROP_RUNNING,
LAST_PROP
};
NautilusSearchEngineRecent *
nautilus_search_engine_recent_new (void)
{
return g_object_new (NAUTILUS_TYPE_SEARCH_ENGINE_RECENT, NULL);
}
static void
nautilus_search_engine_recent_finalize (GObject *object)
{
NautilusSearchEngineRecent *self = NAUTILUS_SEARCH_ENGINE_RECENT (object);
g_clear_handle_id (&self->add_hits_idle_id, g_source_remove);
g_cancellable_cancel (self->cancellable);
g_clear_object (&self->query);
g_clear_object (&self->cancellable);
G_OBJECT_CLASS (nautilus_search_engine_recent_parent_class)->finalize (object);
}
typedef struct
{
NautilusSearchEngineRecent *recent;
GList *hits;
} SearchHitsData;
static gboolean
search_thread_add_hits_idle (gpointer user_data)
{
SearchHitsData *search_hits = user_data;
g_autoptr (NautilusSearchEngineRecent) self = search_hits->recent;
NautilusSearchProvider *provider = NAUTILUS_SEARCH_PROVIDER (self);
self->add_hits_idle_id = 0;
if (!g_cancellable_is_cancelled (self->cancellable))
{
nautilus_search_provider_hits_added (provider, search_hits->hits);
DEBUG ("Recent engine add hits");
}
self->running = FALSE;
g_list_free_full (search_hits->hits, g_object_unref);
g_clear_object (&self->cancellable);
g_free (search_hits);
nautilus_search_provider_finished (provider,
NAUTILUS_SEARCH_PROVIDER_STATUS_NORMAL);
g_object_notify (G_OBJECT (provider), "running");
return FALSE;
}
static void
search_add_hits_idle (NautilusSearchEngineRecent *self,
GList *hits)
{
SearchHitsData *search_hits;
if (self->add_hits_idle_id != 0)
{
g_list_free_full (hits, g_object_unref);
return;
}
search_hits = g_new0 (SearchHitsData, 1);
search_hits->recent = g_object_ref (self);
search_hits->hits = hits;
self->add_hits_idle_id = g_idle_add (search_thread_add_hits_idle, search_hits);
}
static gboolean
is_file_valid_recursive (NautilusSearchEngineRecent *self,
GFile *file,
GDateTime **mtime,
GDateTime **atime,
GDateTime **ctime,
GError **error)
{
g_autoptr (GFileInfo) file_info = NULL;
file_info = g_file_query_info (file, FILE_ATTRIBS,
G_FILE_QUERY_INFO_NONE,
self->cancellable, error);
if (*error != NULL)
{
return FALSE;
}
if (!g_file_info_get_attribute_boolean (file_info,
G_FILE_ATTRIBUTE_ACCESS_CAN_READ))
{
return FALSE;
}
if (mtime && atime && ctime)
{
*mtime = g_file_info_get_modification_date_time (file_info);
*atime = g_file_info_get_access_date_time (file_info);
*ctime = g_file_info_get_creation_date_time (file_info);
}
if (!nautilus_query_get_show_hidden_files (self->query))
{
if (!g_file_info_get_is_hidden (file_info) &&
!g_file_info_get_is_backup (file_info))
{
g_autoptr (GFile) parent = g_file_get_parent (file);
if (parent)
{
return is_file_valid_recursive (self, parent,
NULL, NULL, NULL,
error);
}
}
else
{
return FALSE;
}
}
return TRUE;
}
static gpointer
recent_thread_func (gpointer user_data)
{
g_autoptr (NautilusSearchEngineRecent) self = NAUTILUS_SEARCH_ENGINE_RECENT (user_data);
g_autoptr (GPtrArray) date_range = NULL;
g_autoptr (GFile) query_location = NULL;
g_autoptr (GPtrArray) mime_types = NULL;
GList *recent_items;
GList *hits;
GList *l;
g_return_val_if_fail (self->query, NULL);
hits = NULL;
recent_items = gtk_recent_manager_get_items (self->recent_manager);
mime_types = nautilus_query_get_mime_types (self->query);
date_range = nautilus_query_get_date_range (self->query);
query_location = nautilus_query_get_location (self->query);
for (l = recent_items; l != NULL; l = l->next)
{
GtkRecentInfo *info = l->data;
g_autoptr (GFile) file = NULL;
const gchar *uri;
const gchar *name;
gdouble rank;
uri = gtk_recent_info_get_uri (info);
file = g_file_new_for_uri (uri);
if (!g_file_has_prefix (file, query_location))
{
continue;
}
if (g_cancellable_is_cancelled (self->cancellable))
{
break;
}
name = gtk_recent_info_get_display_name (info);
rank = nautilus_query_matches_string (self->query, name);
if (rank <= 0)
{
g_autofree char *short_name = gtk_recent_info_get_short_name (info);
rank = nautilus_query_matches_string (self->query, short_name);
}
if (rank > 0)
{
NautilusSearchHit *hit;
g_autoptr (GDateTime) mtime = NULL;
g_autoptr (GDateTime) atime = NULL;
g_autoptr (GDateTime) ctime = NULL;
g_autoptr (GError) error = NULL;
if (!gtk_recent_info_is_local (info))
{
continue;
}
if (!is_file_valid_recursive (self, file, &mtime, &atime, &ctime, &error))
{
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
{
break;
}
if (error != NULL &&
!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS))
{
g_debug ("Impossible to read recent file info: %s",
error->message);
}
continue;
}
if (mime_types->len > 0)
{
const gchar *mime_type = gtk_recent_info_get_mime_type (info);
gboolean found = FALSE;
for (gint i = 0; mime_type != NULL && i < mime_types->len; i++)
{
if (g_content_type_is_a (mime_type, g_ptr_array_index (mime_types, i)))
{
found = TRUE;
break;
}
}
if (!found)
{
continue;
}
}
if (date_range != NULL)
{
NautilusQuerySearchType type;
GDateTime *target_date;
GDateTime *initial_date;
GDateTime *end_date;
initial_date = g_ptr_array_index (date_range, 0);
end_date = g_ptr_array_index (date_range, 1);
type = nautilus_query_get_search_type (self->query);
switch (type)
{
case NAUTILUS_QUERY_SEARCH_TYPE_LAST_ACCESS:
{
target_date = atime;
}
break;
case NAUTILUS_QUERY_SEARCH_TYPE_LAST_MODIFIED:
{
target_date = mtime;
}
break;
case NAUTILUS_QUERY_SEARCH_TYPE_CREATED:
{
target_date = ctime;
}
break;
default:
{
target_date = NULL;
}
}
if (!nautilus_date_time_is_between_dates (target_date,
initial_date,
end_date))
{
continue;
}
}
hit = nautilus_search_hit_new (uri);
nautilus_search_hit_set_fts_rank (hit, rank);
nautilus_search_hit_set_modification_time (hit, mtime);
nautilus_search_hit_set_access_time (hit, atime);
nautilus_search_hit_set_creation_time (hit, ctime);
hits = g_list_prepend (hits, hit);
}
}
search_add_hits_idle (self, hits);
g_list_free_full (recent_items, (GDestroyNotify) gtk_recent_info_unref);
return NULL;
}
static void
nautilus_search_engine_recent_start (NautilusSearchProvider *provider)
{
NautilusSearchEngineRecent *self = NAUTILUS_SEARCH_ENGINE_RECENT (provider);
g_autoptr (GFile) location = NULL;
g_autoptr (GThread) thread = NULL;
g_return_if_fail (self->query);
g_return_if_fail (self->cancellable == NULL);
location = nautilus_query_get_location (self->query);
if (!is_recursive_search (NAUTILUS_SEARCH_ENGINE_TYPE_INDEXED,
nautilus_query_get_recursive (self->query),
location))
{
search_add_hits_idle (self, NULL);
return;
}
self->running = TRUE;
self->cancellable = g_cancellable_new ();
thread = g_thread_new ("nautilus-search-recent", recent_thread_func,
g_object_ref (self));
g_object_notify (G_OBJECT (provider), "running");
}
static void
nautilus_search_engine_recent_stop (NautilusSearchProvider *provider)
{
NautilusSearchEngineRecent *self = NAUTILUS_SEARCH_ENGINE_RECENT (provider);
if (self->cancellable != NULL)
{
DEBUG ("Recent engine stop");
g_cancellable_cancel (self->cancellable);
}
self->running = FALSE;
}
static void
nautilus_search_engine_recent_set_query (NautilusSearchProvider *provider,
NautilusQuery *query)
{
NautilusSearchEngineRecent *self = NAUTILUS_SEARCH_ENGINE_RECENT (provider);
g_clear_object (&self->query);
self->query = g_object_ref (query);
}
static gboolean
nautilus_search_engine_recent_is_running (NautilusSearchProvider *provider)
{
NautilusSearchEngineRecent *self = NAUTILUS_SEARCH_ENGINE_RECENT (provider);
return self->running;
}
static void
nautilus_search_engine_recent_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
NautilusSearchProvider *provider = NAUTILUS_SEARCH_PROVIDER (object);
switch (prop_id)
{
case PROP_RUNNING:
{
gboolean running;
running = nautilus_search_engine_recent_is_running (provider);
g_value_set_boolean (value, running);
}
break;
default:
{
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
}
static void
nautilus_search_provider_init (NautilusSearchProviderInterface *iface)
{
iface->set_query = nautilus_search_engine_recent_set_query;
iface->start = nautilus_search_engine_recent_start;
iface->stop = nautilus_search_engine_recent_stop;
iface->is_running = nautilus_search_engine_recent_is_running;
}
static void
nautilus_search_engine_recent_class_init (NautilusSearchEngineRecentClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->finalize = nautilus_search_engine_recent_finalize;
object_class->get_property = nautilus_search_engine_recent_get_property;
g_object_class_override_property (object_class, PROP_RUNNING, "running");
}
static void
nautilus_search_engine_recent_init (NautilusSearchEngineRecent *self)
{
self->recent_manager = gtk_recent_manager_get_default ();
}