512 lines
16 KiB
C
512 lines
16 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
|
|
* vi:set noexpandtab tabstop=8 shiftwidth=8:
|
|
*
|
|
* Copyright (C) 2024 Red Hat <www.redhat.com>
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
|
|
/**
|
|
* SECTION:gs-plugin-job-url-to-app
|
|
* @short_description: A plugin job on an app
|
|
*
|
|
* #GsPluginJobUrlToApp is a #GsPluginJob representing an operation to
|
|
* convert a URL into a #GsApp.
|
|
*
|
|
* This class is a wrapper around #GsPluginClass.url_to_app_async
|
|
* calling it for all loaded plugins, with #GsPluginJobRefine used to refine the
|
|
* results.
|
|
*
|
|
* Retrieve the resulting #GsAppList using
|
|
* gs_plugin_job_url_to_app_get_result_list().
|
|
*
|
|
* Since: 47
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <glib.h>
|
|
#include <glib-object.h>
|
|
|
|
#include "gs-app.h"
|
|
#include "gs-app-list-private.h"
|
|
#include "gs-enums.h"
|
|
#include "gs-plugin-job.h"
|
|
#include "gs-plugin-job-file-to-app.h"
|
|
#include "gs-plugin-job-private.h"
|
|
#include "gs-plugin-job-refine.h"
|
|
#include "gs-plugin-job-url-to-app.h"
|
|
#include "gs-plugin-types.h"
|
|
|
|
struct _GsPluginJobUrlToApp
|
|
{
|
|
GsPluginJob parent;
|
|
|
|
/* Input arguments. */
|
|
gchar *url; /* (owned) (not nullable) */
|
|
GsPluginUrlToAppFlags flags;
|
|
|
|
/* In-progress data. */
|
|
GError *saved_error; /* (owned) (nullable) */
|
|
guint n_pending_ops;
|
|
gboolean did_refine;
|
|
gboolean did_file_to_app;
|
|
GsAppList *in_progress_list; /* (owned) (nullable) */
|
|
|
|
/* Results. */
|
|
GsAppList *result_list; /* (owned) (nullable) */
|
|
};
|
|
|
|
G_DEFINE_TYPE (GsPluginJobUrlToApp, gs_plugin_job_url_to_app, GS_TYPE_PLUGIN_JOB)
|
|
|
|
typedef enum {
|
|
PROP_FLAGS = 1,
|
|
PROP_URL,
|
|
} GsPluginJobUrlToAppProperty;
|
|
|
|
static GParamSpec *props[PROP_URL + 1] = { NULL, };
|
|
|
|
static void
|
|
gs_plugin_job_url_to_app_dispose (GObject *object)
|
|
{
|
|
GsPluginJobUrlToApp *self = GS_PLUGIN_JOB_URL_TO_APP (object);
|
|
|
|
g_assert (self->saved_error == NULL);
|
|
g_assert (self->n_pending_ops == 0);
|
|
|
|
g_clear_pointer (&self->url, g_free);
|
|
g_clear_object (&self->result_list);
|
|
g_clear_object (&self->in_progress_list);
|
|
|
|
G_OBJECT_CLASS (gs_plugin_job_url_to_app_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gs_plugin_job_url_to_app_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GsPluginJobUrlToApp *self = GS_PLUGIN_JOB_URL_TO_APP (object);
|
|
|
|
switch ((GsPluginJobUrlToAppProperty) prop_id) {
|
|
case PROP_FLAGS:
|
|
g_value_set_flags (value, self->flags);
|
|
break;
|
|
case PROP_URL:
|
|
g_value_set_string (value, self->url);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gs_plugin_job_url_to_app_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GsPluginJobUrlToApp *self = GS_PLUGIN_JOB_URL_TO_APP (object);
|
|
|
|
switch ((GsPluginJobUrlToAppProperty) prop_id) {
|
|
case PROP_FLAGS:
|
|
/* Construct only. */
|
|
g_assert (self->flags == 0);
|
|
self->flags = g_value_get_flags (value);
|
|
g_object_notify_by_pspec (object, props[prop_id]);
|
|
break;
|
|
case PROP_URL:
|
|
/* Construct only. */
|
|
g_assert (self->url == NULL);
|
|
self->url = g_value_dup_string (value);
|
|
g_assert (self->url != NULL);
|
|
g_object_notify_by_pspec (object, props[prop_id]);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void plugin_app_func_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data);
|
|
static void finish_op (GTask *task,
|
|
GsAppList *list,
|
|
GError *error);
|
|
static void file_to_app_job_finished_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data);
|
|
static void finish_file_to_app_op (GTask *task,
|
|
GsAppList *list,
|
|
GError *error);
|
|
static void refine_job_finished_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data);
|
|
static void finish_refine_op (GTask *task,
|
|
GsAppList *list,
|
|
GError *error);
|
|
static gboolean gs_plugin_job_url_to_app_is_valid_filter (GsApp *app,
|
|
gpointer user_data);
|
|
|
|
static void
|
|
gs_plugin_job_url_to_app_run_async (GsPluginJob *job,
|
|
GsPluginLoader *plugin_loader,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
GsPluginJobUrlToApp *self = GS_PLUGIN_JOB_URL_TO_APP (job);
|
|
g_autoptr(GTask) task = NULL;
|
|
GPtrArray *plugins; /* (element-type GsPlugin) */
|
|
gboolean anything_ran = FALSE;
|
|
g_autoptr(GError) local_error = NULL;
|
|
|
|
task = g_task_new (job, cancellable, callback, user_data);
|
|
g_task_set_source_tag (task, gs_plugin_job_url_to_app_run_async);
|
|
g_task_set_task_data (task, g_object_ref (plugin_loader), (GDestroyNotify) g_object_unref);
|
|
|
|
/* run each plugin, keeping a counter of pending operations which is
|
|
* initialised to 1 until all the operations are started */
|
|
self->n_pending_ops = 1;
|
|
plugins = gs_plugin_loader_get_plugins (plugin_loader);
|
|
|
|
for (guint i = 0; i < plugins->len; i++) {
|
|
GsPlugin *plugin = g_ptr_array_index (plugins, i);
|
|
GsPluginClass *plugin_class = GS_PLUGIN_GET_CLASS (plugin);
|
|
|
|
if (!gs_plugin_get_enabled (plugin))
|
|
continue;
|
|
if (plugin_class->url_to_app_async == NULL)
|
|
continue;
|
|
|
|
/* at least one plugin supports this vfunc */
|
|
anything_ran = TRUE;
|
|
|
|
/* Handle cancellation */
|
|
if (g_cancellable_set_error_if_cancelled (cancellable, &local_error))
|
|
break;
|
|
|
|
/* run the plugin */
|
|
self->n_pending_ops++;
|
|
plugin_class->url_to_app_async (plugin, self->url, self->flags, cancellable, plugin_app_func_cb, g_object_ref (task));
|
|
}
|
|
|
|
if (!anything_ran)
|
|
g_debug ("no plugin could handle url-to-app operation");
|
|
|
|
finish_op (task, NULL, g_steal_pointer (&local_error));
|
|
}
|
|
|
|
static void
|
|
plugin_app_func_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
GsPlugin *plugin = GS_PLUGIN (source_object);
|
|
GsPluginClass *plugin_class = GS_PLUGIN_GET_CLASS (plugin);
|
|
g_autoptr(GTask) task = G_TASK (g_steal_pointer (&user_data));
|
|
g_autoptr(GsAppList) list = NULL;
|
|
g_autoptr(GError) local_error = NULL;
|
|
|
|
list = plugin_class->url_to_app_finish (plugin, result, &local_error);
|
|
gs_plugin_status_update (plugin, NULL, GS_PLUGIN_STATUS_FINISHED);
|
|
|
|
g_assert (list != NULL || local_error != NULL);
|
|
|
|
finish_op (task, list, g_steal_pointer (&local_error));
|
|
}
|
|
|
|
/* @error is (transfer full) if non-%NULL */
|
|
static void
|
|
finish_op (GTask *task,
|
|
GsAppList *list,
|
|
GError *error)
|
|
{
|
|
GsPluginJobUrlToApp *self = g_task_get_source_object (task);
|
|
GsPluginLoader *plugin_loader = g_task_get_task_data (task);
|
|
GCancellable *cancellable = g_task_get_cancellable (task);
|
|
g_autoptr(GError) error_owned = g_steal_pointer (&error);
|
|
g_autofree gchar *job_debug = NULL;
|
|
|
|
if (error_owned != NULL && self->saved_error == NULL)
|
|
self->saved_error = g_steal_pointer (&error_owned);
|
|
else if (error_owned != NULL)
|
|
g_debug ("Additional error while url-to-app: %s", error_owned->message);
|
|
|
|
g_assert (self->n_pending_ops > 0);
|
|
self->n_pending_ops--;
|
|
|
|
if (list != NULL) {
|
|
if (self->in_progress_list == NULL)
|
|
self->in_progress_list = gs_app_list_new ();
|
|
gs_app_list_add_list (self->in_progress_list, list);
|
|
}
|
|
|
|
if (self->n_pending_ops > 0)
|
|
return;
|
|
|
|
/* Once all the url-to-app operations are complete, try file-to-app if
|
|
* they produced no results and the URI uses the `file` scheme. */
|
|
if ((self->in_progress_list == NULL || gs_app_list_length (self->in_progress_list) == 0) &&
|
|
g_ascii_strncasecmp (self->url, "file:", strlen ("file:")) == 0) {
|
|
g_autoptr(GFile) file = g_file_new_for_uri (self->url);
|
|
g_autoptr(GsPluginJob) file_to_app_job = NULL;
|
|
|
|
file_to_app_job = gs_plugin_job_file_to_app_new (file,
|
|
(self->flags & GS_PLUGIN_URL_TO_APP_FLAGS_INTERACTIVE) != 0 ?
|
|
GS_PLUGIN_FILE_TO_APP_FLAGS_INTERACTIVE :
|
|
GS_PLUGIN_FILE_TO_APP_FLAGS_NONE);
|
|
gs_plugin_loader_job_process_async (plugin_loader, file_to_app_job, cancellable,
|
|
file_to_app_job_finished_cb, g_object_ref (task));
|
|
return;
|
|
}
|
|
|
|
/* Fall through without calling file-to-app. */
|
|
finish_file_to_app_op (task, self->in_progress_list, NULL);
|
|
}
|
|
|
|
static void
|
|
file_to_app_job_finished_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
g_autoptr(GTask) task = G_TASK (g_steal_pointer (&user_data));
|
|
g_autoptr(GsAppList) list = NULL;
|
|
g_autoptr(GError) local_error = NULL;
|
|
|
|
list = gs_plugin_loader_job_process_finish (GS_PLUGIN_LOADER (source_object), result, &local_error);
|
|
g_prefix_error_literal (&local_error, "Failed to file-to-app from file: URL:");
|
|
|
|
finish_file_to_app_op (task, list, g_steal_pointer (&local_error));
|
|
}
|
|
|
|
/* @error is (transfer full) if non-%NULL */
|
|
static void
|
|
finish_file_to_app_op (GTask *task,
|
|
GsAppList *list,
|
|
GError *error)
|
|
{
|
|
GsPluginJobUrlToApp *self = g_task_get_source_object (task);
|
|
GsPluginLoader *plugin_loader = g_task_get_task_data (task);
|
|
GCancellable *cancellable = g_task_get_cancellable (task);
|
|
g_autoptr(GError) error_owned = g_steal_pointer (&error);
|
|
|
|
if (error_owned != NULL && self->saved_error == NULL)
|
|
self->saved_error = g_steal_pointer (&error_owned);
|
|
else if (error_owned != NULL)
|
|
g_debug ("Additional error while converting URL to app: %s", error_owned->message);
|
|
|
|
g_set_object (&self->in_progress_list, list);
|
|
|
|
/* Now refine the results. */
|
|
if (self->in_progress_list != NULL) {
|
|
GsPluginRefineFlags refine_flags = gs_plugin_job_get_refine_flags (GS_PLUGIN_JOB (self));
|
|
|
|
if (refine_flags != GS_PLUGIN_REFINE_FLAGS_NONE) {
|
|
g_autoptr(GsPluginJob) refine_job = NULL;
|
|
|
|
/* to not have filtered out repositories */
|
|
refine_flags |= GS_PLUGIN_REFINE_FLAGS_DISABLE_FILTERING;
|
|
|
|
refine_job = gs_plugin_job_refine_new (self->in_progress_list, refine_flags);
|
|
gs_plugin_loader_job_process_async (plugin_loader, refine_job, cancellable,
|
|
refine_job_finished_cb, g_object_ref (task));
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Fall through without refining. */
|
|
finish_refine_op (task, self->in_progress_list, NULL);
|
|
}
|
|
|
|
static void
|
|
refine_job_finished_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
g_autoptr(GTask) task = G_TASK (g_steal_pointer (&user_data));
|
|
g_autoptr(GsAppList) list = NULL;
|
|
g_autoptr(GError) local_error = NULL;
|
|
|
|
list = gs_plugin_loader_job_process_finish (GS_PLUGIN_LOADER (source_object), result, &local_error);
|
|
g_prefix_error_literal (&local_error, "Failed to refine url-to-app apps:");
|
|
|
|
finish_refine_op (task, list, g_steal_pointer (&local_error));
|
|
}
|
|
|
|
/* @error is (transfer full) if non-%NULL */
|
|
static void
|
|
finish_refine_op (GTask *task,
|
|
GsAppList *list,
|
|
GError *error)
|
|
{
|
|
GsPluginJobUrlToApp *self = g_task_get_source_object (task);
|
|
g_autoptr(GError) error_owned = g_steal_pointer (&error);
|
|
g_autofree gchar *job_debug = NULL;
|
|
|
|
if (error_owned != NULL && self->saved_error == NULL)
|
|
self->saved_error = g_steal_pointer (&error_owned);
|
|
else if (error_owned != NULL)
|
|
g_debug ("Additional error while converting URL to app: %s", error_owned->message);
|
|
|
|
g_clear_object (&self->result_list);
|
|
self->result_list = (list != NULL) ? g_object_ref (list) : NULL;
|
|
|
|
if (self->result_list != NULL)
|
|
gs_app_list_filter (self->result_list, gs_plugin_job_url_to_app_is_valid_filter, self);
|
|
|
|
/* only allow one result */
|
|
if (self->saved_error == NULL) {
|
|
if (self->result_list == NULL ||
|
|
gs_app_list_length (self->result_list) == 0) {
|
|
g_autofree gchar *str = gs_plugin_job_to_string (GS_PLUGIN_JOB (self));
|
|
g_set_error (&self->saved_error,
|
|
GS_PLUGIN_ERROR,
|
|
GS_PLUGIN_ERROR_NOT_SUPPORTED,
|
|
"no application was created for %s", str);
|
|
} else if (gs_app_list_length (self->result_list) > 1) {
|
|
g_autofree gchar *str = gs_plugin_job_to_string (GS_PLUGIN_JOB (self));
|
|
g_debug ("expected one, but received %u apps for %s", gs_app_list_length (self->result_list), str);
|
|
}
|
|
|
|
/* Ensure the right icon is set on all the apps. */
|
|
if (self->result_list != NULL) {
|
|
for (guint i = 0; i < gs_app_list_length (self->result_list); i++) {
|
|
GsApp *app = gs_app_list_index (self->result_list, i);
|
|
if (!gs_app_has_icons (app)) {
|
|
g_autoptr(GIcon) ic = NULL;
|
|
const gchar *icon_name;
|
|
if (gs_app_has_quirk (app, GS_APP_QUIRK_HAS_SOURCE))
|
|
icon_name = "x-package-repository";
|
|
else
|
|
icon_name = "system-component-application";
|
|
ic = g_themed_icon_new (icon_name);
|
|
gs_app_add_icon (app, ic);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* show elapsed time */
|
|
job_debug = gs_plugin_job_to_string (GS_PLUGIN_JOB (self));
|
|
g_debug ("%s", job_debug);
|
|
|
|
if (self->saved_error != NULL)
|
|
g_task_return_error (task, g_steal_pointer (&self->saved_error));
|
|
else
|
|
g_task_return_boolean (task, TRUE);
|
|
g_signal_emit_by_name (G_OBJECT (self), "completed");
|
|
}
|
|
|
|
static gboolean
|
|
gs_plugin_job_url_to_app_is_valid_filter (GsApp *app,
|
|
gpointer user_data)
|
|
{
|
|
GsPluginJob *plugin_job = user_data;
|
|
|
|
return gs_plugin_loader_app_is_valid (app, gs_plugin_job_get_refine_flags (plugin_job));
|
|
}
|
|
|
|
static gboolean
|
|
gs_plugin_job_url_to_app_run_finish (GsPluginJob *self,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
return g_task_propagate_boolean (G_TASK (result), error);
|
|
}
|
|
|
|
static void
|
|
gs_plugin_job_url_to_app_class_init (GsPluginJobUrlToAppClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GsPluginJobClass *job_class = GS_PLUGIN_JOB_CLASS (klass);
|
|
|
|
object_class->dispose = gs_plugin_job_url_to_app_dispose;
|
|
object_class->get_property = gs_plugin_job_url_to_app_get_property;
|
|
object_class->set_property = gs_plugin_job_url_to_app_set_property;
|
|
|
|
job_class->run_async = gs_plugin_job_url_to_app_run_async;
|
|
job_class->run_finish = gs_plugin_job_url_to_app_run_finish;
|
|
|
|
/**
|
|
* GsPluginJobUrlToApp:url: (not nullable)
|
|
*
|
|
* A URL to convert to a #GsApp.
|
|
*
|
|
* Since: 47
|
|
*/
|
|
props[PROP_URL] =
|
|
g_param_spec_string ("url", "URL",
|
|
"A URL to convert to a #GsApp.",
|
|
NULL,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
/**
|
|
* GsPluginJobUrlToApp:flags:
|
|
*
|
|
* Flags affecting how the operation runs.
|
|
*
|
|
* Since: 47
|
|
*/
|
|
props[PROP_FLAGS] =
|
|
g_param_spec_flags ("flags", "Flags",
|
|
"Flags affecting how the operation runs.",
|
|
GS_TYPE_PLUGIN_URL_TO_APP_FLAGS,
|
|
GS_PLUGIN_URL_TO_APP_FLAGS_NONE,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
g_object_class_install_properties (object_class, G_N_ELEMENTS (props), props);
|
|
}
|
|
|
|
static void
|
|
gs_plugin_job_url_to_app_init (GsPluginJobUrlToApp *self)
|
|
{
|
|
}
|
|
|
|
/**
|
|
* gs_plugin_job_url_to_app_new:
|
|
* @url: (not nullable): a URL to run the operation on
|
|
* @flags: flags affecting how the operation runs
|
|
*
|
|
* Create a new #GsPluginJobUrlToApp to convert the given @url.
|
|
*
|
|
* Returns: (transfer full): a new #GsPluginJobUrlToApp
|
|
* Since: 47
|
|
*/
|
|
GsPluginJob *
|
|
gs_plugin_job_url_to_app_new (const gchar *url,
|
|
GsPluginUrlToAppFlags flags)
|
|
{
|
|
g_return_val_if_fail (url != NULL && g_uri_is_valid (url, G_URI_FLAGS_NONE, NULL), NULL);
|
|
|
|
return g_object_new (GS_TYPE_PLUGIN_JOB_URL_TO_APP,
|
|
"url", url,
|
|
"flags", flags,
|
|
NULL);
|
|
}
|
|
|
|
/**
|
|
* gs_plugin_job_url_to_app_get_result_list:
|
|
* @self: a #GsPluginJobUrlToApp
|
|
*
|
|
* Get the list of apps converted from the given URL.
|
|
*
|
|
* If this is called before the job is complete, %NULL will be returned.
|
|
*
|
|
* Returns: (transfer none) (nullable): the job results, or %NULL on error
|
|
* or if called before the job has completed
|
|
*
|
|
* Since: 47
|
|
*/
|
|
GsAppList *
|
|
gs_plugin_job_url_to_app_get_result_list (GsPluginJobUrlToApp *self)
|
|
{
|
|
g_return_val_if_fail (GS_IS_PLUGIN_JOB_URL_TO_APP (self), NULL);
|
|
|
|
return self->result_list;
|
|
}
|