summaryrefslogtreecommitdiff
path: root/app/flatpak-builtins-install.c
diff options
context:
space:
mode:
Diffstat (limited to 'app/flatpak-builtins-install.c')
-rw-r--r--app/flatpak-builtins-install.c264
1 files changed, 264 insertions, 0 deletions
diff --git a/app/flatpak-builtins-install.c b/app/flatpak-builtins-install.c
new file mode 100644
index 0000000..1034dac
--- /dev/null
+++ b/app/flatpak-builtins-install.c
@@ -0,0 +1,264 @@
+/*
+ * Copyright © 2014 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 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * Authors:
+ * Alexander Larsson <alexl@redhat.com>
+ */
+
+#include "config.h"
+
+#include <locale.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <string.h>
+
+#include <gio/gunixinputstream.h>
+
+#include "libgsystem.h"
+#include "libglnx/libglnx.h"
+
+#include "flatpak-builtins.h"
+#include "flatpak-utils.h"
+#include "flatpak-chain-input-stream.h"
+
+static char *opt_arch;
+static char **opt_gpg_file;
+static char **opt_subpaths;
+static gboolean opt_no_pull;
+static gboolean opt_no_deploy;
+static gboolean opt_runtime;
+static gboolean opt_app;
+static gboolean opt_bundle;
+
+static GOptionEntry options[] = {
+ { "arch", 0, 0, G_OPTION_ARG_STRING, &opt_arch, "Arch to install for", "ARCH" },
+ { "no-pull", 0, 0, G_OPTION_ARG_NONE, &opt_no_pull, "Don't pull, only install from local cache", },
+ { "no-deploy", 0, 0, G_OPTION_ARG_NONE, &opt_no_deploy, "Don't deploy, only download to local cache", },
+ { "runtime", 0, 0, G_OPTION_ARG_NONE, &opt_runtime, "Look for runtime with the specified name", },
+ { "app", 0, 0, G_OPTION_ARG_NONE, &opt_app, "Look for app with the specified name", },
+ { "bundle", 0, 0, G_OPTION_ARG_NONE, &opt_bundle, "Install from local bundle file", },
+ { "gpg-file", 0, 0, G_OPTION_ARG_FILENAME_ARRAY, &opt_gpg_file, "Check bundle signatures with GPG key from FILE (- for stdin)", "FILE" },
+ { "subpath", 0, 0, G_OPTION_ARG_FILENAME_ARRAY, &opt_subpaths, "Only install this subpath", "path" },
+ { NULL }
+};
+
+static GBytes *
+read_gpg_data (GCancellable *cancellable,
+ GError **error)
+{
+ g_autoptr(GInputStream) source_stream = NULL;
+ guint n_keyrings = 0;
+ g_autoptr(GPtrArray) streams = NULL;
+
+ if (opt_gpg_file != NULL)
+ n_keyrings = g_strv_length (opt_gpg_file);
+
+ guint ii;
+
+ streams = g_ptr_array_new_with_free_func (g_object_unref);
+
+ for (ii = 0; ii < n_keyrings; ii++)
+ {
+ GInputStream *input_stream = NULL;
+
+ if (strcmp (opt_gpg_file[ii], "-") == 0)
+ {
+ input_stream = g_unix_input_stream_new (STDIN_FILENO, FALSE);
+ }
+ else
+ {
+ g_autoptr(GFile) file = g_file_new_for_path (opt_gpg_file[ii]);
+ input_stream = G_INPUT_STREAM (g_file_read (file, cancellable, error));
+
+ if (input_stream == NULL)
+ return NULL;
+ }
+
+ /* Takes ownership. */
+ g_ptr_array_add (streams, input_stream);
+ }
+
+ /* Chain together all the --keyring options as one long stream. */
+ source_stream = (GInputStream *) flatpak_chain_input_stream_new (streams);
+
+ return flatpak_read_stream (source_stream, FALSE, error);
+}
+
+gboolean
+install_bundle (FlatpakDir *dir,
+ GOptionContext *context,
+ int argc, char **argv,
+ GCancellable *cancellable,
+ GError **error)
+{
+ gboolean ret = FALSE;
+
+ g_autoptr(GFile) deploy_base = NULL;
+ g_autoptr(GFile) file = NULL;
+ const char *filename;
+ g_autofree char *ref = NULL;
+ g_autofree char *origin = NULL;
+ gboolean added_remote = FALSE;
+ g_autofree char *to_checksum = NULL;
+ g_auto(GStrv) parts = NULL;
+ g_autoptr(GBytes) gpg_data = NULL;
+ g_autofree char *remote = NULL;
+ OstreeRepo *repo;
+ g_autoptr(GVariant) metadata = NULL;
+ g_autofree char *basename = NULL;
+
+ if (argc < 2)
+ return usage_error (context, "bundle filename must be specified", error);
+
+ filename = argv[1];
+
+ repo = flatpak_dir_get_repo (dir);
+
+ file = g_file_new_for_commandline_arg (filename);
+
+ metadata = flatpak_bundle_load (file, &to_checksum,
+ &ref,
+ &origin,
+ NULL,
+ &gpg_data,
+ error);
+ if (metadata == NULL)
+ return FALSE;
+
+ if (opt_gpg_file != NULL)
+ {
+ /* Override gpg_data from file */
+ gpg_data = read_gpg_data (cancellable, error);
+ if (gpg_data == NULL)
+ return FALSE;
+ }
+
+ parts = flatpak_decompose_ref (ref, error);
+ if (parts == NULL)
+ return FALSE;
+
+ deploy_base = flatpak_dir_get_deploy_dir (dir, ref);
+
+ if (g_file_query_exists (deploy_base, cancellable))
+ return flatpak_fail (error, "%s branch %s already installed", parts[1], parts[3]);
+
+ /* Add a remote for later updates */
+ basename = g_file_get_basename (file);
+ remote = flatpak_dir_create_origin_remote (dir,
+ origin,
+ parts[1],
+ basename,
+ gpg_data,
+ cancellable,
+ error);
+ if (remote == NULL)
+ return FALSE;
+
+ /* From here we need to goto out on error, to clean up */
+ added_remote = TRUE;
+
+ if (!flatpak_dir_ensure_repo (dir, cancellable, error))
+ goto out;
+
+ if (!flatpak_pull_from_bundle (flatpak_dir_get_repo (dir),
+ file,
+ remote,
+ ref,
+ gpg_data != NULL,
+ cancellable,
+ error))
+ goto out;
+
+ if (!flatpak_dir_deploy_install (dir, ref, remote, NULL, cancellable, error))
+ goto out;
+
+ ret = TRUE;
+
+out:
+ if (added_remote && !ret)
+ ostree_repo_remote_delete (repo, remote, NULL, NULL);
+
+ return ret;
+}
+
+gboolean
+flatpak_builtin_install (int argc, char **argv, GCancellable *cancellable, GError **error)
+{
+ g_autoptr(GOptionContext) context = NULL;
+ g_autoptr(FlatpakDir) dir = NULL;
+ g_autoptr(GFile) deploy_base = NULL;
+ const char *repository;
+ const char *name;
+ const char *branch = NULL;
+ g_autofree char *ref = NULL;
+ g_autofree char *installed_ref = NULL;
+ gboolean is_app;
+ g_autoptr(GError) my_error = NULL;
+
+ context = g_option_context_new ("REPOSITORY NAME [BRANCH] - Install an application or runtime");
+
+ if (!flatpak_option_context_parse (context, options, &argc, &argv, 0, &dir, cancellable, error))
+ return FALSE;
+
+ if (opt_bundle)
+ return install_bundle (dir, context, argc, argv, cancellable, error);
+
+ if (argc < 3)
+ return usage_error (context, "REPOSITORY and NAME must be specified", error);
+
+ repository = argv[1];
+ name = argv[2];
+ if (argc >= 4)
+ branch = argv[3];
+
+ if (!opt_app && !opt_runtime)
+ opt_app = opt_runtime = TRUE;
+
+ installed_ref = flatpak_dir_find_installed_ref (dir,
+ name,
+ branch,
+ opt_arch,
+ opt_app, opt_runtime, &is_app,
+ &my_error);
+ if (installed_ref != NULL)
+ return flatpak_fail (error, "%s %s, branch %s is already installed",
+ is_app ? "App" : "Runtime", name, branch ? branch : "master");
+
+ if (!g_error_matches (my_error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND))
+ {
+ g_propagate_error (error, g_steal_pointer (&my_error));
+ return FALSE;
+ }
+
+ ref = flatpak_dir_find_remote_ref (dir, repository, name, branch, opt_arch,
+ opt_app, opt_runtime, &is_app, cancellable, error);
+ if (ref == NULL)
+ return FALSE;
+
+ deploy_base = flatpak_dir_get_deploy_dir (dir, ref);
+ if (g_file_query_exists (deploy_base, cancellable))
+ return flatpak_fail (error, "Ref %s already deployed", ref);
+
+ if (!flatpak_dir_install (dir,
+ opt_no_pull,
+ opt_no_deploy,
+ ref, repository, opt_subpaths,
+ NULL,
+ cancellable, error))
+ return FALSE;
+
+ return TRUE;
+}