[ostree] core: Add local-clone builtin
- From: Colin Walters <walters src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [ostree] core: Add local-clone builtin
- Date: Fri, 18 Nov 2011 23:52:40 +0000 (UTC)
commit a04273167308c227d1d0fb4aa5fbdb40981bd23e
Author: Colin Walters <walters verbum org>
Date: Fri Nov 18 18:50:53 2011 -0500
core: Add local-clone builtin
This is useful for converting between e.g. archive and non-archive
repositories.
Makefile-ostree.am | 1 +
src/ostree/main.c | 1 +
src/ostree/ot-builtin-local-clone.c | 228 +++++++++++++++++++++++++++++++++++
src/ostree/ot-builtins.h | 1 +
tests/t0000-basic.sh | 13 ++-
tests/t0001-archive.sh | 14 ++-
6 files changed, 256 insertions(+), 2 deletions(-)
---
diff --git a/Makefile-ostree.am b/Makefile-ostree.am
index aa04c94..cb0b7c6 100644
--- a/Makefile-ostree.am
+++ b/Makefile-ostree.am
@@ -28,6 +28,7 @@ ostree_SOURCES = src/ostree/main.c \
src/ostree/ot-builtin-diff.c \
src/ostree/ot-builtin-fsck.c \
src/ostree/ot-builtin-init.c \
+ src/ostree/ot-builtin-local-clone.c \
src/ostree/ot-builtin-log.c \
src/ostree/ot-builtin-run-triggers.c \
src/ostree/ot-builtin-remote.c \
diff --git a/src/ostree/main.c b/src/ostree/main.c
index 88dd313..83b85a2 100644
--- a/src/ostree/main.c
+++ b/src/ostree/main.c
@@ -35,6 +35,7 @@ static OstreeBuiltin builtins[] = {
{ "init", ostree_builtin_init, 0 },
{ "commit", ostree_builtin_commit, 0 },
{ "compose", ostree_builtin_compose, 0 },
+ { "local-clone", ostree_builtin_local_clone, 0 },
{ "log", ostree_builtin_log, 0 },
#ifdef HAVE_LIBSOUP_GNOME
{ "pull", ostree_builtin_pull, 0 },
diff --git a/src/ostree/ot-builtin-local-clone.c b/src/ostree/ot-builtin-local-clone.c
new file mode 100644
index 0000000..11a4503
--- /dev/null
+++ b/src/ostree/ot-builtin-local-clone.c
@@ -0,0 +1,228 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
+ *
+ * Copyright (C) 2011 Colin Walters <walters verbum org>
+ *
+ * This library 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, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ * Author: Colin Walters <walters verbum org>
+ */
+
+#include "config.h"
+
+#include "ot-builtins.h"
+#include "ostree.h"
+
+#include <glib/gi18n.h>
+
+static GOptionEntry options[] = {
+ { NULL }
+};
+
+typedef struct {
+ OstreeRepo *src_repo;
+ OstreeRepo *dest_repo;
+ gboolean uids_differ;
+} OtLocalCloneData;
+
+static gboolean
+copy_dir_contents (GFile *src,
+ GFile *dest,
+ GCancellable *cancellable,
+ GError **error)
+{
+ gboolean ret = FALSE;
+ GFile *child_src = NULL;
+ GFile *child_dest = NULL;
+ GFileEnumerator *dir_enum = NULL;
+ GFileInfo *file_info = NULL;
+ GError *temp_error = NULL;
+
+ dir_enum = g_file_enumerate_children (src, OSTREE_GIO_FAST_QUERYINFO,
+ G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
+ cancellable, error);
+ if (!dir_enum)
+ goto out;
+ while ((file_info = g_file_enumerator_next_file (dir_enum, cancellable, &temp_error)) != NULL)
+ {
+ const char *name = g_file_info_get_name (file_info);
+
+ g_clear_object (&child_src);
+ child_src = g_file_get_child (src, name);
+ g_clear_object (&child_dest);
+ child_dest = g_file_get_child (dest, name);
+
+ if (!g_file_copy (child_src, child_dest, G_FILE_COPY_OVERWRITE | G_FILE_COPY_NOFOLLOW_SYMLINKS,
+ cancellable, NULL, NULL, error))
+ goto out;
+
+ g_clear_object (&file_info);
+ }
+ if (temp_error)
+ {
+ g_propagate_error (error, temp_error);
+ goto out;
+ }
+
+ ret = TRUE;
+ out:
+ g_clear_object (&dir_enum);
+ g_clear_object (&child_src);
+ g_clear_object (&child_dest);
+ g_clear_object (&file_info);
+ return ret;
+}
+
+static void
+object_iter_callback (OstreeRepo *repo,
+ const char *checksum,
+ OstreeObjectType objtype,
+ GFile *objfile,
+ GFileInfo *file_info,
+ gpointer user_data)
+{
+ OtLocalCloneData *data = user_data;
+ GFile *dest = NULL;
+ GError *error = NULL;
+ GFile *tmpdir = NULL;
+ gboolean did_exist;
+
+ if ((objtype == OSTREE_OBJECT_TYPE_FILE
+ && ostree_repo_is_archive (data->src_repo))
+ || data->uids_differ)
+ {
+ tmpdir = ostree_repo_get_tmpdir (data->dest_repo);
+ dest = g_file_get_child (tmpdir, checksum);
+
+ if (!ostree_unpack_object (objfile, objtype,
+ dest, NULL, &error))
+ goto out;
+ if (!ostree_repo_store_object_trusted (data->dest_repo,
+ dest,
+ checksum,
+ objtype,
+ FALSE,
+ &did_exist,
+ &error))
+ goto out;
+ }
+ else
+ {
+ if (!ostree_repo_store_object_trusted (data->dest_repo,
+ objfile,
+ checksum,
+ objtype,
+ FALSE,
+ &did_exist,
+ &error))
+ goto out;
+ }
+
+ out:
+ if (dest)
+ (void) g_file_delete (dest, NULL, NULL);
+ g_clear_object (&dest);
+ if (error != NULL)
+ {
+ g_printerr ("%s\n", error->message);
+ g_clear_error (&error);
+ }
+}
+
+gboolean
+ostree_builtin_local_clone (int argc, char **argv, const char *repo_path, GError **error)
+{
+ gboolean ret = FALSE;
+ GOptionContext *context;
+ const char *destination;
+ OtLocalCloneData data;
+ GFile *src_repo_dir = NULL;
+ GFile *dest_repo_dir = NULL;
+ GFileInfo *src_info = NULL;
+ GFileInfo *dest_info = NULL;
+ GFile *src_dir = NULL;
+ GFile *dest_dir = NULL;
+
+ context = g_option_context_new ("DEST ... - Create new repository DEST");
+ g_option_context_add_main_entries (context, options, NULL);
+
+ if (!g_option_context_parse (context, &argc, &argv, error))
+ goto out;
+
+ data.src_repo = ostree_repo_new (repo_path);
+ if (!ostree_repo_check (data.src_repo, error))
+ goto out;
+
+ if (argc < 1)
+ {
+ gchar *help = g_option_context_get_help (context, TRUE, NULL);
+ g_printerr ("%s\n", help);
+ g_free (help);
+ g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_FAILED,
+ "DESTINATION must be specified");
+ goto out;
+ }
+
+ destination = argv[1];
+
+ data.dest_repo = ostree_repo_new (destination);
+ if (!ostree_repo_check (data.dest_repo, error))
+ goto out;
+
+ src_repo_dir = ot_gfile_new_for_path (ostree_repo_get_path (data.src_repo));
+ dest_repo_dir = ot_gfile_new_for_path (ostree_repo_get_path (data.dest_repo));
+
+ src_info = g_file_query_info (src_repo_dir, OSTREE_GIO_FAST_QUERYINFO,
+ G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
+ NULL, error);
+ if (!src_info)
+ goto out;
+ dest_info = g_file_query_info (dest_repo_dir, OSTREE_GIO_FAST_QUERYINFO,
+ G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
+ NULL, error);
+ if (!dest_info)
+ goto out;
+
+ data.uids_differ = g_file_info_get_attribute_uint32 (src_info, "unix::uid") != g_file_info_get_attribute_uint32 (dest_info, "unix::uid");
+
+ if (!ostree_repo_iter_objects (data.src_repo, object_iter_callback, &data, error))
+ goto out;
+
+ src_dir = g_file_resolve_relative_path (src_repo_dir, "refs/heads");
+ dest_dir = g_file_resolve_relative_path (dest_repo_dir, "refs/heads");
+ if (!copy_dir_contents (src_dir, dest_dir, NULL, error))
+ goto out;
+ g_clear_object (&src_dir);
+ g_clear_object (&dest_dir);
+
+ src_dir = g_file_resolve_relative_path (src_repo_dir, "tags");
+ dest_dir = g_file_resolve_relative_path (dest_repo_dir, "tags");
+ if (!copy_dir_contents (src_dir, dest_dir, NULL, error))
+ goto out;
+
+ ret = TRUE;
+ out:
+ if (context)
+ g_option_context_free (context);
+ g_clear_object (&src_repo_dir);
+ g_clear_object (&dest_repo_dir);
+ g_clear_object (&src_info);
+ g_clear_object (&dest_info);
+ g_clear_object (&src_dir);
+ g_clear_object (&dest_dir);
+ g_clear_object (&data.src_repo);
+ g_clear_object (&data.dest_repo);
+ return ret;
+}
diff --git a/src/ostree/ot-builtins.h b/src/ostree/ot-builtins.h
index bd98534..2b98522 100644
--- a/src/ostree/ot-builtins.h
+++ b/src/ostree/ot-builtins.h
@@ -44,6 +44,7 @@ gboolean ostree_builtin_commit (int argc, char **argv, const char *repo, GError
gboolean ostree_builtin_compose (int argc, char **argv, const char *repo, GError **error);
gboolean ostree_builtin_diff (int argc, char **argv, const char *repo, GError **error);
gboolean ostree_builtin_init (int argc, char **argv, const char *repo, GError **error);
+gboolean ostree_builtin_local_clone (int argc, char **argv, const char *repo, GError **error);
gboolean ostree_builtin_log (int argc, char **argv, const char *repo, GError **error);
gboolean ostree_builtin_pull (int argc, char **argv, const char *repo, GError **error);
gboolean ostree_builtin_run_triggers (int argc, char **argv, const char *repo, GError **error);
diff --git a/tests/t0000-basic.sh b/tests/t0000-basic.sh
index df1c3d5..0db1527 100755
--- a/tests/t0000-basic.sh
+++ b/tests/t0000-basic.sh
@@ -19,7 +19,7 @@
set -e
-echo "1..15"
+echo "1..17"
. libtest.sh
@@ -137,3 +137,14 @@ assert_file_has_content ${test_tmpdir}/show 'example.com'
assert_file_has_content ${test_tmpdir}/show 'buildid'
echo "ok metadata content"
+cd ${test_tmpdir}
+mkdir repo2
+ostree --repo=repo2 init
+$OSTREE local-clone repo2
+echo "ok local clone"
+
+cd ${test_tmpdir}
+ostree --repo=repo2 checkout test2 test2-checkout-from-local-clone
+cd test2-checkout-from-local-clone
+assert_file_has_content yet/another/tree/green 'leaf'
+echo "ok local clone checkout"
diff --git a/tests/t0001-archive.sh b/tests/t0001-archive.sh
index 69e9b36..dc7bfa1 100755
--- a/tests/t0001-archive.sh
+++ b/tests/t0001-archive.sh
@@ -21,7 +21,7 @@ set -e
. libtest.sh
-echo '1..3'
+echo '1..5'
setup_test_repository "archive"
echo "ok setup"
@@ -35,3 +35,15 @@ assert_has_file baz/cow
assert_file_has_content baz/cow moo
assert_has_file baz/deeper/ohyeah
echo "ok content"
+
+cd ${test_tmpdir}
+mkdir repo2
+ostree --repo=repo2 init
+$OSTREE local-clone repo2
+echo "ok local clone"
+
+cd ${test_tmpdir}
+ostree --repo=repo2 checkout test2 test2-checkout-from-local-clone
+cd test2-checkout-from-local-clone
+assert_file_has_content baz/cow moo
+echo "ok local clone checkout"
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]