/* * 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 (); }