/* vi:set et sw=2 sts=2 cin cino=t0,f0,(0,{s,>2s,n-s,^-s,e-s: * Copyright © 2015 Red Hat, Inc * * 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 . * * Authors: * Alexander Larsson */ #include "config.h" #include #include #include #include #include "flatpak-dir-private.h" #include "flatpak-enum-types.h" #include "flatpak-error.h" #include "flatpak-installation-private.h" #include "flatpak-installation.h" #include "flatpak-installed-ref-private.h" #include "flatpak-instance-private.h" #include "flatpak-related-ref-private.h" #include "flatpak-progress-private.h" #include "flatpak-remote-private.h" #include "flatpak-remote-ref-private.h" #include "flatpak-run-private.h" #include "flatpak-transaction-private.h" #include "flatpak-utils-private.h" /** * SECTION:flatpak-installation * @Title: FlatpakInstallation * @Short_description: Installation information * @See_also: FlatpakTransaction * * FlatpakInstallation is the toplevel object that software installers * should use to operate on an flatpak applications. * * An FlatpakInstallation object provides information about an installation * location for flatpak applications. Typical installation locations are either * system-wide (in $prefix/var/lib/flatpak) or per-user (in ~/.local/share/flatpak). * * FlatpakInstallation can list configured remotes as well as installed application * and runtime references (in short: refs), and it can add, remove and modify remotes. * * FlatpakInstallation can also run, install, update and uninstall applications and * runtimes, but #FlatpakTransaction is a better, high-level API for these tasks. * * To get a list of all configured installations, use flatpak_get_system_installations(), * together with flatpak_installation_new_user(). * * The FlatpakInstallation API is threadsafe in the sense that it is safe to run two * operations at the same time, in different threads (or processes). */ typedef struct _FlatpakInstallationPrivate FlatpakInstallationPrivate; G_LOCK_DEFINE_STATIC (dir); struct _FlatpakInstallationPrivate { /* All raw access to this should be protected by the dir lock. The FlatpakDir object is mostly threadsafe (apart from pull transactions being a singleton on it), however we replace it during flatpak_installation_drop_caches(), so every user needs to keep its own reference alive until done. */ FlatpakDir *dir_unlocked; char *display_name; }; G_DEFINE_TYPE_WITH_PRIVATE (FlatpakInstallation, flatpak_installation, G_TYPE_OBJECT) enum { PROP_0, }; static void flatpak_installation_finalize (GObject *object) { FlatpakInstallation *self = FLATPAK_INSTALLATION (object); FlatpakInstallationPrivate *priv = flatpak_installation_get_instance_private (self); g_object_unref (priv->dir_unlocked); g_free (priv->display_name); G_OBJECT_CLASS (flatpak_installation_parent_class)->finalize (object); } static void flatpak_installation_class_init (FlatpakInstallationClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = flatpak_installation_finalize; /* Avoid weird recursive type initialization deadlocks from libsoup */ g_type_ensure (G_TYPE_SOCKET); } static void flatpak_installation_init (FlatpakInstallation *self) { } static FlatpakInstallation * flatpak_installation_new_steal_dir (FlatpakDir *dir, GCancellable *cancellable, GError **error) { FlatpakInstallation *self; FlatpakInstallationPrivate *priv; if (!flatpak_dir_maybe_ensure_repo (dir, NULL, error)) { g_object_unref (dir); return NULL; } self = g_object_new (FLATPAK_TYPE_INSTALLATION, NULL); priv = flatpak_installation_get_instance_private (self); priv->dir_unlocked = dir; return self; } FlatpakInstallation * flatpak_installation_new_for_dir (FlatpakDir *dir, GCancellable *cancellable, GError **error) { return flatpak_installation_new_steal_dir (g_object_ref (dir), cancellable, error); } /** * flatpak_installation_set_no_interaction: * @self: a #FlatpakInstallation * @no_interaction: Whether to disallow interactive authorization for operations * * This method can be used to prevent interactive authorization dialogs to appear * for operations on @self. This is useful for background operations that are not * directly triggered by a user action. * * By default, interaction is allowed. * * Since: 1.1.1 */ void flatpak_installation_set_no_interaction (FlatpakInstallation *self, gboolean no_interaction) { FlatpakInstallationPrivate *priv = flatpak_installation_get_instance_private (self); flatpak_dir_set_no_interaction (priv->dir_unlocked, no_interaction); } /** * flatpak_installation_get_no_interaction: * @self: a #FlatpakTransaction * * Returns the value set with flatpak_installation_set_no_interaction(). * * Returns: %TRUE if interactive authorization dialogs are not allowed * * Since: 1.1.1 */ gboolean flatpak_installation_get_no_interaction (FlatpakInstallation *self) { FlatpakInstallationPrivate *priv = flatpak_installation_get_instance_private (self); return flatpak_dir_get_no_interaction (priv->dir_unlocked); } /** * flatpak_get_default_arch: * * Returns the canonical name for the arch of the current machine. * * Returns: an arch string */ const char * flatpak_get_default_arch (void) { return flatpak_get_arch (); } /** * flatpak_get_supported_arches: * * Returns the canonical names for the arches that are supported (i.e. can run) * on the current machine, in order of priority (default is first). * * Returns: a zero terminated array of arch strings */ const char * const * flatpak_get_supported_arches (void) { return (const char * const *) flatpak_get_arches (); } /** * flatpak_get_system_installations: * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists the system installations according to the current configuration and current * availability (e.g. doesn't return a configured installation if not reachable). * * Returns: (transfer container) (element-type FlatpakInstallation): a GPtrArray of * #FlatpakInstallation instances * * Since: 0.8 */ GPtrArray * flatpak_get_system_installations (GCancellable *cancellable, GError **error) { g_autoptr(GPtrArray) system_dirs = NULL; g_autoptr(GPtrArray) installs = NULL; GPtrArray *ret = NULL; int i; system_dirs = flatpak_dir_get_system_list (cancellable, error); if (system_dirs == NULL) goto out; installs = g_ptr_array_new_with_free_func ((GDestroyNotify) g_object_unref); for (i = 0; i < system_dirs->len; i++) { g_autoptr(GError) local_error = NULL; FlatpakDir *install_dir = g_ptr_array_index (system_dirs, i); g_autoptr(FlatpakInstallation) installation = NULL; installation = flatpak_installation_new_for_dir (install_dir, cancellable, &local_error); if (installation != NULL) g_ptr_array_add (installs, g_steal_pointer (&installation)); else { /* Warn about the problem and continue without listing this installation. */ g_autofree char *dir_name = flatpak_dir_get_name (install_dir); g_warning ("Unable to create FlatpakInstallation for %s: %s", dir_name, local_error->message); } } if (installs->len == 0) { g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND, "No system installations found"); } ret = g_steal_pointer (&installs); out: return ret; } /** * flatpak_installation_new_system: * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Creates a new #FlatpakInstallation for the default system-wide installation. * * Returns: (transfer full): a new #FlatpakInstallation */ FlatpakInstallation * flatpak_installation_new_system (GCancellable *cancellable, GError **error) { return flatpak_installation_new_steal_dir (flatpak_dir_get_system_default (), cancellable, error); } /** * flatpak_installation_new_system_with_id: * @id: (nullable): the ID of the system-wide installation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Creates a new #FlatpakInstallation for the system-wide installation @id. * * Returns: (transfer full): a new #FlatpakInstallation * * Since: 0.8 */ FlatpakInstallation * flatpak_installation_new_system_with_id (const char *id, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) install_dir = NULL; g_autoptr(FlatpakInstallation) installation = NULL; g_autoptr(GError) local_error = NULL; install_dir = flatpak_dir_get_system_by_id (id, cancellable, error); if (install_dir == NULL) return NULL; installation = flatpak_installation_new_for_dir (install_dir, cancellable, &local_error); if (installation == NULL) { g_info ("Error creating Flatpak installation: %s", local_error->message); g_propagate_error (error, g_steal_pointer (&local_error)); } g_info ("Found Flatpak installation for '%s'", id); return g_steal_pointer (&installation); } /** * flatpak_installation_new_user: * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Creates a new #FlatpakInstallation for the per-user installation. * * Returns: (transfer full): a new #FlatpakInstallation */ FlatpakInstallation * flatpak_installation_new_user (GCancellable *cancellable, GError **error) { return flatpak_installation_new_steal_dir (flatpak_dir_get_user (), cancellable, error); } /** * flatpak_installation_new_for_path: * @path: a #GFile * @user: whether this is a user-specific location * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Creates a new #FlatpakInstallation for the installation at the given @path. * * Returns: (transfer full): a new #FlatpakInstallation */ FlatpakInstallation * flatpak_installation_new_for_path (GFile *path, gboolean user, GCancellable *cancellable, GError **error) { return flatpak_installation_new_steal_dir (flatpak_dir_new (path, user), cancellable, error); } static FlatpakDir * _flatpak_installation_get_dir (FlatpakInstallation *self, gboolean ensure_repo, GError **error) { FlatpakInstallationPrivate *priv = flatpak_installation_get_instance_private (self); FlatpakDir *dir; G_LOCK (dir); if (ensure_repo && flatpak_dir_get_repo (priv->dir_unlocked) == NULL) { if (!flatpak_dir_ensure_repo (priv->dir_unlocked, NULL, error)) { dir = NULL; goto out; } } dir = g_object_ref (priv->dir_unlocked); out: G_UNLOCK (dir); return dir; } FlatpakDir * flatpak_installation_get_dir (FlatpakInstallation *self, GError **error) { return _flatpak_installation_get_dir (self, TRUE, error); } static FlatpakDir * flatpak_installation_get_dir_maybe_no_repo (FlatpakInstallation *self) { return _flatpak_installation_get_dir (self, FALSE, NULL); } FlatpakDir * flatpak_installation_clone_dir_noensure (FlatpakInstallation *self) { g_autoptr(FlatpakDir) dir_clone = NULL; g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir_maybe_no_repo (self); /* Pull, prune, etc are not threadsafe, so we work on a copy */ dir_clone = flatpak_dir_clone (dir); return g_steal_pointer (&dir_clone); } FlatpakDir * flatpak_installation_clone_dir (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir_clone = NULL; g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; /* Pull, prune, etc are not threadsafe, so we work on a copy */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return NULL; return g_steal_pointer (&dir_clone); } /** * flatpak_installation_drop_caches: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Drops all internal (in-memory) caches. For instance, this may be needed to pick up new or changed * remotes configured outside this installation instance. * * Returns: %TRUE on success, %FALSE on error */ gboolean flatpak_installation_drop_caches (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { FlatpakInstallationPrivate *priv = flatpak_installation_get_instance_private (self); FlatpakDir *clone, *old; gboolean res = FALSE; G_LOCK (dir); old = priv->dir_unlocked; clone = flatpak_dir_clone (priv->dir_unlocked); if (flatpak_dir_maybe_ensure_repo (clone, cancellable, error)) { priv->dir_unlocked = clone; g_object_unref (old); res = TRUE; } G_UNLOCK (dir); return res; } /** * flatpak_installation_get_is_user: * @self: a #FlatpakInstallation * * Returns whether the installation is for a user-specific location. * * Returns: %TRUE if @self is a per-user installation */ gboolean flatpak_installation_get_is_user (FlatpakInstallation *self) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); return flatpak_dir_is_user (dir); } /** * flatpak_installation_get_path: * @self: a #FlatpakInstallation * * Returns the installation location for @self. * * Returns: (transfer full): an #GFile */ GFile * flatpak_installation_get_path (FlatpakInstallation *self) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); return g_object_ref (flatpak_dir_get_path (dir)); } /** * flatpak_installation_get_id: * @self: a #FlatpakInstallation * * Returns the ID of the installation for @self. * * The ID for the default system installation is "default". * The ID for the user installation is "user". * * Returns: (transfer none): a string with the installation's ID * * Since: 0.8 */ const char * flatpak_installation_get_id (FlatpakInstallation *self) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); return flatpak_dir_get_id (dir); } /** * flatpak_installation_get_display_name: * @self: a #FlatpakInstallation * * Returns the display name of the installation for @self. * * Note that this function may return %NULL if the installation * does not have a display name. * * Returns: (transfer none): a string with the installation's display name * * Since: 0.8 */ const char * flatpak_installation_get_display_name (FlatpakInstallation *self) { FlatpakInstallationPrivate *priv = flatpak_installation_get_instance_private (self); g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); if (priv->display_name == NULL) priv->display_name = flatpak_dir_get_display_name (dir); return (const char *) priv->display_name; } /** * flatpak_installation_get_priority: * @self: a #FlatpakInstallation * * Returns the numeric priority of the installation for @self. * * Returns: an integer with the configured priority value * * Since: 0.8 */ gint flatpak_installation_get_priority (FlatpakInstallation *self) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); return flatpak_dir_get_priority (dir); } /** * flatpak_installation_get_storage_type: * @self: a #FlatpakInstallation * * Returns the type of storage of the installation for @self. * * Returns: a #FlatpakStorageType * * Since: 0.8 */FlatpakStorageType flatpak_installation_get_storage_type (FlatpakInstallation *self) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); switch (flatpak_dir_get_storage_type (dir)) { case FLATPAK_DIR_STORAGE_TYPE_HARD_DISK: return FLATPAK_STORAGE_TYPE_HARD_DISK; case FLATPAK_DIR_STORAGE_TYPE_SDCARD: return FLATPAK_STORAGE_TYPE_SDCARD; case FLATPAK_DIR_STORAGE_TYPE_MMC: return FLATPAK_STORAGE_TYPE_MMC; case FLATPAK_DIR_STORAGE_TYPE_NETWORK: return FLATPAK_STORAGE_TYPE_NETWORK; default: return FLATPAK_STORAGE_TYPE_DEFAULT; } return FLATPAK_STORAGE_TYPE_DEFAULT; } /** * flatpak_installation_launch: * @self: a #FlatpakInstallation * @name: name of the app to launch * @arch: (nullable): which architecture to launch (default: current architecture) * @branch: (nullable): which branch of the application (default: "master") * @commit: (nullable): the commit of @branch to launch * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Launch an installed application. * * You can use flatpak_installation_get_installed_ref() or * flatpak_installation_get_current_installed_app() to find out what builds * are available, in order to get a value for @commit. * * Returns: %TRUE, unless an error occurred */ gboolean flatpak_installation_launch (FlatpakInstallation *self, const char *name, const char *arch, const char *branch, const char *commit, GCancellable *cancellable, GError **error) { return flatpak_installation_launch_full (self, FLATPAK_LAUNCH_FLAGS_NONE, name, arch, branch, commit, NULL, cancellable, error); } /** * flatpak_installation_launch_full: * @self: a #FlatpakInstallation * @flags: set of #FlatpakLaunchFlags * @name: name of the app to launch * @arch: (nullable): which architecture to launch (default: current architecture) * @branch: (nullable): which branch of the application (default: "master") * @commit: (nullable): the commit of @branch to launch * @instance_out: (nullable): return location for a #FlatpakInstance * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Launch an installed application. * * You can use flatpak_installation_get_installed_ref() or * flatpak_installation_get_current_installed_app() to find out what builds * are available, in order to get a value for @commit. * * Compared to flatpak_installation_launch(), this function returns a #FlatpakInstance * that can be used to get information about the running instance. You can also use * it to wait for the instance to be done with g_child_watch_add() if you pass the * #FLATPAK_LAUNCH_FLAGS_DO_NOT_REAP flag. * * Returns: %TRUE, unless an error occurred * * Since: 1.1 */ gboolean flatpak_installation_launch_full (FlatpakInstallation *self, FlatpakLaunchFlags flags, const char *name, const char *arch, const char *branch, const char *commit, FlatpakInstance **instance_out, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDeploy) app_deploy = NULL; g_autoptr(FlatpakDecomposed) app_ref = NULL; g_autofree char *instance_dir = NULL; FlatpakRunFlags run_flags; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; app_ref = flatpak_decomposed_new_from_parts (FLATPAK_KINDS_APP, name, arch, branch, error); if (app_ref == NULL) return FALSE; app_deploy = flatpak_dir_load_deployed (dir, app_ref, commit, cancellable, error); if (app_deploy == NULL) return FALSE; run_flags = FLATPAK_RUN_FLAG_BACKGROUND; if (flags & FLATPAK_LAUNCH_FLAGS_DO_NOT_REAP) run_flags |= FLATPAK_RUN_FLAG_DO_NOT_REAP; if (!flatpak_run_app (app_ref, app_deploy, NULL, NULL, NULL, NULL, NULL, NULL, 0, run_flags, NULL, NULL, NULL, 0, -1, &instance_dir, cancellable, error)) return FALSE; if (instance_out) *instance_out = flatpak_instance_new (instance_dir); return TRUE; } static FlatpakInstalledRef * get_ref (FlatpakDir *dir, FlatpakDecomposed *ref, GCancellable *cancellable, GError **error) { const char *origin = NULL; const char *commit = NULL; const char *alt_id = NULL; g_autofree char *latest_alt_id = NULL; g_autoptr(GFile) deploy_dir = NULL; g_autoptr(GFile) deploy_subdir = NULL; g_autofree char *deploy_path = NULL; g_autofree char *latest_commit = NULL; g_autofree char *deploy_subdirname = NULL; g_autoptr(GBytes) deploy_data = NULL; g_autofree const char **subpaths = NULL; g_autofree char *collection_id = NULL; g_autoptr(GHashTable) content_rating = NULL; gboolean is_current = FALSE; guint64 installed_size = 0; deploy_data = flatpak_dir_get_deploy_data (dir, ref, FLATPAK_DEPLOY_VERSION_CURRENT, cancellable, error); if (deploy_data == NULL) return NULL; origin = flatpak_deploy_data_get_origin (deploy_data); commit = flatpak_deploy_data_get_commit (deploy_data); alt_id = flatpak_deploy_data_get_alt_id (deploy_data); subpaths = flatpak_deploy_data_get_subpaths (deploy_data); installed_size = flatpak_deploy_data_get_installed_size (deploy_data); deploy_dir = flatpak_dir_get_deploy_dir (dir, ref); deploy_subdirname = flatpak_dir_get_deploy_subdir (dir, commit, subpaths); deploy_subdir = g_file_get_child (deploy_dir, deploy_subdirname); deploy_path = g_file_get_path (deploy_subdir); if (flatpak_decomposed_is_app (ref)) { g_autofree char *id = flatpak_decomposed_dup_id (ref); g_autoptr(FlatpakDecomposed) current = flatpak_dir_current_ref (dir, id, cancellable); if (current && flatpak_decomposed_equal (ref, current)) is_current = TRUE; } latest_commit = flatpak_dir_read_latest (dir, origin, flatpak_decomposed_get_ref (ref), &latest_alt_id, NULL, NULL); collection_id = flatpak_dir_get_remote_collection_id (dir, origin); content_rating = flatpak_deploy_data_get_appdata_content_rating (deploy_data); return flatpak_installed_ref_new (ref, alt_id ? alt_id : commit, latest_alt_id ? latest_alt_id : latest_commit, origin, collection_id, subpaths, deploy_path, installed_size, is_current, flatpak_deploy_data_get_eol (deploy_data), flatpak_deploy_data_get_eol_rebase (deploy_data), flatpak_deploy_data_get_appdata_name (deploy_data), flatpak_deploy_data_get_appdata_summary (deploy_data), flatpak_deploy_data_get_appdata_version (deploy_data), flatpak_deploy_data_get_appdata_license (deploy_data), flatpak_deploy_data_get_appdata_content_rating_type (deploy_data), content_rating); } /** * flatpak_installation_get_installed_ref: * @self: a #FlatpakInstallation * @kind: whether this is an app or runtime * @name: name of the app/runtime to fetch * @arch: (nullable): which architecture to fetch (default: current architecture) * @branch: (nullable): which branch to fetch (default: "master") * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Returns information about an installed ref, such as the available builds, * its size, location, etc. * * Returns: (transfer full): an #FlatpakInstalledRef, or %NULL if an error occurred */ FlatpakInstalledRef * flatpak_installation_get_installed_ref (FlatpakInstallation *self, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GFile) deploy = NULL; g_autoptr(FlatpakDecomposed) ref = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; if (arch == NULL) arch = flatpak_get_arch (); ref = flatpak_decomposed_new_from_parts (flatpak_kinds_from_kind (kind), name, arch, branch, error); if (ref == NULL) return NULL; deploy = flatpak_dir_get_if_deployed (dir, ref, NULL, cancellable); if (deploy == NULL) { flatpak_fail_error (error, FLATPAK_ERROR_NOT_INSTALLED, _("Ref %s not installed"), flatpak_decomposed_get_ref (ref)); return NULL; } return get_ref (dir, ref, cancellable, error); } /** * flatpak_installation_get_current_installed_app: * @self: a #FlatpakInstallation * @name: the name of the app * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Get the last build of reference @name that was installed with * flatpak_installation_install(), or %NULL if the reference has * never been installed locally. * * Returns: (transfer full): an #FlatpakInstalledRef */ FlatpakInstalledRef * flatpak_installation_get_current_installed_app (FlatpakInstallation *self, const char *name, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GFile) deploy = NULL; g_autoptr(FlatpakDecomposed) current = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; current = flatpak_dir_current_ref (dir, name, cancellable); if (current) deploy = flatpak_dir_get_if_deployed (dir, current, NULL, cancellable); if (deploy == NULL) { flatpak_fail_error (error, FLATPAK_ERROR_NOT_INSTALLED, _("App %s not installed"), name); return NULL; } return get_ref (dir, current, cancellable, error); } /** * flatpak_installation_list_installed_refs: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists the installed references. * * Returns: (transfer container) (element-type FlatpakInstalledRef): a GPtrArray of * #FlatpakInstalledRef instances */ GPtrArray * flatpak_installation_list_installed_refs (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); g_autoptr(GPtrArray) decomposed_app = NULL; g_autoptr(GPtrArray) decomposed_runtime = NULL; g_autoptr(GPtrArray) refs = g_ptr_array_new_with_free_func (g_object_unref); int i; decomposed_app = flatpak_dir_list_refs (dir, FLATPAK_KINDS_APP, cancellable, error); if (decomposed_app == NULL) return NULL; for (i = 0; i < decomposed_app->len; i++) { g_autoptr(GError) local_error = NULL; FlatpakDecomposed *decomposed = g_ptr_array_index (decomposed_app, i); FlatpakInstalledRef *ref = get_ref (dir, decomposed, cancellable, &local_error); if (ref != NULL) g_ptr_array_add (refs, ref); else g_warning ("Unexpected failure getting ref for %s: %s", flatpak_decomposed_get_ref (decomposed), local_error->message); } decomposed_runtime = flatpak_dir_list_refs (dir,FLATPAK_KINDS_RUNTIME, cancellable, error); if (decomposed_runtime == NULL) return NULL; for (i = 0; i < decomposed_runtime->len; i++) { g_autoptr(GError) local_error = NULL; FlatpakDecomposed *decomposed = g_ptr_array_index (decomposed_runtime, i); FlatpakInstalledRef *ref = get_ref (dir, decomposed, cancellable, &local_error); if (ref != NULL) g_ptr_array_add (refs, ref); else g_warning ("Unexpected failure getting ref for %s: %s", flatpak_decomposed_get_ref (decomposed), local_error->message); } return g_steal_pointer (&refs); } /** * flatpak_installation_list_installed_refs_by_kind: * @self: a #FlatpakInstallation * @kind: the kind of installation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists the installed references of a specific kind. * * Returns: (transfer container) (element-type FlatpakInstalledRef): a GPtrArray of * #FlatpakInstalledRef instances */ GPtrArray * flatpak_installation_list_installed_refs_by_kind (FlatpakInstallation *self, FlatpakRefKind kind, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); g_autoptr(GPtrArray) all_decomposed = NULL; g_autoptr(GPtrArray) refs = g_ptr_array_new_with_free_func (g_object_unref); int i; all_decomposed = flatpak_dir_list_refs (dir, flatpak_kinds_from_kind (kind), cancellable, error); if (all_decomposed == NULL) return NULL; for (i = 0; i < all_decomposed->len; i++) { FlatpakDecomposed *decomposed = g_ptr_array_index (all_decomposed, i); g_autoptr(GError) local_error = NULL; FlatpakInstalledRef *ref = get_ref (dir, decomposed, cancellable, &local_error); if (ref != NULL) g_ptr_array_add (refs, ref); else g_warning ("Unexpected failure getting ref for %s: %s", flatpak_decomposed_get_ref (decomposed), local_error->message); } return g_steal_pointer (&refs); } static gboolean end_of_lifed_with_rebase (FlatpakTransaction *transaction, const char *remote, const char *ref, const char *reason, const char *rebased_to_ref, const char **previous_ids, GPtrArray **eol_rebase_refs) { if (rebased_to_ref == NULL || remote == NULL) return FALSE; /* No need to call flatpak_transaction_add_uninstall() and * flatpak_transaction_add_rebase() here since we only care about what needs * an update */ g_ptr_array_add (*eol_rebase_refs, g_strdup (ref)); return TRUE; } static gboolean transaction_ready (FlatpakTransaction *transaction, GHashTable **related_to_ops) { GList *ops = flatpak_transaction_get_operations (transaction); for (GList *l = ops; l != NULL; l = l->next) { FlatpakTransactionOperation *op = l->data; GPtrArray *op_related_to_ops = flatpak_transaction_operation_get_related_to_ops (op); /* (element-type FlatpakTransactionOperation) */ FlatpakTransactionOperationType type = flatpak_transaction_operation_get_operation_type (op); /* There may be an uninstall op if a runtime will now be considered * unused after the updates */ if (type == FLATPAK_TRANSACTION_OPERATION_UNINSTALL) { const char *ref = flatpak_transaction_operation_get_ref (op); g_info ("Update transaction wants to uninstall %s", ref); continue; } g_hash_table_insert (*related_to_ops, g_object_ref (op), op_related_to_ops ? g_ptr_array_ref (op_related_to_ops) : NULL); } g_list_free_full (ops, g_object_unref); /* Abort the transaction; we only wanted to know what it would do */ return FALSE; } static gint installed_ref_compare (gconstpointer _iref_a, gconstpointer _iref_b) { const FlatpakInstalledRef *iref_a = *(const FlatpakInstalledRef **)_iref_a; const FlatpakInstalledRef *iref_b = *(const FlatpakInstalledRef **)_iref_b; const char *ref_a = flatpak_ref_format_ref_cached (FLATPAK_REF (iref_a)); const char *ref_b = flatpak_ref_format_ref_cached (FLATPAK_REF (iref_b)); return strcmp (ref_a, ref_b); } /** * flatpak_installation_list_installed_refs_for_update: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists the installed apps and runtimes that have an update available, either * from the configured remote or locally available but not deployed (see * flatpak_transaction_set_no_deploy()). * * This also checks if any of #FlatpakInstalledRef has a missing #FlatpakRelatedRef * (which has `should-download` set to %TRUE) or runtime. If so, it adds the * ref to the returning #GPtrArray to pull in the #FlatpakRelatedRef or runtime * again via an update operation in #FlatpakTransaction. * * In case more than one app needs an update of the same runtime or extension, * this function will return all of those apps. * * Returns: (transfer container) (element-type FlatpakInstalledRef): a GPtrArray of * #FlatpakInstalledRef instances, or %NULL on error */ GPtrArray * flatpak_installation_list_installed_refs_for_update (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { g_autoptr(GPtrArray) installed_refs = NULL; /* (element-type FlatpakInstalledRef) */ g_autoptr(GHashTable) installed_refs_hash = NULL; /* (element-type utf8 FlatpakInstalledRef) */ g_autoptr(GPtrArray) installed_refs_for_update = NULL; /* (element-type FlatpakInstalledRef) */ g_autoptr(GHashTable) installed_refs_for_update_set = NULL; /* (element-type utf8) */ g_autoptr(GHashTable) related_to_ops = NULL; /* (element-type FlatpakTransactionOperation GPtrArray) */ g_autoptr(GPtrArray) eol_rebase_refs = NULL; /* (element-type utf8) */ g_autoptr(FlatpakTransaction) transaction = NULL; g_autoptr(GError) local_error = NULL; installed_refs = flatpak_installation_list_installed_refs (self, cancellable, error); if (installed_refs == NULL) return NULL; /* Here we use a FlatpakTransaction to determine what needs updating, and * abort it before actually doing the updates. This ensures we are consistent * with the CLI update command. */ transaction = flatpak_transaction_new_for_installation (self, cancellable, error); if (transaction == NULL) return NULL; /* CLI transactions set this. */ flatpak_transaction_add_default_dependency_sources (transaction); installed_refs_hash = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL); for (guint i = 0; i < installed_refs->len; i++) { FlatpakInstalledRef *installed_ref = g_ptr_array_index (installed_refs, i); const char *ref = flatpak_ref_format_ref_cached (FLATPAK_REF (installed_ref)); /* This hash table will be used later for efficient search */ g_hash_table_insert (installed_refs_hash, g_strdup (ref), installed_ref); if (flatpak_transaction_add_update (transaction, ref, NULL, NULL, &local_error)) continue; if (g_error_matches (local_error, FLATPAK_ERROR, FLATPAK_ERROR_REMOTE_NOT_FOUND)) { g_info ("%s: Unable to update %s: %s", G_STRFUNC, ref, local_error->message); g_clear_error (&local_error); } else { g_propagate_error (error, g_steal_pointer (&local_error)); return NULL; } } eol_rebase_refs = g_ptr_array_new_with_free_func (g_free); related_to_ops = g_hash_table_new_full (g_direct_hash, g_direct_equal, g_object_unref, null_safe_g_ptr_array_unref); g_signal_connect (transaction, "end-of-lifed-with-rebase", G_CALLBACK (end_of_lifed_with_rebase), &eol_rebase_refs); g_signal_connect (transaction, "ready-pre-auth", G_CALLBACK (transaction_ready), &related_to_ops); flatpak_transaction_run (transaction, cancellable, &local_error); g_assert (local_error != NULL); if (!g_error_matches (local_error, FLATPAK_ERROR, FLATPAK_ERROR_ABORTED)) { g_propagate_error (error, g_steal_pointer (&local_error)); return NULL; } g_clear_error (&local_error); installed_refs_for_update = g_ptr_array_new_with_free_func (g_object_unref); installed_refs_for_update_set = g_hash_table_new (g_str_hash, g_str_equal); /* For each ref that would be affected by the transaction, if it is * installed, add it to the list to be returned and otherwise add the ref * that caused it be added. We need to cover all of the following cases: * 1. For an app or runtime that has an update available, add it to the list * (including a locale extension which needs more subpaths downloaded). * 2. For an app or extension that has a missing runtime, add the * app/extension to the list. * 3. For an app that's missing a "should-download" related ref, add the app * to the list. */ GLNX_HASH_TABLE_FOREACH_KV (related_to_ops, FlatpakTransactionOperation *, op, GPtrArray *, op_related_to_ops) { const char *op_ref = flatpak_transaction_operation_get_ref (op); FlatpakInstalledRef *installed_ref; /* Here we use the existing installed_refs_hash instead of get_ref() * since staying in memory should be more efficient than disk I/O */ installed_ref = g_hash_table_lookup (installed_refs_hash, op_ref); if (installed_ref != NULL) { if (!g_hash_table_contains (installed_refs_for_update_set, op_ref)) { g_hash_table_add (installed_refs_for_update_set, (char *)op_ref); g_info ("%s: Installed ref %s needs update", G_STRFUNC, op_ref); g_ptr_array_add (installed_refs_for_update, g_object_ref (installed_ref)); } } else { for (gsize i = 0; op_related_to_ops != NULL && i < op_related_to_ops->len; i++) { FlatpakTransactionOperation *related_to_op = g_ptr_array_index (op_related_to_ops, i); const char *related_op_ref = flatpak_transaction_operation_get_ref (related_to_op); if (!g_hash_table_contains (installed_refs_for_update_set, related_op_ref)) { installed_ref = g_hash_table_lookup (installed_refs_hash, related_op_ref); if (installed_ref != NULL) { g_hash_table_add (installed_refs_for_update_set, (char *)related_op_ref); g_info ("%s: Installed ref %s needs update", G_STRFUNC, related_op_ref); g_ptr_array_add (installed_refs_for_update, g_object_ref (installed_ref)); } } } } /* Note: installed_ref could be NULL if for example op is installing a * related ref of a missing runtime. */ } /* Also handle any renames since those won't be in related_to_ops */ for (guint i = 0; i < eol_rebase_refs->len; i++) { const char *rebased_ref = g_ptr_array_index (eol_rebase_refs, i); FlatpakInstalledRef *installed_ref = g_hash_table_lookup (installed_refs_hash, rebased_ref); if (installed_ref != NULL) { if (!g_hash_table_contains (installed_refs_for_update_set, rebased_ref)) { g_hash_table_add (installed_refs_for_update_set, (char *)rebased_ref); g_info ("%s: Installed ref %s needs update", G_STRFUNC, rebased_ref); g_ptr_array_add (installed_refs_for_update, g_object_ref (installed_ref)); } } } /* Remove non-determinism for the sake of the unit tests */ g_ptr_array_sort (installed_refs_for_update, installed_ref_compare); return g_steal_pointer (&installed_refs_for_update); } /* Find all USB and LAN repositories which share the same collection ID as * @remote_name, and add a #FlatpakRemote to @remotes for each of them. The caller * must initialise @remotes. Returns %TRUE without modifying @remotes if the * given remote doesn’t have a collection ID configured or if the @dir doesn’t * have a repo. * * FIXME: If this were async, the parallelisation could be handled in the caller. */ /** * flatpak_installation_list_remotes_by_type: * @self: a #FlatpakInstallation * @types: (array length=num_types): an array of #FlatpakRemoteType * @num_types: the number of types provided in @types * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists only the remotes whose type is included in the @types argument. * * Since flatpak 1.7 this will never return any types except FLATPAK_REMOTE_TYPE_STATIC. * Equivalent functionallity to FLATPAK_REMOTE_TYPE_USB can be had by listing remote refs * with FLATPAK_QUERY_FLAGS_ONLY_SIDELOADED. * * Returns: (transfer container) (element-type FlatpakRemote): a GPtrArray of * #FlatpakRemote instances */ GPtrArray * flatpak_installation_list_remotes_by_type (FlatpakInstallation *self, const FlatpakRemoteType *types, gsize num_types, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); g_autoptr(FlatpakDir) dir_clone = NULL; g_auto(GStrv) remote_names = NULL; g_autoptr(GPtrArray) remotes = g_ptr_array_new_with_free_func (g_object_unref); const guint NUM_FLATPAK_REMOTE_TYPES = 3; gboolean types_filter[NUM_FLATPAK_REMOTE_TYPES]; gsize i; remote_names = flatpak_dir_list_remotes (dir, cancellable, error); if (remote_names == NULL) return NULL; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_maybe_ensure_repo (dir_clone, cancellable, error)) return NULL; /* If NULL or an empty array of types is passed then we list all types */ for (i = 0; i < NUM_FLATPAK_REMOTE_TYPES; ++i) { if (types != NULL && num_types != 0) types_filter[i] = FALSE; else types_filter[i] = TRUE; } for (i = 0; i < num_types; ++i) { g_return_val_if_fail (types[i] < NUM_FLATPAK_REMOTE_TYPES, NULL); types_filter[types[i]] = TRUE; } for (i = 0; remote_names[i] != NULL; ++i) { /* These days we only support static remotes */ if (types_filter[FLATPAK_REMOTE_TYPE_STATIC]) g_ptr_array_add (remotes, flatpak_remote_new_with_dir (remote_names[i], dir_clone)); } return g_steal_pointer (&remotes); } /** * flatpak_installation_list_remotes: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists the static remotes, in priority (highest first) order. For same * priority, an earlier added remote comes before a later added one. * * Returns: (transfer container) (element-type FlatpakRemote): a GPtrArray of * #FlatpakRemote instances */ GPtrArray * flatpak_installation_list_remotes (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { const FlatpakRemoteType types[] = { FLATPAK_REMOTE_TYPE_STATIC }; return flatpak_installation_list_remotes_by_type (self, types, 1, cancellable, error); } /** * flatpak_installation_modify_remote: * @self: a #FlatpakInstallation * @remote: the modified #FlatpakRemote * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Saves changes in the @remote object. * * Returns: %TRUE if the modifications have been committed successfully */ gboolean flatpak_installation_modify_remote (FlatpakInstallation *self, FlatpakRemote *remote, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); g_autoptr(FlatpakDir) dir_clone = NULL; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_maybe_ensure_repo (dir_clone, cancellable, error)) return FALSE; if (!flatpak_remote_commit (remote, dir_clone, cancellable, error)) return FALSE; /* Make sure we pick up the new config */ flatpak_installation_drop_caches (self, NULL, NULL); return TRUE; } /** * flatpak_installation_add_remote: * @self: a #FlatpakInstallation * @remote: the new #FlatpakRemote * @if_needed: if %TRUE, only add if it doesn't exists * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Adds a new @remote object to the set of remotes. This is similar * to flatpak_installation_modify_remote() for non-existing remote * names. However, if the named remote already exists then instead of * modifying it it fails with %FLATPAK_ERROR_ALREADY_INSTALLED, or if * @if_needed is true it silently succeeds without doing anything. * * As an exception to the last, if the local config has a filter defined, * but the new remote unsets the filter (for example, it comes from an * unfiltered .flatpakref via flatpak_remote_new_from_file()) the the local * remote filter gets reset. This is to allow the setup where there is a * default setup of a filtered remote, yet you can still use the standard * flatpakref file to get the full contents without getting two remotes. * * Returns: %TRUE if the modifications have been committed successfully * Since: 1.3.4 */ gboolean flatpak_installation_add_remote (FlatpakInstallation *self, FlatpakRemote *remote, gboolean if_needed, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); g_autoptr(FlatpakDir) dir_clone = NULL; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_maybe_ensure_repo (dir_clone, cancellable, error)) return FALSE; if (flatpak_dir_has_remote (dir, flatpak_remote_get_name (remote), NULL)) { if (!if_needed) return flatpak_fail_error (error, FLATPAK_ERROR_ALREADY_INSTALLED, _("Remote '%s' already exists"), flatpak_remote_get_name (remote)); if (!flatpak_remote_commit_filter (remote, dir_clone, cancellable, error)) return FALSE; return TRUE; } if (!flatpak_remote_commit (remote, dir_clone, cancellable, error)) return FALSE; /* Make sure we pick up the new config */ flatpak_installation_drop_caches (self, NULL, NULL); return TRUE; } /** * flatpak_installation_remove_remote: * @self: a #FlatpakInstallation * @name: the name of the remote to remove * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Removes the remote with the given name from the installation. * * Returns: %TRUE if the remote has been removed successfully */ gboolean flatpak_installation_remove_remote (FlatpakInstallation *self, const char *name, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return FALSE; if (!flatpak_dir_remove_remote (dir, FALSE, name, cancellable, error)) return FALSE; /* Make sure we pick up the new config */ flatpak_installation_drop_caches (self, NULL, NULL); return TRUE; } /** * flatpak_installation_set_config_sync: * @self: a #FlatpakInstallation * @key: the name of the key to set * @value: the new value, or %NULL to unset * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Set a global configuration option for the installation, currently * the only supported keys are `languages`, which is a semicolon-separated * list of language codes like `"sv;en;pl"`, or `""` to mean all languages, * and `extra-languages`, which is a semicolon-separated list of locale * identifiers like `"en;en_DK;zh_HK.big5hkscs;uz_UZ.utf8@cyrillic"`. * * Returns: %TRUE if the option was set correctly */ gboolean flatpak_installation_set_config_sync (FlatpakInstallation *self, const char *key, const char *value, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return FALSE; if (!flatpak_dir_set_config (dir, key, value, error)) return FALSE; /* Make sure we pick up the new config */ flatpak_installation_drop_caches (self, NULL, NULL); return TRUE; } /** * flatpak_installation_get_config: * @self: a #FlatpakInstallation * @key: the name of the key to get * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Get a global configuration option for the installation, see * flatpak_installation_set_config_sync() for supported keys. * * Returns: The (newly allocated) value, or %NULL on error (%G_KEY_FILE_ERROR_KEY_NOT_FOUND error if key is not set) */ char * flatpak_installation_get_config (FlatpakInstallation *self, const char *key, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; return flatpak_dir_get_config (dir, key, error); } /** * flatpak_installation_get_default_languages: * @self: a #FlatpakInstallation * @error: return location for a #GError * * Get the default languages used by the installation to decide which * subpaths to install of locale extensions. This list may also be used * by frontends like GNOME Software to decide which language-specific apps * to display. An empty array means that all languages should be installed. * * Returns: (array zero-terminated=1) (element-type utf8) (transfer full): * A possibly empty array of strings, or %NULL on error. * Since: 1.5.0 */ char ** flatpak_installation_get_default_languages (FlatpakInstallation *self, GError **error) { g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; return flatpak_dir_get_locale_languages (dir); } /** * flatpak_installation_get_default_locales: * @self: a #FlatpakInstallation * @error: return location for a #GError * * Like flatpak_installation_get_default_languages() but includes territory * information (e.g. `en_US` rather than `en`) which may be included in the * `extra-languages` configuration. * * Strings returned by this function are in the format specified by * [`setlocale()`](man:setlocale): `language[_territory][.codeset][@modifier]`. * * Returns: (array zero-terminated=1) (element-type utf8) (transfer full): * A possibly empty array of locale strings, or %NULL on error. * Since: 1.5.1 */ char ** flatpak_installation_get_default_locales (FlatpakInstallation *self, GError **error) { g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; return flatpak_dir_get_locales (dir); } /** * flatpak_installation_get_min_free_space_bytes: * @self: a #FlatpakInstallation * @out_bytes: (out): Location to store the result * @error: Return location for a #GError * * Returns the min-free-space config value from the OSTree repository of this installation. * * Applications can use this value, together with information about the available * disk space and the size of pending updates or installs, to estimate whether a * pull operation will fail due to running out of disk space. * * Returns: %TRUE on success, or %FALSE on error. * Since: 1.1 */ gboolean flatpak_installation_get_min_free_space_bytes (FlatpakInstallation *self, guint64 *out_bytes, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; dir = flatpak_installation_get_dir (self, NULL); if (dir == NULL) return FALSE; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, NULL, error)) return FALSE; return ostree_repo_get_min_free_space_bytes (flatpak_dir_get_repo (dir_clone), out_bytes, error); } /** * flatpak_installation_update_remote_sync: * @self: a #FlatpakInstallation * @name: the name of the remote to update * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Updates the local configuration of a remote repository by fetching * the related information from the summary file in the remote OSTree * repository and committing the changes to the local installation. * * Returns: %TRUE if the remote has been updated successfully * * Since: 0.6.13 */ gboolean flatpak_installation_update_remote_sync (FlatpakInstallation *self, const char *name, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return FALSE; if (!flatpak_dir_update_remote_configuration (dir, name, NULL, NULL, cancellable, error)) return FALSE; /* Make sure we pick up the new config */ flatpak_installation_drop_caches (self, NULL, NULL); return TRUE; } /** * flatpak_installation_get_remote_by_name: * @self: a #FlatpakInstallation * @name: a remote name * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Looks up a remote by name. * * Returns: (transfer full): a #FlatpakRemote instance, or %NULL with @error * set */ FlatpakRemote * flatpak_installation_get_remote_by_name (FlatpakInstallation *self, const gchar *name, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); g_autoptr(FlatpakDir) dir_clone = NULL; if (!flatpak_dir_has_remote (dir, name, error)) return NULL; /* We clone the dir here to make sure we re-read the latest ostree repo config, in case it has local changes */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return NULL; return flatpak_remote_new_with_dir (name, dir_clone); } /** * flatpak_installation_load_app_overrides: * @self: a #FlatpakInstallation * @app_id: an application id * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Loads the metadata overrides file for an application. * * Returns: (transfer full): the contents of the overrides files, * or %NULL if an error occurred */ char * flatpak_installation_load_app_overrides (FlatpakInstallation *self, const char *app_id, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; char *metadata_contents; gsize metadata_size; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; metadata_contents = flatpak_dir_load_override (dir, app_id, &metadata_size, error); if (metadata_contents == NULL) return NULL; return metadata_contents; } /** * flatpak_installation_install_bundle: * @self: a #FlatpakInstallation * @file: a #GFile that is an flatpak bundle * @progress: (scope call) (nullable): progress callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_install_bundle() * instead. It has a lot more interesting features. * * Install an application or runtime from an flatpak bundle file. * See flatpak-build-bundle(1) for how to create bundles. * * Returns: (transfer full): The ref for the newly installed app or %NULL on failure * Deprecated: 1.7.0: Use flatpak_transaction_add_install_bundle() instead. */ FlatpakInstalledRef * flatpak_installation_install_bundle (FlatpakInstallation *self, GFile *file, FlatpakProgressCallback progress, gpointer progress_data, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; g_autoptr(FlatpakDecomposed) ref = NULL; g_autofree char *remote = NULL; FlatpakInstalledRef *result = NULL; gboolean created_remote; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; remote = flatpak_dir_ensure_bundle_remote (dir, file, NULL, &ref, NULL, NULL, &created_remote, cancellable, error); if (remote == NULL) return NULL; /* Make sure we pick up the new config */ if (created_remote) flatpak_installation_drop_caches (self, NULL, NULL); /* Pull, prune, etc are not threadsafe, so we work on a copy */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return NULL; if (!flatpak_dir_install_bundle (dir_clone, file, remote, NULL, cancellable, error)) return NULL; if (flatpak_decomposed_is_app (ref)) flatpak_dir_run_triggers (dir_clone, cancellable, NULL); result = get_ref (dir, ref, cancellable, error); if (result == NULL) return NULL; return result; } /** * flatpak_installation_install_ref_file: * @self: a #FlatpakInstallation * @ref_file_data: The ref file contents * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_install_flatpakref() * instead. It has a lot more interesting features. * * Creates a remote based on the passed in .flatpakref file contents * in @ref_file_data and returns the #FlatpakRemoteRef that can be used * to install it. * * Note, the #FlatpakRemoteRef will not have the commit field set, or other details, to * avoid unnecessary roundtrips. If you need that you have to resolve it * explicitly with flatpak_installation_fetch_remote_ref_sync (). * * Returns: (transfer full): a #FlatpakRemoteRef if the remote has been added successfully, %NULL * on error. * * Since: 0.6.10 * Deprecated: 1.7.0: Use flatpak_transaction_add_install_flatpakref() instead. */ FlatpakRemoteRef * flatpak_installation_install_ref_file (FlatpakInstallation *self, GBytes *ref_file_data, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autofree char *remote = NULL; g_autoptr(FlatpakDecomposed) ref = NULL; g_autofree char *collection_id = NULL; g_autoptr(GKeyFile) keyfile = g_key_file_new (); dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; if (!g_key_file_load_from_data (keyfile, g_bytes_get_data (ref_file_data, NULL), g_bytes_get_size (ref_file_data), 0, error)) return FALSE; if (!flatpak_dir_create_remote_for_ref_file (dir, keyfile, NULL, &remote, &collection_id, &ref, error)) return NULL; if (!flatpak_installation_drop_caches (self, cancellable, error)) return NULL; return flatpak_remote_ref_new (ref, NULL, remote, collection_id, NULL); } /** * flatpak_installation_install_full: * @self: a #FlatpakInstallation * @flags: set of #FlatpakInstallFlags flag * @remote_name: name of the remote to use * @kind: what this ref contains (an #FlatpakRefKind) * @name: name of the app/runtime to fetch * @arch: (nullable): which architecture to fetch (default: current architecture) * @branch: (nullable): which branch to fetch (default: 'master') * @subpaths: (nullable) (array zero-terminated=1): A list of subpaths to fetch, or %NULL for everything * @progress: (scope call) (nullable): progress callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_install() * instead. It has a lot more interesting features. * * Install a new application or runtime. * * Note that this function was originally written to always return a * #FlatpakInstalledRef. Since 0.9.13, passing * FLATPAK_INSTALL_FLAGS_NO_DEPLOY will only pull refs into the local flatpak * repository without deploying them, however this function will * be unable to provide information on the installed ref, so * FLATPAK_ERROR_ONLY_PULLED will be set and the caller must respond * accordingly. * * Returns: (transfer full): The ref for the newly installed app or %NULL on failure * Deprecated: 1.7.0: Use flatpak_transaction_add_install() instead. */ FlatpakInstalledRef * flatpak_installation_install_full (FlatpakInstallation *self, FlatpakInstallFlags flags, const char *remote_name, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, const char * const *subpaths, FlatpakProgressCallback progress_cb, gpointer progress_data, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDecomposed) ref = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; g_autoptr(FlatpakProgress) progress = NULL; g_autoptr(GFile) deploy_dir = NULL; g_autoptr(FlatpakRemoteState) state = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; ref = flatpak_decomposed_new_from_parts (flatpak_kinds_from_kind (kind), name, arch, branch, error); if (ref == NULL) return NULL; deploy_dir = flatpak_dir_get_if_deployed (dir, ref, NULL, cancellable); if (deploy_dir != NULL) { flatpak_fail_error (error, FLATPAK_ERROR_ALREADY_INSTALLED, _("%s branch %s already installed"), name, branch ? branch : "master"); return NULL; } state = flatpak_dir_get_remote_state_optional (dir, remote_name, FALSE, cancellable, error); if (state == NULL) return NULL; /* Pull, prune, etc are not threadsafe, so we work on a copy */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return NULL; if (progress_cb) progress = flatpak_progress_new (progress_cb, progress_data); if (!flatpak_dir_install (dir_clone, (flags & FLATPAK_INSTALL_FLAGS_NO_PULL) != 0, (flags & FLATPAK_INSTALL_FLAGS_NO_DEPLOY) != 0, (flags & FLATPAK_INSTALL_FLAGS_NO_STATIC_DELTAS) != 0, FALSE, FALSE, FALSE, state, ref, NULL, (const char **) subpaths, NULL, NULL, NULL, NULL, progress, cancellable, error)) return NULL; if (!(flags & FLATPAK_INSTALL_FLAGS_NO_TRIGGERS) && flatpak_decomposed_is_app (ref)) flatpak_dir_run_triggers (dir_clone, cancellable, NULL); /* Note that if the caller sets FLATPAK_INSTALL_FLAGS_NO_DEPLOY we must * always return an error, as explained above. Otherwise get_ref will * always return an error. */ if ((flags & FLATPAK_INSTALL_FLAGS_NO_DEPLOY) != 0) { flatpak_fail_error (error, FLATPAK_ERROR_ONLY_PULLED, _("As requested, %s was only pulled, but not installed"), name); return NULL; } return get_ref (dir, ref, cancellable, error); } /** * flatpak_installation_install: * @self: a #FlatpakInstallation * @remote_name: name of the remote to use * @kind: what this ref contains (an #FlatpakRefKind) * @name: name of the app/runtime to fetch * @arch: (nullable): which architecture to fetch (default: current architecture) * @branch: (nullable): which branch to fetch (default: 'master') * @progress: (scope call) (nullable): progress callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_install() * instead. It has a lot more interesting features. * * Install a new application or runtime. * * Note that this function was originally written to always return a * #FlatpakInstalledRef. Since 0.9.13, passing * FLATPAK_INSTALL_FLAGS_NO_DEPLOY will only pull refs into the local flatpak * repository without deploying them, however this function will * be unable to provide information on the installed ref, so * FLATPAK_ERROR_ONLY_PULLED will be set and the caller must respond * accordingly. * * Returns: (transfer full): The ref for the newly installed app or %NULL on failure * Deprecated: 1.7.0: Use flatpak_transaction_add_install() instead. */ FlatpakInstalledRef * flatpak_installation_install (FlatpakInstallation *self, const char *remote_name, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, FlatpakProgressCallback progress, gpointer progress_data, GCancellable *cancellable, GError **error) { G_GNUC_BEGIN_IGNORE_DEPRECATIONS return flatpak_installation_install_full (self, FLATPAK_INSTALL_FLAGS_NONE, remote_name, kind, name, arch, branch, NULL, progress, progress_data, cancellable, error); G_GNUC_END_IGNORE_DEPRECATIONS } /** * flatpak_installation_update_full: * @self: a #FlatpakInstallation * @flags: set of #FlatpakUpdateFlags flag * @kind: whether this is an app or runtime * @name: name of the app or runtime to update * @arch: (nullable): architecture of the app or runtime to update (default: current architecture) * @branch: (nullable): name of the branch of the app or runtime to update (default: master) * @subpaths: (nullable) (array zero-terminated=1): A list of subpaths to fetch, or %NULL for everything * @progress: (scope call) (nullable): the callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_update() * instead. It has a lot more interesting features. * * Update an application or runtime. * * If the specified package is not installed, then %FLATPAK_ERROR_NOT_INSTALLED * will be thrown. * * If no updates could be found on the remote end and the package is * already up to date, then %FLATPAK_ERROR_ALREADY_INSTALLED will be thrown. * * Returns: (transfer full): The ref for the newly updated app or %NULL on failure * Deprecated: 1.7.0: Use flatpak_transaction_add_update() instead. */ FlatpakInstalledRef * flatpak_installation_update_full (FlatpakInstallation *self, FlatpakUpdateFlags flags, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, const char * const *subpaths, FlatpakProgressCallback progress_cb, gpointer progress_data, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDecomposed) ref = NULL; g_autoptr(GFile) deploy_dir = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; g_autoptr(FlatpakProgress) progress = NULL; g_autofree char *remote_name = NULL; FlatpakInstalledRef *result = NULL; g_autofree char *target_commit = NULL; g_autoptr(FlatpakRemoteState) state = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; ref = flatpak_decomposed_new_from_parts (flatpak_kinds_from_kind (kind), name, arch, branch, error); if (ref == NULL) return NULL; deploy_dir = flatpak_dir_get_if_deployed (dir, ref, NULL, cancellable); if (deploy_dir == NULL) { flatpak_fail_error (error, FLATPAK_ERROR_NOT_INSTALLED, _("%s branch %s is not installed"), name, flatpak_decomposed_get_branch (ref)); return NULL; } remote_name = flatpak_dir_get_origin (dir, ref, cancellable, error); if (remote_name == NULL) return NULL; state = flatpak_dir_get_remote_state_optional (dir, remote_name, FALSE, cancellable, error); if (state == NULL) return NULL; target_commit = flatpak_dir_check_for_update (dir, state, ref, NULL, (const char **) subpaths, (flags & FLATPAK_UPDATE_FLAGS_NO_PULL) != 0, cancellable, error); if (target_commit == NULL) return NULL; /* Pull, prune, etc are not threadsafe, so we work on a copy */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return NULL; if (progress_cb) progress = flatpak_progress_new (progress_cb, progress_data); if (!flatpak_dir_update (dir_clone, (flags & FLATPAK_UPDATE_FLAGS_NO_PULL) != 0, (flags & FLATPAK_UPDATE_FLAGS_NO_DEPLOY) != 0, (flags & FLATPAK_UPDATE_FLAGS_NO_STATIC_DELTAS) != 0, FALSE, FALSE, FALSE, state, ref, target_commit, (const char **) subpaths, NULL, NULL, NULL, NULL, progress, cancellable, error)) return NULL; if (!(flags & FLATPAK_UPDATE_FLAGS_NO_TRIGGERS) && flatpak_decomposed_is_app (ref)) flatpak_dir_run_triggers (dir_clone, cancellable, NULL); result = get_ref (dir, ref, cancellable, error); if (result == NULL) return NULL; /* We don't get prunable objects if not pulling or if NO_PRUNE is passed */ if (!(flags & FLATPAK_UPDATE_FLAGS_NO_PULL) && !(flags & FLATPAK_UPDATE_FLAGS_NO_PRUNE)) flatpak_dir_prune (dir_clone, cancellable, NULL); return result; } /** * flatpak_installation_update: * @self: a #FlatpakInstallation * @flags: set of #FlatpakUpdateFlags flag * @kind: whether this is an app or runtime * @name: name of the app or runtime to update * @arch: (nullable): architecture of the app or runtime to update (default: current architecture) * @branch: (nullable): name of the branch of the app or runtime to update (default: master) * @progress: (scope call) (nullable): the callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_update() * instead. It has a lot more interesting features. * * Update an application or runtime. * * If the specified package is not installed, then %FLATPAK_ERROR_NOT_INSTALLED * will be thrown. * * If no updates could be found on the remote end and the package is * already up to date, then %FLATPAK_ERROR_ALREADY_INSTALLED will be thrown. * * Returns: (transfer full): The ref for the newly updated app or %NULL on failure * Deprecated: 1.7.0: Use flatpak_transaction_add_update() instead. */ FlatpakInstalledRef * flatpak_installation_update (FlatpakInstallation *self, FlatpakUpdateFlags flags, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, FlatpakProgressCallback progress, gpointer progress_data, GCancellable *cancellable, GError **error) { G_GNUC_BEGIN_IGNORE_DEPRECATIONS return flatpak_installation_update_full (self, flags, kind, name, arch, branch, NULL, progress, progress_data, cancellable, error); G_GNUC_END_IGNORE_DEPRECATIONS } /** * flatpak_installation_uninstall: * @self: a #FlatpakInstallation * @kind: what this ref contains (an #FlatpakRefKind) * @name: name of the app or runtime to uninstall * @arch: (nullable): architecture of the app or runtime to uninstall; if * %NULL, flatpak_get_default_arch() is assumed * @branch: (nullable): name of the branch of the app or runtime to uninstall; * if %NULL, `master` is assumed * @progress: (scope call) (nullable): the callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_uninstall() * instead. It has a lot more interesting features. * * Uninstall an application or runtime. * * Returns: %TRUE on success * Deprecated: 1.7.0: Use flatpak_transaction_add_uninstall() instead. */ FLATPAK_EXTERN gboolean flatpak_installation_uninstall (FlatpakInstallation *self, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, FlatpakProgressCallback progress, gpointer progress_data, GCancellable *cancellable, GError **error) { G_GNUC_BEGIN_IGNORE_DEPRECATIONS return flatpak_installation_uninstall_full (self, FLATPAK_UNINSTALL_FLAGS_NONE, kind, name, arch, branch, progress, progress_data, cancellable, error); G_GNUC_END_IGNORE_DEPRECATIONS } /** * flatpak_installation_uninstall_full: * @self: a #FlatpakInstallation * @flags: set of #FlatpakUninstallFlags flags * @kind: what this ref contains (an #FlatpakRefKind) * @name: name of the app or runtime to uninstall * @arch: (nullable): architecture of the app or runtime to uninstall; if * %NULL, flatpak_get_default_arch() is assumed * @branch: (nullable): name of the branch of the app or runtime to uninstall; * if %NULL, `master` is assumed * @progress: (scope call) (nullable): the callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * This is an old deprecated function, you should use * #FlatpakTransaction and flatpak_transaction_add_uninstall() * instead. It has a lot more interesting features. * * Uninstall an application or runtime. * * Returns: %TRUE on success * * Since: 0.11.8 * Deprecated: 1.7.0: Use flatpak_transaction_add_uninstall() instead. */ gboolean flatpak_installation_uninstall_full (FlatpakInstallation *self, FlatpakUninstallFlags flags, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, FlatpakProgressCallback progress, gpointer progress_data, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDecomposed) ref = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; ref = flatpak_decomposed_new_from_parts (flatpak_kinds_from_kind (kind), name, arch, branch, error); if (ref == NULL) return FALSE; /* prune, etc are not threadsafe, so we work on a copy */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return FALSE; if (!flatpak_dir_uninstall (dir_clone, ref, FLATPAK_HELPER_UNINSTALL_FLAGS_NONE, cancellable, error)) return FALSE; if (!(flags & FLATPAK_UNINSTALL_FLAGS_NO_TRIGGERS) && flatpak_decomposed_is_app (ref)) flatpak_dir_run_triggers (dir_clone, cancellable, NULL); if (!(flags & FLATPAK_UNINSTALL_FLAGS_NO_PRUNE)) flatpak_dir_prune (dir_clone, cancellable, NULL); return TRUE; } /** * flatpak_installation_fetch_remote_size_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @ref: the ref * @download_size: (out): return location for the (maximum) download size * @installed_size: (out): return location for the installed size * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Gets information about the maximum amount of data that needs to be transferred * to pull the ref from a remote repository, and about the amount of * local disk space that is required to check out this commit. * * Note that if there are locally available data that are in the ref, which is common * for instance if you're doing an update then the real download size may be smaller * than what is returned here. * * NOTE: Since 0.11.4 this information is accessible in FlatpakRemoteRef, so this * function is not very useful anymore. * * Returns: %TRUE, unless an error occurred */ gboolean flatpak_installation_fetch_remote_size_sync (FlatpakInstallation *self, const char *remote_name, FlatpakRef *ref, guint64 *download_size, guint64 *installed_size, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakRemoteState) state = NULL; const char *full_ref = flatpak_ref_format_ref_cached (ref); dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; state = flatpak_dir_get_remote_state_optional (dir, remote_name, FALSE, cancellable, error); if (state == NULL) return FALSE; return flatpak_remote_state_load_data (state, full_ref, download_size, installed_size, NULL, error); } /** * flatpak_installation_fetch_remote_metadata_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @ref: the ref * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Obtains the metadata file from a commit. * * NOTE: Since 0.11.4 this information is accessible in FlatpakRemoteRef, so this * function is not very useful anymore. * * Returns: (transfer full): a #GBytes containing the flatpak metadata file, * or %NULL if an error occurred */ GBytes * flatpak_installation_fetch_remote_metadata_sync (FlatpakInstallation *self, const char *remote_name, FlatpakRef *ref, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakRemoteState) state = NULL; const char *full_ref = flatpak_ref_format_ref_cached (ref); g_autofree char *res = NULL; gsize len; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; state = flatpak_dir_get_remote_state_optional (dir, remote_name, FALSE, cancellable, error); if (state == NULL) return FALSE; if (!flatpak_remote_state_load_data (state, full_ref, NULL, NULL, &res, error)) return NULL; len = strlen (res); return g_bytes_new_take (g_steal_pointer (&res), len); } /** * flatpak_installation_list_remote_refs_sync: * @self: a #FlatpakInstallation * @remote_or_uri: the name or URI of the remote * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists all the applications and runtimes in a remote. * * Returns: (transfer container) (element-type FlatpakRemoteRef): a GPtrArray of * #FlatpakRemoteRef instances */ GPtrArray * flatpak_installation_list_remote_refs_sync (FlatpakInstallation *self, const char *remote_or_uri, GCancellable *cancellable, GError **error) { return flatpak_installation_list_remote_refs_sync_full (self, remote_or_uri, 0, cancellable, error); } /** * flatpak_installation_list_remote_refs_sync_full: * @self: a #FlatpakInstallation * @remote_or_uri: the name or URI of the remote * @flags: set of #FlatpakQueryFlags * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists all the applications and runtimes in a remote. * * Returns: (transfer container) (element-type FlatpakRemoteRef): a GPtrArray of * #FlatpakRemoteRef instances * * Since: 1.3.3 */ GPtrArray * flatpak_installation_list_remote_refs_sync_full (FlatpakInstallation *self, const char *remote_or_uri, FlatpakQueryFlags flags, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GPtrArray) refs = g_ptr_array_new_with_free_func (g_object_unref); g_autoptr(FlatpakRemoteState) state = NULL; g_autoptr(GHashTable) ht = NULL; g_autoptr(GError) local_error = NULL; GHashTableIter iter; gpointer key; gpointer value; gboolean only_sideloaded = (flags & FLATPAK_QUERY_FLAGS_ONLY_SIDELOADED) != 0; gboolean only_cached = (flags & FLATPAK_QUERY_FLAGS_ONLY_CACHED) != 0; gboolean all_arches = (flags & FLATPAK_QUERY_FLAGS_ALL_ARCHES) != 0; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; if (only_sideloaded) { state = flatpak_dir_get_remote_state_local_only (dir, remote_or_uri, cancellable, error); if (state == NULL) return NULL; } else { state = flatpak_dir_get_remote_state (dir, remote_or_uri, only_cached, cancellable, error); if (state == NULL) return NULL; if (all_arches && !flatpak_remote_state_ensure_subsummary_all_arches (state, dir, only_cached, cancellable, error)) return NULL; } if (!flatpak_dir_list_remote_refs (dir, state, &ht, cancellable, &local_error)) { if (only_sideloaded) { /* Just return no sideloaded refs rather than a summary download failed error if there are none */ return g_steal_pointer (&refs); } else { g_propagate_error (error, g_steal_pointer (&local_error)); return NULL; } } g_hash_table_iter_init (&iter, ht); while (g_hash_table_iter_next (&iter, &key, &value)) { FlatpakDecomposed *decomposed = key; const gchar *ref_commit = value; FlatpakRemoteRef *ref; ref = flatpak_remote_ref_new (decomposed, ref_commit, remote_or_uri, state->collection_id, state); if (ref) g_ptr_array_add (refs, ref); } return g_steal_pointer (&refs); } /** * flatpak_installation_fetch_remote_ref_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @kind: what this ref contains (an #FlatpakRefKind) * @name: name of the app/runtime to fetch * @arch: (nullable): which architecture to fetch (default: current architecture) * @branch: (nullable): which branch to fetch (default: 'master') * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Gets the current remote branch of a ref in the remote. * * Returns: (transfer full): a #FlatpakRemoteRef instance, or %NULL */ FlatpakRemoteRef * flatpak_installation_fetch_remote_ref_sync (FlatpakInstallation *self, const char *remote_name, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, GCancellable *cancellable, GError **error) { return flatpak_installation_fetch_remote_ref_sync_full (self, remote_name, kind, name, arch, branch, 0, cancellable, error); } /** * flatpak_installation_fetch_remote_ref_sync_full: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @kind: what this ref contains (an #FlatpakRefKind) * @name: name of the app/runtime to fetch * @arch: (nullable): which architecture to fetch (default: current architecture) * @branch: (nullable): which branch to fetch (default: 'master') * @flags: set of #FlatpakQueryFlags * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Gets the current remote branch of a ref in the remote. * * Returns: (transfer full): a #FlatpakRemoteRef instance, or %NULL * * Since: 1.3.3 */ FlatpakRemoteRef * flatpak_installation_fetch_remote_ref_sync_full (FlatpakInstallation *self, const char *remote_name, FlatpakRefKind kind, const char *name, const char *arch, const char *branch, FlatpakQueryFlags flags, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GHashTable) ht = NULL; g_autoptr(FlatpakRemoteState) state = NULL; g_autoptr(FlatpakDecomposed) ref = NULL; const char *checksum; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; ref = flatpak_decomposed_new_from_parts (flatpak_kinds_from_kind (kind), name, arch, branch, error); if (ref == NULL) return NULL; if (flags & FLATPAK_QUERY_FLAGS_ONLY_SIDELOADED) state = flatpak_dir_get_remote_state_local_only (dir, remote_name, cancellable, error); else state = flatpak_dir_get_remote_state (dir, remote_name, (flags & FLATPAK_QUERY_FLAGS_ONLY_CACHED) != 0, cancellable, error); if (state == NULL) return NULL; if (!flatpak_dir_list_remote_refs (dir, state, &ht, cancellable, error)) return NULL; checksum = g_hash_table_lookup (ht, ref); if (checksum != NULL) return flatpak_remote_ref_new (ref, checksum, remote_name, state->collection_id, state); g_set_error (error, FLATPAK_ERROR, FLATPAK_ERROR_REF_NOT_FOUND, "Reference %s doesn't exist in remote %s", flatpak_decomposed_get_ref (ref), remote_name); return NULL; } /** * flatpak_installation_update_appstream_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @arch: (nullable): Architecture to update, or %NULL for the local machine arch * @out_changed: (nullable): Set to %TRUE if the contents of the appstream changed, %FALSE if nothing changed * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Updates the local copy of appstream for @remote_name for the specified @arch. * If you need progress feedback, use flatpak_installation_update_appstream_full_sync(). * * Returns: %TRUE on success, or %FALSE on error */ gboolean flatpak_installation_update_appstream_sync (FlatpakInstallation *self, const char *remote_name, const char *arch, gboolean *out_changed, GCancellable *cancellable, GError **error) { return flatpak_installation_update_appstream_full_sync (self, remote_name, arch, NULL, NULL, out_changed, cancellable, error); } /** * flatpak_installation_update_appstream_full_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @arch: (nullable): Architecture to update, or %NULL for the local machine arch * @progress: (scope call) (nullable): progress callback * @progress_data: (closure progress) (nullable): user data passed to @progress * @out_changed: (nullable): Set to %TRUE if the contents of the appstream changed, %FALSE if nothing changed * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Updates the local copy of appstream for @remote_name for the specified @arch. * * Returns: %TRUE on success, or %FALSE on error */ gboolean flatpak_installation_update_appstream_full_sync (FlatpakInstallation *self, const char *remote_name, const char *arch, FlatpakProgressCallback progress_cb, gpointer progress_data, gboolean *out_changed, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(FlatpakDir) dir_clone = NULL; g_autoptr(FlatpakProgress) progress = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; /* Pull, prune, etc are not threadsafe, so we work on a copy */ dir_clone = flatpak_dir_clone (dir); if (!flatpak_dir_ensure_repo (dir_clone, cancellable, error)) return FALSE; if (progress_cb) progress = flatpak_progress_new (progress_cb, progress_data); return flatpak_dir_update_appstream (dir_clone, remote_name, arch, out_changed, progress, cancellable, error); } /** * flatpak_installation_create_monitor: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Gets monitor object for the installation. The returned file monitor will * emit the #GFileMonitor::changed signal whenever an application or runtime * was installed, uninstalled or updated. * * Returns: (transfer full): a new #GFileMonitor instance, or %NULL on error */ GFileMonitor * flatpak_installation_create_monitor (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = flatpak_installation_get_dir_maybe_no_repo (self); g_autoptr(GFile) path = NULL; path = flatpak_dir_get_changed_path (dir); return g_file_monitor_file (path, G_FILE_MONITOR_NONE, cancellable, error); } /** * flatpak_installation_list_remote_related_refs_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @ref: the ref * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists all the available refs on @remote_name that are related to * @ref, and the subpaths to use. These are things that are * interesting to install, update, or uninstall together with * @ref. For instance, locale data or debug information. * * The returned list contains all available related refs, but not * every one should always be installed. For example, * flatpak_related_ref_should_download() returns %TRUE if the * reference should be installed/updated with the app, and * flatpak_related_ref_should_delete() returns %TRUE if it * should be uninstalled with the main ref. * * The commit property of each #FlatpakRelatedRef is not guaranteed to be * non-%NULL. * * Returns: (transfer container) (element-type FlatpakRelatedRef): a GPtrArray of * #FlatpakRelatedRef instances * * Since: 0.6.7 */ GPtrArray * flatpak_installation_list_remote_related_refs_sync (FlatpakInstallation *self, const char *remote_name, const char *ref, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GPtrArray) related = NULL; g_autoptr(GPtrArray) refs = g_ptr_array_new_with_free_func (g_object_unref); g_autoptr(FlatpakRemoteState) state = NULL; g_autoptr(FlatpakDecomposed) decomposed = NULL; int i; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; decomposed = flatpak_decomposed_new_from_ref (ref, error); if (decomposed == NULL) return NULL; state = flatpak_dir_get_remote_state_optional (dir, remote_name, FALSE, cancellable, error); if (state == NULL) return NULL; related = flatpak_dir_find_remote_related (dir, state, decomposed, FALSE, cancellable, error); if (related == NULL) return NULL; for (i = 0; i < related->len; i++) { FlatpakRelated *rel = g_ptr_array_index (related, i); FlatpakRelatedRef *rel_ref; rel_ref = flatpak_related_ref_new (flatpak_decomposed_get_ref (rel->ref), rel->commit, rel->subpaths, rel->download, rel->delete); if (rel_ref) g_ptr_array_add (refs, rel_ref); } return g_steal_pointer (&refs); } /** * flatpak_installation_list_installed_related_refs_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote providing @ref * @ref: the ref * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists all the locally installed refs that are related to @ref. These are * things that are interesting to install, update, or uninstall together with * @ref. For instance, locale data or debug information. * * Note that while the related refs are usually installed from the same remote * as @ref (@remote_name), it is possible they were installed from another * remote. * * This function is similar to flatpak_installation_list_remote_related_refs_sync, * but instead of looking at what is available on the remote, it only looks * at the locally installed refs. This is useful for instance when you're * looking for related refs to uninstall, or when you're planning to use * FLATPAK_UPDATE_FLAGS_NO_PULL to install previously pulled refs. * * Returns: (transfer container) (element-type FlatpakRelatedRef): a GPtrArray of * #FlatpakRelatedRef instances * * Since: 0.6.7 */ GPtrArray * flatpak_installation_list_installed_related_refs_sync (FlatpakInstallation *self, const char *remote_name, const char *ref, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GPtrArray) related = NULL; g_autoptr(GPtrArray) refs = g_ptr_array_new_with_free_func (g_object_unref); g_autoptr(FlatpakDecomposed) decomposed = NULL; int i; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; decomposed = flatpak_decomposed_new_from_ref (ref, error); if (decomposed == NULL) return NULL; related = flatpak_dir_find_local_related (dir, decomposed, remote_name, TRUE, cancellable, error); if (related == NULL) return NULL; for (i = 0; i < related->len; i++) { FlatpakRelated *rel = g_ptr_array_index (related, i); FlatpakRelatedRef *rel_ref; rel_ref = flatpak_related_ref_new (flatpak_decomposed_get_ref (rel->ref), rel->commit, rel->subpaths, rel->download, rel->delete); if (rel_ref) g_ptr_array_add (refs, rel_ref); } return g_steal_pointer (&refs); } /** * flatpak_installation_list_remote_related_refs_for_installed_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @ref: the ref * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists all the available refs on @remote_name that are related to @ref, and * which are appropriate for the installed version of @ref. For example if the * installed version of org.videolan.VLC has a related ref of * org.videolan.VLC.Plugin.bdj//3-19.08 and the remote version of VLC has a * related ref of org.videolan.VLC.Plugin.bdj//3-20.08, this function will only * return the 3-19.08 branch. * * See also the related functions * flatpak_installation_list_remote_related_refs_sync() and * flatpak_installation_list_installed_related_refs_sync(). * * The returned list contains all available related refs, but not * every one should always be installed. For example, * flatpak_related_ref_should_download() returns %TRUE if the * reference should be installed/updated with the app, and * flatpak_related_ref_should_delete() returns %TRUE if it * should be uninstalled with the main ref. * * The commit property of each #FlatpakRelatedRef is not guaranteed to be * non-%NULL. * * Returns: (transfer container) (element-type FlatpakRelatedRef): a GPtrArray of * #FlatpakRelatedRef instances * * Since: 1.11.1 */ GPtrArray * flatpak_installation_list_remote_related_refs_for_installed_sync (FlatpakInstallation *self, const char *remote_name, const char *ref, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GPtrArray) related = NULL; g_autoptr(GPtrArray) refs = g_ptr_array_new_with_free_func (g_object_unref); g_autoptr(FlatpakRemoteState) state = NULL; g_autoptr(FlatpakDecomposed) decomposed = NULL; int i; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; decomposed = flatpak_decomposed_new_from_ref (ref, error); if (decomposed == NULL) return NULL; state = flatpak_dir_get_remote_state_optional (dir, remote_name, FALSE, cancellable, error); if (state == NULL) return NULL; related = flatpak_dir_find_remote_related (dir, state, decomposed, TRUE, /* use_installed_metadata */ cancellable, error); if (related == NULL) return NULL; for (i = 0; i < related->len; i++) { FlatpakRelated *rel = g_ptr_array_index (related, i); FlatpakRelatedRef *rel_ref; rel_ref = flatpak_related_ref_new (flatpak_decomposed_get_ref (rel->ref), rel->commit, rel->subpaths, rel->download, rel->delete); if (rel_ref) g_ptr_array_add (refs, rel_ref); } return g_steal_pointer (&refs); } /** * flatpak_installation_remove_local_ref_sync: * @self: a #FlatpakInstallation * @remote_name: the name of the remote * @ref: the ref * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Remove the OSTree ref given by @remote_name:@ref from the local flatpak * repository. The next time the underlying OSTree repo is pruned, objects * which were attached to that ref will be removed. This is useful if you * pulled a flatpak ref using flatpak_installation_install_full() and * specified %FLATPAK_INSTALL_FLAGS_NO_DEPLOY but then decided not to * deploy the ref later on and want to remove the local ref to prevent it * from taking up disk space. Note that this will not remove the objects * referred to by @ref from the underlying OSTree repo, you should use * flatpak_installation_prune_local_repo() to do that. * * Since: 0.10.0 * Returns: %TRUE on success */ gboolean flatpak_installation_remove_local_ref_sync (FlatpakInstallation *self, const char *remote_name, const char *ref, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; return flatpak_dir_remove_ref (dir, remote_name, ref, cancellable, error); } /** * flatpak_installation_cleanup_local_refs_sync: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Remove all OSTree refs from the local flatpak repository which are not * in a deployed state. The next time the underlying OSTree repo is pruned, * objects which were attached to that ref will be removed. This is useful if * you pulled a flatpak refs using flatpak_installation_install_full() and * specified %FLATPAK_INSTALL_FLAGS_NO_DEPLOY but then decided not to * deploy the refs later on and want to remove the local refs to prevent them * from taking up disk space. Note that this will not remove the objects * referred to by @ref from the underlying OSTree repo, you should use * flatpak_installation_prune_local_repo() to do that. * * Since: 0.10.0 * Returns: %TRUE on success */ gboolean flatpak_installation_cleanup_local_refs_sync (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; return flatpak_dir_cleanup_undeployed_refs (dir, cancellable, error); } /** * flatpak_installation_prune_local_repo: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Remove all orphaned OSTree objects from the underlying OSTree repo in * @self. * * Since: 0.10.0 * Returns: %TRUE on success */ gboolean flatpak_installation_prune_local_repo (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; return flatpak_dir_prune (dir, cancellable, error); } /** * flatpak_installation_run_triggers: * @self: a #FlatpakInstallation * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Run the trigger commands to update the files exported by the apps in * @self. Should be used after one or more app install, upgrade or * uninstall operations with the %FLATPAK_INSTALL_FLAGS_NO_TRIGGERS, * %FLATPAK_UPDATE_FLAGS_NO_TRIGGERS or %FLATPAK_UNINSTALL_FLAGS_NO_TRIGGERS * flags set. * * Since: 1.0.3 * Returns: %TRUE on success */ gboolean flatpak_installation_run_triggers (FlatpakInstallation *self, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return FALSE; return flatpak_dir_run_triggers (dir, cancellable, error); } /** * flatpak_installation_list_unused_refs: * @self: a #FlatpakInstallation * @arch: (nullable): if non-%NULL, the architecture of refs to collect * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists the installed references that are not 'used'. * * A reference is used if it is either an application, * or the runtime or sdk of a used ref, or an extension of a used ref. * Pinned runtimes are also considered used; see flatpak-pin(1) and * flatpak_installation_list_pinned_refs(). * * Returns: (transfer container) (element-type FlatpakInstalledRef): a GPtrArray of * #FlatpakInstalledRef instances * * Since: 1.1.2 */ GPtrArray * flatpak_installation_list_unused_refs (FlatpakInstallation *self, const char *arch, GCancellable *cancellable, GError **error) { return flatpak_installation_list_unused_refs_with_options (self, arch, NULL, NULL, cancellable, error); } /** * flatpak_installation_list_unused_refs_with_options: * @self: a #FlatpakInstallation * @arch: (nullable): if non-%NULL, the architecture of refs to collect * @metadata_injection: (nullable): if non-%NULL, a #GHashTable mapping refs to * #GKeyFile objects, which when available will * be used instead of installed metadata * @options: (nullable): if non-%NULL, a GVariant a{sv} with an extensible set * of options * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Like flatpak_installation_list_unused_refs() but supports an extensible set * of options as well as an @metadata_injection parameter. The following are * currently defined: * * * exclude-refs (as): Act as if these refs are not installed even if they * are when determining the set of unused refs * * filter-by-eol (b): Only return refs as unused if they are End-Of-Life. * Note that if this option is combined with other filters (of which there * are none currently) non-EOL refs may also be returned. * * Returns: (transfer container) (element-type FlatpakInstalledRef): a GPtrArray of * #FlatpakInstalledRef instances * * Since: 1.9.1 */ GPtrArray * flatpak_installation_list_unused_refs_with_options (FlatpakInstallation *self, const char *arch, GHashTable *metadata_injection, GVariant *options, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GPtrArray) refs = NULL; g_auto(GStrv) refs_strv = NULL; g_autofree char **refs_to_exclude = NULL; gboolean filter_by_eol = FALSE; if (options) { (void) g_variant_lookup (options, "exclude-refs", "^a&s", &refs_to_exclude); (void) g_variant_lookup (options, "filter-by-eol", "b", &filter_by_eol); } dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; refs_strv = flatpak_dir_list_unused_refs (dir, arch, metadata_injection, NULL, (const char * const *)refs_to_exclude, filter_by_eol, cancellable, error); if (refs_strv == NULL) return NULL; refs = g_ptr_array_new_with_free_func (g_object_unref); for (char **iter = refs_strv; iter && *iter; iter++) { g_autoptr(GError) local_error = NULL; FlatpakInstalledRef *ref = NULL; g_autoptr(FlatpakDecomposed) decomposed = flatpak_decomposed_new_from_ref (*iter, &local_error); if (decomposed) ref = get_ref (dir, decomposed, cancellable, &local_error); if (ref != NULL) g_ptr_array_add (refs, ref); else g_warning ("Unexpected failure getting ref for %s: %s", flatpak_decomposed_get_ref (decomposed), local_error->message); } return g_steal_pointer (&refs); } /** * flatpak_installation_list_pinned_refs: * @self: a #FlatpakInstallation * @arch: (nullable): if non-%NULL, the architecture of refs to collect * @cancellable: (nullable): a #GCancellable * @error: return location for a #GError * * Lists the installed references that are pinned, meaning they will not be * returned by flatpak_installation_list_unused_refs() and won't be removed * unless explicitly specified for removal. * * Refs appear here either because they have been pinned automatically by * Flatpak or because the user pinned them; see flatpak-pin(1). * * Returns: (transfer container) (element-type FlatpakInstalledRef): a GPtrArray of * #FlatpakInstalledRef instances * * Since: 1.9.1 */ GPtrArray * flatpak_installation_list_pinned_refs (FlatpakInstallation *self, const char *arch, GCancellable *cancellable, GError **error) { g_autoptr(FlatpakDir) dir = NULL; g_autoptr(GPtrArray) refs = NULL; g_autoptr(GPtrArray) runtime_refs = NULL; int i; dir = flatpak_installation_get_dir (self, error); if (dir == NULL) return NULL; runtime_refs = flatpak_dir_list_refs (dir, FLATPAK_KINDS_RUNTIME, cancellable, error); if (runtime_refs == NULL) return NULL; refs = g_ptr_array_new_with_free_func (g_object_unref); for (i = 0; i < runtime_refs->len; i++) { FlatpakDecomposed *decomposed = g_ptr_array_index (runtime_refs, i); if (arch != NULL && !flatpak_decomposed_is_arch (decomposed, arch)) continue; if (flatpak_dir_ref_is_pinned (dir, flatpak_decomposed_get_ref (decomposed))) { g_autoptr(GError) local_error = NULL; FlatpakInstalledRef *ref = get_ref (dir, decomposed, cancellable, &local_error); if (ref != NULL) g_ptr_array_add (refs, ref); else g_warning ("Unexpected failure getting ref for %s: %s", flatpak_decomposed_get_ref (decomposed), local_error->message); } } return g_steal_pointer (&refs); }