[hacktree] checkout: New command
- From: Colin Walters <walters src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [hacktree] checkout: New command
- Date: Sat, 15 Oct 2011 04:45:44 +0000 (UTC)
commit 401ab27c11d6ab8a70449c7223e35d6c34e0586f
Author: Colin Walters <walters verbum org>
Date: Sat Oct 15 00:45:07 2011 -0400
checkout: New command
Makefile-hacktree.am | 1 +
src/ht-builtin-checkout.c | 81 ++++++++++++++++++++
src/ht-builtins.h | 1 +
src/libhacktree/hacktree-repo.c | 159 ++++++++++++++++++++++++++++++++++++++-
src/libhacktree/hacktree-repo.h | 2 +-
src/main.c | 1 +
tests/libtest.sh | 17 ++++
tests/t0004-checkout-test1.sh | 36 +++++++++
8 files changed, 296 insertions(+), 2 deletions(-)
---
diff --git a/Makefile-hacktree.am b/Makefile-hacktree.am
index d3e757d..3dfa8ca 100644
--- a/Makefile-hacktree.am
+++ b/Makefile-hacktree.am
@@ -46,6 +46,7 @@ bin_PROGRAMS += hacktree
hacktree_SOURCES = src/main.c \
src/ht-builtins.h \
+ src/ht-builtin-checkout.c \
src/ht-builtin-commit.c \
src/ht-builtin-fsck.c \
src/ht-builtin-init.c \
diff --git a/src/ht-builtin-checkout.c b/src/ht-builtin-checkout.c
new file mode 100644
index 0000000..5ed32b9
--- /dev/null
+++ b/src/ht-builtin-checkout.c
@@ -0,0 +1,81 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
+ *
+ * Copyright (C) 2011 Colin Walters <walters verbum org>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; 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 "ht-builtins.h"
+#include "hacktree.h"
+
+#include <glib/gi18n.h>
+
+static char *repo_path;
+
+static GOptionEntry options[] = {
+ { "repo", 0, 0, G_OPTION_ARG_FILENAME, &repo_path, "Repository path", "repo" },
+ { NULL }
+};
+
+gboolean
+hacktree_builtin_checkout (int argc, char **argv, const char *prefix, GError **error)
+{
+ GOptionContext *context;
+ gboolean ret = FALSE;
+ HacktreeRepo *repo = NULL;
+ int i;
+ const char *commit;
+ const char *destination;
+
+ context = g_option_context_new ("COMMIT DESTINATION - Check out a commit into a filesystem tree");
+ g_option_context_add_main_entries (context, options, NULL);
+
+ if (!g_option_context_parse (context, &argc, &argv, error))
+ goto out;
+
+ if (repo_path == NULL)
+ repo_path = ".";
+
+ repo = hacktree_repo_new (repo_path);
+ if (!hacktree_repo_check (repo, error))
+ goto out;
+
+ if (argc < 3)
+ {
+ 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,
+ "COMMIT and DESTINATION must be specified");
+ goto out;
+ }
+
+ commit = argv[1];
+ destination = argv[2];
+
+ if (!hacktree_repo_checkout (repo, commit, destination, error))
+ goto out;
+
+ ret = TRUE;
+ out:
+ if (context)
+ g_option_context_free (context);
+ g_clear_object (&repo);
+ return ret;
+}
diff --git a/src/ht-builtins.h b/src/ht-builtins.h
index da5a286..e1db9c1 100644
--- a/src/ht-builtins.h
+++ b/src/ht-builtins.h
@@ -36,6 +36,7 @@ typedef struct {
int flags; /* HacktreeBuiltinFlags */
} HacktreeBuiltin;
+gboolean hacktree_builtin_checkout (int argc, char **argv, const char *prefix, GError **error);
gboolean hacktree_builtin_commit (int argc, char **argv, const char *prefix, GError **error);
gboolean hacktree_builtin_init (int argc, char **argv, const char *prefix, GError **error);
gboolean hacktree_builtin_link_file (int argc, char **argv, const char *prefix, GError **error);
diff --git a/src/libhacktree/hacktree-repo.c b/src/libhacktree/hacktree-repo.c
index ae72545..57fb466 100644
--- a/src/libhacktree/hacktree-repo.c
+++ b/src/libhacktree/hacktree-repo.c
@@ -425,7 +425,7 @@ import_directory_meta (HacktreeRepo *self,
HACKTREE_DIR_META_VERSION,
(guint32)stbuf.st_uid,
(guint32)stbuf.st_gid,
- (guint32)(stbuf.st_mode & ~S_IFMT),
+ (guint32)(stbuf.st_mode & (S_ISUID|S_ISGID|S_ISVTX|S_IRWXU|S_IRWXG|S_IRWXO)),
g_variant_new_fixed_array (G_VARIANT_TYPE ("y"),
xattrs, xattr_len, 1));
g_variant_ref_sink (dirmeta);
@@ -1424,3 +1424,160 @@ hacktree_repo_get_head (HacktreeRepo *self)
return priv->current_head;
}
+
+static gboolean
+resolve_ref (HacktreeRepo *self,
+ const char *ref,
+ char **resolved,
+ GError **error)
+{
+ if (strcmp (ref, "HEAD") == 0)
+ {
+ *resolved = g_strdup (hacktree_repo_get_head (self));
+ return TRUE;
+ }
+ else if (strlen (ref) == 64)
+ {
+ *resolved = g_strdup (ref);
+ return TRUE;
+ }
+ g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
+ "Invalid ref '%s' (must be SHA256 or HEAD)", ref);
+ return FALSE;
+}
+
+
+static gboolean
+checkout_tree (HacktreeRepo *self,
+ ParsedTreeData *tree,
+ const char *destination,
+ GError **error);
+
+static gboolean
+checkout_one_directory (HacktreeRepo *self,
+ const char *destination,
+ const char *dirname,
+ ParsedDirectoryData *dir,
+ GError **error)
+{
+ gboolean ret = FALSE;
+ char *dest_path = NULL;
+ guint32 version, uid, gid, mode;
+ GVariant *xattr_variant = NULL;
+ const guint8 *xattrs = NULL;
+ gsize xattr_len;
+
+ dest_path = g_build_filename (destination, dirname, NULL);
+
+ g_variant_get (dir->meta_data, "(uuuu ay)",
+ &version, &uid, &gid, &mode,
+ &xattr_variant);
+ xattrs = g_variant_get_fixed_array (xattr_variant, &xattr_len, 1);
+
+ if (mkdir (dest_path, (mode_t)mode) < 0)
+ {
+ ht_util_set_error_from_errno (error, errno);
+ goto out;
+ }
+
+ if (!checkout_tree (self, dir->tree_data, dest_path, error))
+ goto out;
+
+ /* TODO - xattrs */
+
+ ret = TRUE;
+ out:
+ g_free (dest_path);
+ g_variant_unref (xattr_variant);
+ return ret;
+}
+
+static gboolean
+checkout_tree (HacktreeRepo *self,
+ ParsedTreeData *tree,
+ const char *destination,
+ GError **error)
+{
+ gboolean ret = FALSE;
+ GHashTableIter hash_iter;
+ gpointer key, value;
+
+ g_hash_table_iter_init (&hash_iter, tree->files);
+ while (g_hash_table_iter_next (&hash_iter, &key, &value))
+ {
+ const char *filename = key;
+ const char *checksum = value;
+ char *object_path;
+ char *dest_path;
+
+ object_path = get_object_path (self, checksum, HACKTREE_OBJECT_TYPE_FILE);
+ dest_path = g_build_filename (destination, filename, NULL);
+ if (link (object_path, dest_path) < 0)
+ {
+ ht_util_set_error_from_errno (error, errno);
+ g_free (object_path);
+ g_free (dest_path);
+ goto out;
+ }
+ g_free (object_path);
+ g_free (dest_path);
+ }
+
+ g_hash_table_iter_init (&hash_iter, tree->directories);
+ while (g_hash_table_iter_next (&hash_iter, &key, &value))
+ {
+ const char *dirname = key;
+ ParsedDirectoryData *dir = value;
+
+ if (!checkout_one_directory (self, destination, dirname, dir, error))
+ goto out;
+ }
+
+ ret = TRUE;
+ out:
+ return ret;
+}
+
+gboolean
+hacktree_repo_checkout (HacktreeRepo *self,
+ const char *ref,
+ const char *destination,
+ GError **error)
+{
+ gboolean ret = FALSE;
+ GVariant *commit = NULL;
+ char *resolved = NULL;
+ ParsedTreeData *tree = NULL;
+
+ if (g_file_test (destination, G_FILE_TEST_EXISTS))
+ {
+ g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
+ "Destination path '%s' already exists",
+ destination);
+ goto out;
+ }
+
+ if (!resolve_ref (self, ref, &resolved, error))
+ goto out;
+
+ /* FIXME - perms etc on root directory */
+ if (mkdir (destination, (mode_t)0755) < 0)
+ {
+ ht_util_set_error_from_errno (error, errno);
+ goto out;
+ }
+
+ if (!load_commit_and_trees (self, resolved, &commit, &tree, error))
+ goto out;
+
+ if (!checkout_tree (self, tree, destination, error))
+ goto out;
+
+ ret = TRUE;
+ out:
+ g_free (resolved);
+ if (commit)
+ g_variant_unref (commit);
+ parsed_tree_data_free (tree);
+ return ret;
+}
diff --git a/src/libhacktree/hacktree-repo.h b/src/libhacktree/hacktree-repo.h
index c401dd2..d9d0eee 100644
--- a/src/libhacktree/hacktree-repo.h
+++ b/src/libhacktree/hacktree-repo.h
@@ -78,7 +78,7 @@ gboolean hacktree_repo_commit (HacktreeRepo *self,
GError **error);
gboolean hacktree_repo_checkout (HacktreeRepo *self,
- const char *commit,
+ const char *ref,
const char *destination,
GError **error);
diff --git a/src/main.c b/src/main.c
index c3cafb8..6ccb158 100644
--- a/src/main.c
+++ b/src/main.c
@@ -29,6 +29,7 @@
#include "ht-builtins.h"
static HacktreeBuiltin builtins[] = {
+ { "checkout", hacktree_builtin_checkout, 0 },
{ "init", hacktree_builtin_init, 0 },
{ "commit", hacktree_builtin_commit, 0 },
{ "link-file", hacktree_builtin_link_file, 0 },
diff --git a/tests/libtest.sh b/tests/libtest.sh
index 7fc8ed1..1d565a5 100644
--- a/tests/libtest.sh
+++ b/tests/libtest.sh
@@ -32,4 +32,21 @@ die () {
fi
}
+setup_test_repository1 () {
+ mkdir files
+ cd files
+ ht_files=`pwd`
+ export ht_files
+ echo first > firstfile
+ echo second > secondfile
+
+ mkdir ../repo
+ ht_repo="--repo=../repo"
+ export ht_repo
+ hacktree init $ht_repo
+ hacktree commit $ht_repo -s "Test Commit 1" -b "Commit body first" --add=firstfile
+ hacktree commit $ht_repo -s "Test Commit 2" -b "Commit body first" --add=secondfile
+ hacktree fsck -q $ht_repo
+}
+
trap 'die' EXIT
diff --git a/tests/t0004-checkout-test1.sh b/tests/t0004-checkout-test1.sh
new file mode 100755
index 0000000..1244dda
--- /dev/null
+++ b/tests/t0004-checkout-test1.sh
@@ -0,0 +1,36 @@
+#!/bin/bash
+#
+# Copyright (C) 2011 Colin Walters <walters verbum org>
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program 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 General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; 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>
+
+set -e
+
+. libtest.sh
+
+echo '1..3'
+
+setup_test_repository1
+echo 'ok setup'
+hacktree checkout $ht_repo HEAD $test_tmpdir/checkout1-head
+echo 'ok checkout cmd'
+cd $test_tmpdir/checkout1-head
+test -f firstfile
+test -f secondfile
+echo 'ok checkout verify exists'
+
+
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]