Add search builtin

This allows searching for applications easily greatly improving
the UX of the cli. This relies upon appstream data which currently
must be updated with `flatpak update --appstream`.

Closes: #1136
Approved by: mwleeds
This commit is contained in:
Patrick Griffis 2017-08-30 06:18:39 -04:00 committed by Atomic Bot
parent 3f7321292b
commit 0f33ba53ff
7 changed files with 273 additions and 3 deletions

View File

@ -37,8 +37,10 @@ flatpak_SOURCES = \
app/flatpak-builtins-document-unexport.c \
app/flatpak-builtins-document-info.c \
app/flatpak-builtins-document-list.c \
app/flatpak-builtins-search.c \
$(NULL)
flatpak_LDADD = $(AM_LDADD) $(BASE_LIBS) $(OSTREE_LIBS) $(SOUP_LIBS) $(JSON_LIBS) libglnx.la libflatpak-common.la
flatpak_CFLAGS = $(AM_CFLAGS) $(BASE_CFLAGS) $(OSTREE_CFLAGS) $(SOUP_CFLAGS) $(JSON_CFLAGS) \
flatpak_LDADD = $(AM_LDADD) $(BASE_LIBS) $(OSTREE_LIBS) $(SOUP_LIBS) $(JSON_LIBS) $(APPSTREAM_GLIB_LIBS) \
libglnx.la libflatpak-common.la
flatpak_CFLAGS = $(AM_CFLAGS) $(BASE_CFLAGS) $(OSTREE_CFLAGS) $(SOUP_CFLAGS) $(JSON_CFLAGS) $(APPSTREAM_GLIB_CFLAGS) \
-DLOCALEDIR=\"$(localedir)\"

View File

@ -0,0 +1,260 @@
/*
* Copyright © 2017 Patrick Griffis
*
* This program 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, see <http://www.gnu.org/licenses/>.
*
* Authors:
* Patrick Griffis <tingping@tingping.se>
*/
#include "config.h"
#include <glib/gi18n.h>
#include <appstream-glib.h>
#include "flatpak-builtins.h"
#include "flatpak-dir.h"
#include "flatpak-table-printer.h"
#include "flatpak-utils.h"
static gboolean opt_user;
static gboolean opt_system;
static GOptionEntry options[] = {
{ "user", 0, 0, G_OPTION_ARG_NONE, &opt_user, N_("Search only user installations"), NULL },
{ "system", 0, 0, G_OPTION_ARG_NONE, &opt_system, N_("Search only system-wide installations"), NULL },
{ NULL }
};
static GPtrArray *
get_remote_stores (GPtrArray *dirs, GCancellable *cancellable)
{
GError *error = NULL;
GPtrArray *ret = g_ptr_array_new_with_free_func (g_object_unref);
for (guint i = 0; i < dirs->len; ++i)
{
FlatpakDir *dir = g_ptr_array_index (dirs, i);
g_autofree char *install_path = g_file_get_path (flatpak_dir_get_path (dir));
g_auto(GStrv) remotes = flatpak_dir_list_enumerated_remotes (dir, cancellable, &error);
if (error)
{
g_warning ("%s", error->message);
g_clear_error (&error);
continue;
}
else if (remotes == NULL)
continue;
for (guint j = 0; remotes[j]; ++j)
{
g_autofree char *appstream_path = g_build_filename (install_path, "appstream", remotes[j],
flatpak_get_arch(), "active", "appstream.xml.gz",
NULL);
g_autoptr(GFile) appstream_file = g_file_new_for_path (appstream_path);
g_autoptr(AsStore) store = as_store_new ();
// We want to see multiple versions/branches of same app-id's, e.g. org.gnome.Platform
as_store_set_add_flags (store, as_store_get_add_flags (store) | AS_STORE_ADD_FLAG_USE_UNIQUE_ID);
as_store_from_file (store, appstream_file, NULL, cancellable, &error);
if (error)
{
// We want to ignore this error as it is harmless and valid
// NOTE: appstream-glib doesn't have granular file-not-found error
if (!g_str_has_suffix (error->message, "No such file or directory"))
g_warning ("%s", error->message);
g_clear_error (&error);
continue;
}
g_object_set_data_full (G_OBJECT(store), "remote-name", g_strdup(remotes[j]), g_free);
g_ptr_array_add (ret, g_steal_pointer (&store));
}
}
return ret;
}
typedef struct MatchResult {
AsApp *app;
GPtrArray *remotes;
guint score;
} MatchResult;
static void
match_result_free (MatchResult *result)
{
g_object_unref (result->app);
g_ptr_array_unref (result->remotes);
g_free (result);
}
static MatchResult *
match_result_new (AsApp *app, guint score)
{
MatchResult *result = g_new (MatchResult, 1);
result->app = g_object_ref (app);
result->remotes = g_ptr_array_new_with_free_func (g_free);
result->score = score;
return result;
}
static void
match_result_add_remote (MatchResult *self, const char *remote)
{
for (guint i = 0; i < self->remotes->len; ++i)
{
const char *remote_entry = g_ptr_array_index (self->remotes, i);
if (!strcmp (remote, remote_entry))
return;
}
g_ptr_array_add (self->remotes, g_strdup(remote));
}
static int
compare_by_score (MatchResult *a, MatchResult *b, gpointer user_data)
{
// Reverse order, higher score comes first
return (int)b->score - (int)a->score;
}
static int
compare_apps (MatchResult *a, AsApp *b)
{
return !as_app_equal (a->app, b);
}
static const char *
get_comment_localized (AsApp *app)
{
const char * const * languages = g_get_language_names ();
for (gsize i = 0; languages[i]; ++i)
{
const char *comment = as_app_get_comment (app, languages[i]);
if (comment != NULL)
return comment;
}
return NULL;
}
static void
print_app (MatchResult *res, FlatpakTablePrinter *printer)
{
AsRelease *release = as_app_get_release_default (res->app);
const char *version = release ? as_release_get_version (release) : NULL;
const char *id = as_app_get_id_filename (res->app);
const char *branch = as_app_get_branch (res->app);
flatpak_table_printer_add_column (printer, id);
flatpak_table_printer_add_column (printer, version);
flatpak_table_printer_add_column (printer, branch);
flatpak_table_printer_add_column (printer, g_ptr_array_index (res->remotes, 0));
for (guint i = 1; i < res->remotes->len; ++i)
flatpak_table_printer_append_with_comma (printer, g_ptr_array_index (res->remotes, i));
flatpak_table_printer_add_column (printer, get_comment_localized (res->app));
flatpak_table_printer_finish_row (printer);
}
gboolean
flatpak_builtin_search (int argc, char **argv, GCancellable *cancellable, GError **error)
{
g_autoptr(GPtrArray) dirs = g_ptr_array_new_with_free_func ((GDestroyNotify) g_object_unref);
g_autoptr(GOptionContext) context = g_option_context_new (_("TEXT - Search remote apps/runtimes for text"));
g_option_context_set_translation_domain (context, GETTEXT_PACKAGE);
if (!flatpak_option_context_parse (context, options, &argc, &argv, FLATPAK_BUILTIN_FLAG_NO_DIR,
NULL, cancellable, error))
return FALSE;
// Default: All system and user remotes
if (!opt_user && !opt_system)
opt_user = opt_system = TRUE;
if (opt_user)
g_ptr_array_add (dirs, flatpak_dir_get_user ());
if (opt_system)
g_ptr_array_add (dirs, flatpak_dir_get_system_default ());
if (argc < 2)
return usage_error (context, _("TEXT must be specified"), error);
const char *search_text = argv[1];
GSList *matches = NULL;
// We want a store for each remote so we keep the remote information
// as AsApp doesn't currently contain that information
g_autoptr(GPtrArray) remote_stores = get_remote_stores (dirs, cancellable);
for (guint j = 0; j < remote_stores->len; ++j)
{
AsStore *store = g_ptr_array_index (remote_stores, j);
GPtrArray *apps = as_store_get_apps (store);
for (guint i = 0; i < apps->len; ++i)
{
AsApp *app = g_ptr_array_index (apps, i);
const guint score = as_app_search_matches (app, search_text);
if (score == 0)
continue;
// Avoid duplicate entries, but show multiple remotes
GSList *list_entry = g_slist_find_custom (matches, app,
(GCompareFunc)compare_apps);
MatchResult *result = NULL;
if (list_entry != NULL)
result = list_entry->data;
else
{
result = match_result_new (app, score);
matches = g_slist_insert_sorted_with_data (matches, result,
(GCompareDataFunc)compare_by_score, NULL);
}
match_result_add_remote (result,
g_object_get_data (G_OBJECT(store), "remote-name"));
}
}
if (matches != NULL)
{
FlatpakTablePrinter *printer = flatpak_table_printer_new ();
int col = 0;
flatpak_table_printer_set_column_title (printer, col++, _("Application ID"));
flatpak_table_printer_set_column_title (printer, col++, _("Version"));
flatpak_table_printer_set_column_title (printer, col++, _("Branch"));
flatpak_table_printer_set_column_title (printer, col++, _("Remotes"));
flatpak_table_printer_set_column_title (printer, col++, _("Description"));
g_slist_foreach (matches, (GFunc)print_app, printer);
flatpak_table_printer_print (printer);
flatpak_table_printer_free (printer);
g_slist_free_full (matches, (GDestroyNotify)match_result_free);
}
else
{
g_print ("%s\n", _("No matches found"));
}
return TRUE;
}
gboolean
flatpak_complete_search (FlatpakCompletion *completion)
{
g_autoptr(GOptionContext) context = NULL;
context = g_option_context_new ("");
if (!flatpak_option_context_parse (context, options, &completion->argc, &completion->argv,
FLATPAK_BUILTIN_FLAG_NO_DIR, NULL, NULL, NULL))
return FALSE;
flatpak_complete_options (completion, options);
flatpak_complete_options (completion, global_entries);
return TRUE;
}

View File

@ -85,6 +85,7 @@ BUILTINPROTO (document_list)
BUILTINPROTO (override)
BUILTINPROTO (repo)
BUILTINPROTO (config)
BUILTINPROTO (search)
#undef BUILTINPROTO

View File

@ -66,6 +66,10 @@ static FlatpakCommand commands[] = {
{ "info", N_("Show info for installed app or runtime"), flatpak_builtin_info, flatpak_complete_info },
{ "config", N_("Configure flatpak"), flatpak_builtin_config, flatpak_complete_config },
/* translators: please keep the leading newline and space */
{ N_("\n Finding applications and runtimes") },
{ "search", N_("Search for remote apps/runtimes"), flatpak_builtin_search, flatpak_complete_search },
/* translators: please keep the leading newline and space */
{ N_("\n Running applications") },
{ "run", N_("Run an application"), flatpak_builtin_run, flatpak_complete_run },

View File

@ -8,7 +8,7 @@ dn=$(dirname $0)
pkg_install sudo which attr fuse \
libubsan libasan libtsan \
elfutils git gettext-devel \
elfutils git gettext-devel libappstream-glib-devel \
/usr/bin/{update-mime-database,update-desktop-database,gtk-update-icon-cache}
pkg_install_testing ostree-devel ostree
pkg_install_if_os fedora gjs parallel clang

View File

@ -226,6 +226,8 @@ PKG_CHECK_MODULES(FUSE, [fuse])
PKG_CHECK_MODULES(JSON, [json-glib-1.0])
PKG_CHECK_MODULES(APPSTREAM_GLIB, [appstream-glib])
AC_ARG_ENABLE([seccomp],
AC_HELP_STRING([--disable-seccomp],
[Disable seccomp]),

View File

@ -26,6 +26,7 @@ app/flatpak-builtins-override.c
app/flatpak-builtins-repo.c
app/flatpak-builtins-repo-update.c
app/flatpak-builtins-run.c
app/flatpak-builtins-search.c
app/flatpak-builtins-uninstall.c
app/flatpak-builtins-update.c
app/flatpak-main.c