]> git.proxmox.com Git - libgit2.git/commitdiff
stash: add git_stash_drop()
authornulltoken <emeric.fermas@gmail.com>
Mon, 8 Oct 2012 18:07:55 +0000 (20:07 +0200)
committernulltoken <emeric.fermas@gmail.com>
Fri, 26 Oct 2012 20:11:15 +0000 (22:11 +0200)
include/git2/stash.h
src/reflog.c
src/stash.c
tests-clar/stash/drop.c [new file with mode: 0644]

index 6ebf89ed12033a545576826505fa834cf87bd3c2..3ecd9e88d153e2e01ab9d1b701a445ce7e8aa4e4 100644 (file)
@@ -102,6 +102,21 @@ GIT_EXTERN(int) git_stash_foreach(
        stash_cb callback,
        void *payload);
 
+/**
+ * Remove a single stashed state from the stash list.
+ *
+ * @param repo The owning repository.
+ *
+ * @param index The position within the stash list. 0 points to the
+ * most recent stashed state.
+ *
+ * @return 0 on success, or error code
+ */
+
+GIT_EXTERN(int) git_stash_drop(
+       git_repository *repo,
+       size_t index);
+
 /** @} */
 GIT_END_DECL
 #endif
index f0a6e2a8c146b8bc0178246eea13aecd8e983502..5d1465eca57550c09a912d83f1f6ae840e840223 100644 (file)
@@ -481,7 +481,7 @@ int git_reflog_drop(
 
        /* If the oldest entry has just been removed... */
        if (idx == 0) {
-               /* ...clear the oid_old member of the "new" last entry */
+               /* ...clear the oid_old member of the "new" oldest entry */
                if (git_oid_fromstr(&entry->oid_old, GIT_OID_HEX_ZERO) < 0)
                        return -1;
                
index ed0b20f93f59f43adcd19be0efefd8205cf47f4c..3011f00f0ae3d276715e377208751b478e30abae 100644 (file)
@@ -617,3 +617,43 @@ cleanup:
        git_reflog_free(reflog);
        return error;
 }
+
+int git_stash_drop(
+       git_repository *repo,
+       size_t index)
+{
+       git_reference *stash;
+       git_reflog *reflog = NULL;
+       size_t max;
+       int error;
+
+       if ((error = git_reference_lookup(&stash, repo, GIT_REFS_STASH_FILE)) < 0)
+               return error;
+
+       if ((error = git_reflog_read(&reflog, stash)) < 0)
+               goto cleanup;
+
+       max = git_reflog_entrycount(reflog);
+
+       if (index > max - 1) {
+               error = GIT_ENOTFOUND;
+               giterr_set(GITERR_STASH, "No stashed state at position %" PRIuZ, index);
+               goto cleanup;
+       }
+
+       if ((error = git_reflog_drop(reflog, max - index - 1, true)) < 0)
+               goto cleanup;
+
+       if ((error = git_reflog_write(reflog)) < 0)
+               goto cleanup;
+
+       if (max == 1) {
+               error = git_reference_delete(stash);
+               stash = NULL;
+       }
+
+cleanup:
+       git_reference_free(stash);
+       git_reflog_free(reflog);
+       return error;
+}
diff --git a/tests-clar/stash/drop.c b/tests-clar/stash/drop.c
new file mode 100644 (file)
index 0000000..136a942
--- /dev/null
@@ -0,0 +1,126 @@
+#include "clar_libgit2.h"
+#include "fileops.h"
+#include "stash_helpers.h"
+
+static git_repository *repo;
+static git_signature *signature;
+
+void test_stash_drop__initialize(void)
+{
+       cl_git_pass(git_repository_init(&repo, "stash", 0));
+       cl_git_pass(git_signature_new(&signature, "nulltoken", "emeric.fermas@gmail.com", 1323847743, 60)); /* Wed Dec 14 08:29:03 2011 +0100 */
+}
+
+void test_stash_drop__cleanup(void)
+{
+       git_signature_free(signature);
+       git_repository_free(repo);
+       cl_git_pass(git_futils_rmdir_r("stash", NULL, GIT_DIRREMOVAL_FILES_AND_DIRS));
+}
+
+void test_stash_drop__cannot_drop_from_an_empty_stash(void)
+{
+       cl_assert_equal_i(GIT_ENOTFOUND, git_stash_drop(repo, 0));
+}
+
+static void push_three_states(void)
+{
+       git_oid oid;
+       git_index *index;
+
+       cl_git_mkfile("stash/zero.txt", "content\n");
+       cl_git_pass(git_repository_index(&index, repo));
+       cl_git_pass(git_index_add(index, "zero.txt", 0));
+       commit_staged_files(&oid, index, signature);
+
+       cl_git_mkfile("stash/one.txt", "content\n");
+       cl_git_pass(git_stash_save(&oid, repo, signature, "First", GIT_STASH_INCLUDE_UNTRACKED));
+
+       cl_git_mkfile("stash/two.txt", "content\n");
+       cl_git_pass(git_stash_save(&oid, repo, signature, "Second", GIT_STASH_INCLUDE_UNTRACKED));
+
+       cl_git_mkfile("stash/three.txt", "content\n");
+       cl_git_pass(git_stash_save(&oid, repo, signature, "Third", GIT_STASH_INCLUDE_UNTRACKED));
+
+       git_index_free(index);
+}
+
+void test_stash_drop__cannot_drop_a_non_existing_stashed_state(void)
+{
+       push_three_states();
+
+       cl_assert_equal_i(GIT_ENOTFOUND, git_stash_drop(repo, 666));
+       cl_assert_equal_i(GIT_ENOTFOUND, git_stash_drop(repo, 42));
+       cl_assert_equal_i(GIT_ENOTFOUND, git_stash_drop(repo, 3));
+}
+
+void test_stash_drop__can_purge_the_stash_from_the_top(void)
+{
+       push_three_states();
+
+       cl_git_pass(git_stash_drop(repo, 0));
+       cl_git_pass(git_stash_drop(repo, 0));
+       cl_git_pass(git_stash_drop(repo, 0));
+
+       cl_assert_equal_i(GIT_ENOTFOUND, git_stash_drop(repo, 0));
+}
+
+void test_stash_drop__can_purge_the_stash_from_the_bottom(void)
+{
+       push_three_states();
+
+       cl_git_pass(git_stash_drop(repo, 2));
+       cl_git_pass(git_stash_drop(repo, 1));
+       cl_git_pass(git_stash_drop(repo, 0));
+
+       cl_assert_equal_i(GIT_ENOTFOUND, git_stash_drop(repo, 0));
+}
+
+void test_stash_drop__dropping_an_entry_rewrites_reflog_history(void)
+{
+       git_reference *stash;
+       git_reflog *reflog;
+       const git_reflog_entry *entry;
+       git_oid oid;
+       size_t count;
+
+       push_three_states();
+
+       cl_git_pass(git_reference_lookup(&stash, repo, "refs/stash"));
+       
+       cl_git_pass(git_reflog_read(&reflog, stash));
+       entry = git_reflog_entry_byindex(reflog, 1);
+
+       git_oid_cpy(&oid, git_reflog_entry_oidold(entry));
+       count = git_reflog_entrycount(reflog);
+       
+       git_reflog_free(reflog);
+
+       cl_git_pass(git_stash_drop(repo, 1));
+
+       cl_git_pass(git_reflog_read(&reflog, stash));
+       entry = git_reflog_entry_byindex(reflog, 1);
+
+       cl_assert_equal_i(0, git_oid_cmp(&oid, git_reflog_entry_oidold(entry)));
+       cl_assert_equal_i(count - 1, git_reflog_entrycount(reflog));
+
+       git_reflog_free(reflog);
+
+       git_reference_free(stash);
+}
+
+void test_stash_drop__dropping_the_last_entry_removes_the_stash(void)
+{
+       git_reference *stash;
+
+       push_three_states();
+
+       cl_git_pass(git_reference_lookup(&stash, repo, "refs/stash"));
+       git_reference_free(stash);
+
+       cl_git_pass(git_stash_drop(repo, 0));
+       cl_git_pass(git_stash_drop(repo, 0));
+       cl_git_pass(git_stash_drop(repo, 0));
+
+       cl_assert_equal_i(GIT_ENOTFOUND, git_reference_lookup(&stash, repo, "refs/stash"));
+}