mirror of
https://git.proxmox.com/git/libgit2
synced 2025-05-04 00:55:04 +00:00

If we remove a reference while we're iterating through the packed refs, the position in the iterator will be off.
222 lines
4.8 KiB
C
222 lines
4.8 KiB
C
#include "clar_libgit2.h"
|
|
#include "refs.h"
|
|
#include "vector.h"
|
|
|
|
static git_repository *repo;
|
|
|
|
void test_refs_iterator__initialize(void)
|
|
{
|
|
cl_git_pass(git_repository_open(&repo, cl_fixture("testrepo.git")));
|
|
}
|
|
|
|
void test_refs_iterator__cleanup(void)
|
|
{
|
|
git_repository_free(repo);
|
|
}
|
|
|
|
static const char *refnames[] = {
|
|
"refs/heads/br2",
|
|
"refs/heads/cannot-fetch",
|
|
"refs/heads/chomped",
|
|
"refs/heads/haacked",
|
|
"refs/heads/master",
|
|
"refs/heads/not-good",
|
|
"refs/heads/packed",
|
|
"refs/heads/packed-test",
|
|
"refs/heads/subtrees",
|
|
"refs/heads/test",
|
|
"refs/heads/track-local",
|
|
"refs/heads/trailing",
|
|
"refs/notes/fanout",
|
|
"refs/remotes/test/master",
|
|
"refs/tags/annotated_tag_to_blob",
|
|
"refs/tags/e90810b",
|
|
"refs/tags/hard_tag",
|
|
"refs/tags/point_to_blob",
|
|
"refs/tags/taggerless",
|
|
"refs/tags/test",
|
|
"refs/tags/wrapped_tag",
|
|
};
|
|
|
|
static int refcmp_cb(const void *a, const void *b)
|
|
{
|
|
const git_reference *refa = (const git_reference *)a;
|
|
const git_reference *refb = (const git_reference *)b;
|
|
|
|
return strcmp(refa->name, refb->name);
|
|
}
|
|
|
|
static void assert_all_refnames_match(git_vector *output)
|
|
{
|
|
size_t i;
|
|
git_reference *ref;
|
|
|
|
cl_assert_equal_sz(output->length, ARRAY_SIZE(refnames));
|
|
|
|
git_vector_sort(output);
|
|
|
|
git_vector_foreach(output, i, ref) {
|
|
cl_assert_equal_s(ref->name, refnames[i]);
|
|
git_reference_free(ref);
|
|
}
|
|
|
|
git_vector_free(output);
|
|
}
|
|
|
|
void test_refs_iterator__list(void)
|
|
{
|
|
git_reference_iterator *iter;
|
|
git_vector output;
|
|
git_reference *ref;
|
|
|
|
cl_git_pass(git_vector_init(&output, 32, &refcmp_cb));
|
|
cl_git_pass(git_reference_iterator_new(&iter, repo));
|
|
|
|
while (1) {
|
|
int error = git_reference_next(&ref, iter);
|
|
if (error == GIT_ITEROVER)
|
|
break;
|
|
cl_git_pass(error);
|
|
cl_git_pass(git_vector_insert(&output, ref));
|
|
}
|
|
|
|
git_reference_iterator_free(iter);
|
|
|
|
assert_all_refnames_match(&output);
|
|
}
|
|
|
|
void test_refs_iterator__empty(void)
|
|
{
|
|
git_reference_iterator *iter;
|
|
git_odb *odb;
|
|
git_reference *ref;
|
|
git_repository *empty;
|
|
|
|
cl_git_pass(git_odb_new(&odb));
|
|
cl_git_pass(git_repository_wrap_odb(&empty, odb));
|
|
|
|
cl_git_pass(git_reference_iterator_new(&iter, empty));
|
|
cl_assert_equal_i(GIT_ITEROVER, git_reference_next(&ref, iter));
|
|
|
|
git_reference_iterator_free(iter);
|
|
git_odb_free(odb);
|
|
git_repository_free(empty);
|
|
}
|
|
|
|
static int refs_foreach_cb(git_reference *reference, void *payload)
|
|
{
|
|
git_vector *output = payload;
|
|
cl_git_pass(git_vector_insert(output, reference));
|
|
return 0;
|
|
}
|
|
|
|
void test_refs_iterator__foreach(void)
|
|
{
|
|
git_vector output;
|
|
cl_git_pass(git_vector_init(&output, 32, &refcmp_cb));
|
|
cl_git_pass(git_reference_foreach(repo, refs_foreach_cb, &output));
|
|
assert_all_refnames_match(&output);
|
|
}
|
|
|
|
static int refs_foreach_cancel_cb(git_reference *reference, void *payload)
|
|
{
|
|
int *cancel_after = payload;
|
|
|
|
git_reference_free(reference);
|
|
|
|
if (!*cancel_after)
|
|
return -333;
|
|
(*cancel_after)--;
|
|
return 0;
|
|
}
|
|
|
|
void test_refs_iterator__foreach_can_cancel(void)
|
|
{
|
|
int cancel_after = 3;
|
|
cl_git_fail_with(
|
|
git_reference_foreach(repo, refs_foreach_cancel_cb, &cancel_after),
|
|
-333);
|
|
cl_assert_equal_i(0, cancel_after);
|
|
}
|
|
|
|
static int refs_foreach_name_cb(const char *name, void *payload)
|
|
{
|
|
git_vector *output = payload;
|
|
cl_git_pass(git_vector_insert(output, git__strdup(name)));
|
|
return 0;
|
|
}
|
|
|
|
void test_refs_iterator__foreach_name(void)
|
|
{
|
|
git_vector output;
|
|
size_t i;
|
|
char *name;
|
|
|
|
cl_git_pass(git_vector_init(&output, 32, &git__strcmp_cb));
|
|
cl_git_pass(
|
|
git_reference_foreach_name(repo, refs_foreach_name_cb, &output));
|
|
|
|
cl_assert_equal_sz(output.length, ARRAY_SIZE(refnames));
|
|
git_vector_sort(&output);
|
|
|
|
git_vector_foreach(&output, i, name) {
|
|
cl_assert_equal_s(name, refnames[i]);
|
|
git__free(name);
|
|
}
|
|
|
|
git_vector_free(&output);
|
|
}
|
|
|
|
static int refs_foreach_name_cancel_cb(const char *name, void *payload)
|
|
{
|
|
int *cancel_after = payload;
|
|
if (!*cancel_after)
|
|
return -333;
|
|
GIT_UNUSED(name);
|
|
(*cancel_after)--;
|
|
return 0;
|
|
}
|
|
|
|
void test_refs_iterator__foreach_name_can_cancel(void)
|
|
{
|
|
int cancel_after = 5;
|
|
cl_git_fail_with(
|
|
git_reference_foreach_name(
|
|
repo, refs_foreach_name_cancel_cb, &cancel_after),
|
|
-333);
|
|
cl_assert_equal_i(0, cancel_after);
|
|
}
|
|
|
|
void test_refs_iterator__concurrent_delete(void)
|
|
{
|
|
git_reference_iterator *iter;
|
|
size_t full_count = 0, concurrent_count = 0;
|
|
const char *name;
|
|
int error;
|
|
|
|
git_repository_free(repo);
|
|
repo = cl_git_sandbox_init("testrepo");
|
|
|
|
cl_git_pass(git_reference_iterator_new(&iter, repo));
|
|
while ((error = git_reference_next_name(&name, iter)) == 0) {
|
|
full_count++;
|
|
}
|
|
|
|
git_reference_iterator_free(iter);
|
|
cl_assert_equal_i(GIT_ITEROVER, error);
|
|
|
|
cl_git_pass(git_reference_iterator_new(&iter, repo));
|
|
while ((error = git_reference_next_name(&name, iter)) == 0) {
|
|
cl_git_pass(git_reference_remove(repo, name));
|
|
concurrent_count++;
|
|
}
|
|
|
|
git_reference_iterator_free(iter);
|
|
cl_assert_equal_i(GIT_ITEROVER, error);
|
|
|
|
cl_assert_equal_i(full_count, concurrent_count);
|
|
|
|
cl_git_sandbox_cleanup();
|
|
repo = NULL;
|
|
}
|