Blame tests/checkout/typechange.c

Packit ae9e2a
#include "clar_libgit2.h"
Packit ae9e2a
#include "git2/checkout.h"
Packit ae9e2a
#include "path.h"
Packit ae9e2a
#include "posix.h"
Packit ae9e2a
#include "fileops.h"
Packit ae9e2a
Packit ae9e2a
static git_repository *g_repo = NULL;
Packit ae9e2a
Packit ae9e2a
/*
Packit ae9e2a
From the test repo used for this test:
Packit ae9e2a
--------------------------------------
Packit ae9e2a
Packit ae9e2a
This is a test repo for libgit2 where tree entries have type changes
Packit ae9e2a
Packit ae9e2a
The key types that could be found in tree entries are:
Packit ae9e2a
Packit ae9e2a
1 - GIT_FILEMODE_NEW             = 0000000
Packit ae9e2a
2 - GIT_FILEMODE_TREE            = 0040000
Packit ae9e2a
3 - GIT_FILEMODE_BLOB            = 0100644
Packit ae9e2a
4 - GIT_FILEMODE_BLOB_EXECUTABLE = 0100755
Packit ae9e2a
5 - GIT_FILEMODE_LINK            = 0120000
Packit ae9e2a
6 - GIT_FILEMODE_COMMIT          = 0160000
Packit ae9e2a
Packit ae9e2a
I will try to have every type of transition somewhere in the history
Packit ae9e2a
of this repo.
Packit ae9e2a
Packit ae9e2a
Commits
Packit ae9e2a
-------
Packit ae9e2a
Initial commit - a(1)    b(1)    c(1)    d(1)    e(1)
Packit ae9e2a
Create content - a(1->2) b(1->3) c(1->4) d(1->5) e(1->6)
Packit ae9e2a
Changes #1     - a(2->3) b(3->4) c(4->5) d(5->6) e(6->2)
Packit ae9e2a
Changes #2     - a(3->5) b(4->6) c(5->2) d(6->3) e(2->4)
Packit ae9e2a
Changes #3     - a(5->3) b(6->4) c(2->5) d(3->6) e(4->2)
Packit ae9e2a
Changes #4     - a(3->2) b(4->3) c(5->4) d(6->5) e(2->6)
Packit ae9e2a
Changes #5     - a(2->1) b(3->1) c(4->1) d(5->1) e(6->1)
Packit ae9e2a
Packit ae9e2a
*/
Packit ae9e2a
Packit ae9e2a
static const char *g_typechange_oids[] = {
Packit ae9e2a
	"79b9f23e85f55ea36a472a902e875bc1121a94cb",
Packit ae9e2a
	"9bdb75b73836a99e3dbeea640a81de81031fdc29",
Packit ae9e2a
	"0e7ed140b514b8cae23254cb8656fe1674403aff",
Packit ae9e2a
	"9d0235c7a7edc0889a18f97a42ee6db9fe688447",
Packit ae9e2a
	"9b19edf33a03a0c59cdfc113bfa5c06179bf9b1a",
Packit ae9e2a
	"1b63caae4a5ca96f78e8dfefc376c6a39a142475",
Packit ae9e2a
	"6eae26c90e8ccc4d16208972119c40635489c6f0",
Packit ae9e2a
	NULL
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
static bool g_typechange_empty[] = {
Packit ae9e2a
	true, false, false, false, false, false, true, true
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
static const int g_typechange_expected_conflicts[] = {
Packit ae9e2a
	1, 2, 3, 3, 2, 3, 2
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
static const int g_typechange_expected_untracked[] = {
Packit ae9e2a
	6, 4, 3, 2, 3, 2, 5
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
void test_checkout_typechange__initialize(void)
Packit ae9e2a
{
Packit ae9e2a
	g_repo = cl_git_sandbox_init("typechanges");
Packit ae9e2a
Packit ae9e2a
	cl_fixture_sandbox("submod2_target");
Packit ae9e2a
	p_rename("submod2_target/.gitted", "submod2_target/.git");
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void test_checkout_typechange__cleanup(void)
Packit ae9e2a
{
Packit ae9e2a
	cl_git_sandbox_cleanup();
Packit ae9e2a
	cl_fixture_cleanup("submod2_target");
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static void assert_file_exists(const char *path)
Packit ae9e2a
{
Packit ae9e2a
	cl_assert_(git_path_isfile(path), path);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static void assert_dir_exists(const char *path)
Packit ae9e2a
{
Packit ae9e2a
	cl_assert_(git_path_isdir(path), path);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static void assert_workdir_matches_tree(
Packit ae9e2a
	git_repository *repo, const git_oid *id, const char *root, bool recurse)
Packit ae9e2a
{
Packit ae9e2a
	git_object *obj;
Packit ae9e2a
	git_tree *tree;
Packit ae9e2a
	size_t i, max_i;
Packit ae9e2a
	git_buf path = GIT_BUF_INIT;
Packit ae9e2a
Packit ae9e2a
	if (!root)
Packit ae9e2a
		root = git_repository_workdir(repo);
Packit ae9e2a
	cl_assert(root);
Packit ae9e2a
Packit ae9e2a
	cl_git_pass(git_object_lookup(&obj, repo, id, GIT_OBJ_ANY));
Packit ae9e2a
	cl_git_pass(git_object_peel((git_object **)&tree, obj, GIT_OBJ_TREE));
Packit ae9e2a
	git_object_free(obj);
Packit ae9e2a
Packit ae9e2a
	max_i = git_tree_entrycount(tree);
Packit ae9e2a
Packit ae9e2a
	for (i = 0; i < max_i; ++i) {
Packit ae9e2a
		const git_tree_entry *te = git_tree_entry_byindex(tree, i);
Packit ae9e2a
		cl_assert(te);
Packit ae9e2a
Packit ae9e2a
		cl_git_pass(git_buf_joinpath(&path, root, git_tree_entry_name(te)));
Packit ae9e2a
Packit ae9e2a
		switch (git_tree_entry_type(te)) {
Packit ae9e2a
		case GIT_OBJ_COMMIT:
Packit ae9e2a
			assert_dir_exists(path.ptr);
Packit ae9e2a
			break;
Packit ae9e2a
		case GIT_OBJ_TREE:
Packit ae9e2a
			assert_dir_exists(path.ptr);
Packit ae9e2a
			if (recurse)
Packit ae9e2a
				assert_workdir_matches_tree(
Packit ae9e2a
					repo, git_tree_entry_id(te), path.ptr, true);
Packit ae9e2a
			break;
Packit ae9e2a
		case GIT_OBJ_BLOB:
Packit ae9e2a
			switch (git_tree_entry_filemode(te)) {
Packit ae9e2a
			case GIT_FILEMODE_BLOB:
Packit ae9e2a
			case GIT_FILEMODE_BLOB_EXECUTABLE:
Packit ae9e2a
				assert_file_exists(path.ptr);
Packit ae9e2a
				/* because of cross-platform, don't confirm exec bit yet */
Packit ae9e2a
				break;
Packit ae9e2a
			case GIT_FILEMODE_LINK:
Packit ae9e2a
				cl_assert_(git_path_exists(path.ptr), path.ptr);
Packit ae9e2a
				/* because of cross-platform, don't confirm link yet */
Packit ae9e2a
				break;
Packit ae9e2a
			default:
Packit ae9e2a
				cl_assert(false); /* really?! */
Packit ae9e2a
			}
Packit ae9e2a
			break;
Packit ae9e2a
		default:
Packit ae9e2a
			cl_assert(false); /* really?!! */
Packit ae9e2a
		}
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	git_tree_free(tree);
Packit ae9e2a
	git_buf_free(&path);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void test_checkout_typechange__checkout_typechanges_safe(void)
Packit ae9e2a
{
Packit ae9e2a
	int i;
Packit ae9e2a
	git_object *obj;
Packit ae9e2a
	git_checkout_options opts = GIT_CHECKOUT_OPTIONS_INIT;
Packit ae9e2a
Packit ae9e2a
	for (i = 0; g_typechange_oids[i] != NULL; ++i) {
Packit ae9e2a
		cl_git_pass(git_revparse_single(&obj, g_repo, g_typechange_oids[i]));
Packit ae9e2a
Packit ae9e2a
		opts.checkout_strategy = !i ? GIT_CHECKOUT_FORCE : GIT_CHECKOUT_SAFE;
Packit ae9e2a
Packit ae9e2a
		cl_git_pass(git_checkout_tree(g_repo, obj, &opts));
Packit ae9e2a
Packit ae9e2a
		cl_git_pass(
Packit ae9e2a
			git_repository_set_head_detached(g_repo, git_object_id(obj)));
Packit ae9e2a
Packit ae9e2a
		assert_workdir_matches_tree(g_repo, git_object_id(obj), NULL, true);
Packit ae9e2a
Packit ae9e2a
		git_object_free(obj);
Packit ae9e2a
Packit ae9e2a
		if (!g_typechange_empty[i]) {
Packit ae9e2a
			cl_assert(git_path_isdir("typechanges"));
Packit ae9e2a
			cl_assert(git_path_exists("typechanges/a"));
Packit ae9e2a
			cl_assert(git_path_exists("typechanges/b"));
Packit ae9e2a
			cl_assert(git_path_exists("typechanges/c"));
Packit ae9e2a
			cl_assert(git_path_exists("typechanges/d"));
Packit ae9e2a
			cl_assert(git_path_exists("typechanges/e"));
Packit ae9e2a
		} else {
Packit ae9e2a
			cl_assert(git_path_isdir("typechanges"));
Packit ae9e2a
			cl_assert(!git_path_exists("typechanges/a"));
Packit ae9e2a
			cl_assert(!git_path_exists("typechanges/b"));
Packit ae9e2a
			cl_assert(!git_path_exists("typechanges/c"));
Packit ae9e2a
			cl_assert(!git_path_exists("typechanges/d"));
Packit ae9e2a
			cl_assert(!git_path_exists("typechanges/e"));
Packit ae9e2a
		}
Packit ae9e2a
	}
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
typedef struct {
Packit ae9e2a
	int conflicts;
Packit ae9e2a
	int dirty;
Packit ae9e2a
	int updates;
Packit ae9e2a
	int untracked;
Packit ae9e2a
	int ignored;
Packit ae9e2a
} notify_counts;
Packit ae9e2a
Packit ae9e2a
static int notify_counter(
Packit ae9e2a
	git_checkout_notify_t why,
Packit ae9e2a
	const char *path,
Packit ae9e2a
	const git_diff_file *baseline,
Packit ae9e2a
	const git_diff_file *target,
Packit ae9e2a
	const git_diff_file *workdir,
Packit ae9e2a
	void *payload)
Packit ae9e2a
{
Packit ae9e2a
	notify_counts *cts = payload;
Packit ae9e2a
Packit ae9e2a
	GIT_UNUSED(path);
Packit ae9e2a
	GIT_UNUSED(baseline);
Packit ae9e2a
	GIT_UNUSED(target);
Packit ae9e2a
	GIT_UNUSED(workdir);
Packit ae9e2a
Packit ae9e2a
	switch (why) {
Packit ae9e2a
	case GIT_CHECKOUT_NOTIFY_CONFLICT:  cts->conflicts++; break;
Packit ae9e2a
	case GIT_CHECKOUT_NOTIFY_DIRTY:     cts->dirty++;     break;
Packit ae9e2a
	case GIT_CHECKOUT_NOTIFY_UPDATED:   cts->updates++;   break;
Packit ae9e2a
	case GIT_CHECKOUT_NOTIFY_UNTRACKED: cts->untracked++; break;
Packit ae9e2a
	case GIT_CHECKOUT_NOTIFY_IGNORED:   cts->ignored++;   break;
Packit ae9e2a
	default: break;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static void force_create_file(const char *file)
Packit ae9e2a
{
Packit ae9e2a
	int error = git_futils_rmdir_r(file, NULL,
Packit ae9e2a
		GIT_RMDIR_REMOVE_FILES | GIT_RMDIR_REMOVE_BLOCKERS);
Packit ae9e2a
	cl_assert(!error || error == GIT_ENOTFOUND);
Packit ae9e2a
	cl_git_pass(git_futils_mkpath2file(file, 0777));
Packit ae9e2a
	cl_git_rewritefile(file, "yowza!!");
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int make_submodule_dirty(git_submodule *sm, const char *name, void *payload)
Packit ae9e2a
{
Packit ae9e2a
	git_buf submodulepath = GIT_BUF_INIT;
Packit ae9e2a
	git_buf dirtypath = GIT_BUF_INIT;
Packit ae9e2a
	git_repository *submodule_repo;
Packit ae9e2a
Packit ae9e2a
	GIT_UNUSED(name);
Packit ae9e2a
	GIT_UNUSED(payload);
Packit ae9e2a
Packit ae9e2a
	/* remove submodule directory in preparation for init and repo_init */
Packit ae9e2a
	cl_git_pass(git_buf_joinpath(
Packit ae9e2a
		&submodulepath,
Packit ae9e2a
		git_repository_workdir(g_repo),
Packit ae9e2a
		git_submodule_path(sm)
Packit ae9e2a
	));
Packit ae9e2a
	git_futils_rmdir_r(git_buf_cstr(&submodulepath), NULL, GIT_RMDIR_REMOVE_FILES);
Packit ae9e2a
Packit ae9e2a
	/* initialize submodule's repository */
Packit ae9e2a
	cl_git_pass(git_submodule_repo_init(&submodule_repo, sm, 0));
Packit ae9e2a
Packit ae9e2a
	/* create a file in the submodule workdir to make it dirty */
Packit ae9e2a
	cl_git_pass(
Packit ae9e2a
		git_buf_joinpath(&dirtypath, git_repository_workdir(submodule_repo), "dirty"));
Packit ae9e2a
	force_create_file(git_buf_cstr(&dirtypath));
Packit ae9e2a
Packit ae9e2a
	git_buf_free(&dirtypath);
Packit ae9e2a
	git_buf_free(&submodulepath);
Packit ae9e2a
	git_repository_free(submodule_repo);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void test_checkout_typechange__checkout_with_conflicts(void)
Packit ae9e2a
{
Packit ae9e2a
	int i;
Packit ae9e2a
	git_object *obj;
Packit ae9e2a
	git_checkout_options opts = GIT_CHECKOUT_OPTIONS_INIT;
Packit ae9e2a
	notify_counts cts = {0};
Packit ae9e2a
Packit ae9e2a
	opts.notify_flags =
Packit ae9e2a
		GIT_CHECKOUT_NOTIFY_CONFLICT | GIT_CHECKOUT_NOTIFY_UNTRACKED;
Packit ae9e2a
	opts.notify_cb = notify_counter;
Packit ae9e2a
	opts.notify_payload = &cts;
Packit ae9e2a
Packit ae9e2a
	for (i = 0; g_typechange_oids[i] != NULL; ++i) {
Packit ae9e2a
		cl_git_pass(git_revparse_single(&obj, g_repo, g_typechange_oids[i]));
Packit ae9e2a
Packit ae9e2a
		force_create_file("typechanges/a/blocker");
Packit ae9e2a
		force_create_file("typechanges/b");
Packit ae9e2a
		force_create_file("typechanges/c/sub/sub/file");
Packit ae9e2a
		git_futils_rmdir_r("typechanges/d", NULL, GIT_RMDIR_REMOVE_FILES);
Packit ae9e2a
		p_mkdir("typechanges/d", 0777); /* intentionally empty dir */
Packit ae9e2a
		force_create_file("typechanges/untracked");
Packit ae9e2a
		cl_git_pass(git_submodule_foreach(g_repo, make_submodule_dirty, NULL));
Packit ae9e2a
Packit ae9e2a
		opts.checkout_strategy = GIT_CHECKOUT_SAFE;
Packit ae9e2a
		memset(&cts, 0, sizeof(cts));
Packit ae9e2a
Packit ae9e2a
		cl_git_fail(git_checkout_tree(g_repo, obj, &opts));
Packit ae9e2a
		cl_assert_equal_i(cts.conflicts, g_typechange_expected_conflicts[i]);
Packit ae9e2a
		cl_assert_equal_i(cts.untracked, g_typechange_expected_untracked[i]);
Packit ae9e2a
		cl_assert_equal_i(cts.dirty, 0);
Packit ae9e2a
		cl_assert_equal_i(cts.updates, 0);
Packit ae9e2a
		cl_assert_equal_i(cts.ignored, 0);
Packit ae9e2a
Packit ae9e2a
		opts.checkout_strategy =
Packit ae9e2a
			GIT_CHECKOUT_FORCE | GIT_CHECKOUT_REMOVE_UNTRACKED;
Packit ae9e2a
		memset(&cts, 0, sizeof(cts));
Packit ae9e2a
Packit ae9e2a
		cl_assert(git_path_exists("typechanges/untracked"));
Packit ae9e2a
Packit ae9e2a
		cl_git_pass(git_checkout_tree(g_repo, obj, &opts));
Packit ae9e2a
		cl_assert_equal_i(0, cts.conflicts);
Packit ae9e2a
Packit ae9e2a
		cl_assert(!git_path_exists("typechanges/untracked"));
Packit ae9e2a
Packit ae9e2a
		cl_git_pass(
Packit ae9e2a
			git_repository_set_head_detached(g_repo, git_object_id(obj)));
Packit ae9e2a
Packit ae9e2a
		assert_workdir_matches_tree(g_repo, git_object_id(obj), NULL, true);
Packit ae9e2a
Packit ae9e2a
		git_object_free(obj);
Packit ae9e2a
	}
Packit ae9e2a
}