diff options
| -rw-r--r-- | include/git2/checkout.h | 2 | ||||
| -rw-r--r-- | src/checkout.c | 25 | ||||
| -rw-r--r-- | src/iterator.c | 12 | ||||
| -rw-r--r-- | src/iterator.h | 15 | ||||
| -rw-r--r-- | tests-clar/checkout/index.c | 28 | 
5 files changed, 69 insertions, 13 deletions
| diff --git a/include/git2/checkout.h b/include/git2/checkout.h index f49e87566..a086408c7 100644 --- a/include/git2/checkout.h +++ b/include/git2/checkout.h @@ -236,6 +236,8 @@ typedef struct git_checkout_opts {  	git_strarray paths;  	git_tree *baseline; /** expected content of workdir, defaults to HEAD */ + +	const char *target_directory; /** alternative checkout path to workdir */  } git_checkout_opts;  #define GIT_CHECKOUT_OPTS_VERSION 1 diff --git a/src/checkout.c b/src/checkout.c index 065bb50fd..e3ae38710 100644 --- a/src/checkout.c +++ b/src/checkout.c @@ -858,7 +858,7 @@ static int checkout_submodule(  		return 0;  	if ((error = git_futils_mkdir( -			file->path, git_repository_workdir(data->repo), +			file->path, data->opts.target_directory,  			data->opts.dir_mode, GIT_MKDIR_PATH)) < 0)  		return error; @@ -1030,7 +1030,7 @@ static int checkout_deferred_remove(git_repository *repo, const char *path)  {  #if 0  	int error = git_futils_rmdir_r( -		path, git_repository_workdir(repo), GIT_RMDIR_EMPTY_PARENTS); +		path, data->opts.target_directory, GIT_RMDIR_EMPTY_PARENTS);  	if (error == GIT_ENOTFOUND) {  		error = 0; @@ -1163,7 +1163,8 @@ static int checkout_data_init(  		return -1;  	} -	if ((error = git_repository__ensure_not_bare(repo, "checkout")) < 0) +	if ((!proposed || !proposed->target_directory) && +		(error = git_repository__ensure_not_bare(repo, "checkout")) < 0)  		return error;  	data->repo = repo; @@ -1176,6 +1177,13 @@ static int checkout_data_init(  	else  		memmove(&data->opts, proposed, sizeof(git_checkout_opts)); +	if (!data->opts.target_directory) +		data->opts.target_directory = git_repository_workdir(repo); +	else if (!git_path_isdir(data->opts.target_directory) && +			 (error = git_futils_mkdir(data->opts.target_directory, NULL, +					GIT_DIR_MODE, GIT_MKDIR_VERIFY_DIR)) < 0) +		goto cleanup; +  	/* refresh config and index content unless NO_REFRESH is given */  	if ((data->opts.checkout_strategy & GIT_CHECKOUT_NO_REFRESH) == 0) {  		git_config *cfg; @@ -1238,7 +1246,8 @@ static int checkout_data_init(  	if ((error = git_vector_init(&data->removes, 0, git__strcmp_cb)) < 0 ||  		(error = git_pool_init(&data->pool, 1, 0)) < 0 || -		(error = git_buf_puts(&data->path, git_repository_workdir(repo))) < 0) +		(error = git_buf_puts(&data->path, data->opts.target_directory)) < 0 || +		(error = git_path_to_dir(&data->path)) < 0)  		goto cleanup;  	data->workdir_len = git_buf_len(&data->path); @@ -1286,11 +1295,13 @@ int git_checkout_iterator(  		GIT_ITERATOR_IGNORE_CASE : GIT_ITERATOR_DONT_IGNORE_CASE;  	if ((error = git_iterator_reset(target, data.pfx, data.pfx)) < 0 || -		(error = git_iterator_for_workdir( -			&workdir, data.repo, iterflags | GIT_ITERATOR_DONT_AUTOEXPAND, +		(error = git_iterator_for_workdir_ext( +			&workdir, data.repo, data.opts.target_directory, +			iterflags | GIT_ITERATOR_DONT_AUTOEXPAND,  			data.pfx, data.pfx)) < 0 ||  		(error = git_iterator_for_tree( -			&baseline, data.opts.baseline, iterflags, data.pfx, data.pfx)) < 0) +			&baseline, data.opts.baseline, +			iterflags, data.pfx, data.pfx)) < 0)  		goto cleanup;  	/* Should not have case insensitivity mismatch */ diff --git a/src/iterator.c b/src/iterator.c index 76b0e41d0..5917f63fd 100644 --- a/src/iterator.c +++ b/src/iterator.c @@ -1321,9 +1321,10 @@ static void workdir_iterator__free(git_iterator *self)  	git_ignore__free(&wi->ignores);  } -int git_iterator_for_workdir( +int git_iterator_for_workdir_ext(  	git_iterator **out,  	git_repository *repo, +	const char *repo_workdir,  	git_iterator_flag_t flags,  	const char *start,  	const char *end) @@ -1331,8 +1332,11 @@ int git_iterator_for_workdir(  	int error;  	workdir_iterator *wi; -	if (git_repository__ensure_not_bare(repo, "scan working directory") < 0) -		return GIT_EBAREREPO; +	if (!repo_workdir) { +		if (git_repository__ensure_not_bare(repo, "scan working directory") < 0) +			return GIT_EBAREREPO; +		repo_workdir = git_repository_workdir(repo); +	}  	/* initialize as an fs iterator then do overrides */  	wi = git__calloc(1, sizeof(workdir_iterator)); @@ -1352,7 +1356,7 @@ int git_iterator_for_workdir(  		return error;  	} -	return fs_iterator__initialize(out, &wi->fi, git_repository_workdir(repo)); +	return fs_iterator__initialize(out, &wi->fi, repo_workdir);  } diff --git a/src/iterator.h b/src/iterator.h index 493ff4b2a..ea88fa6a2 100644 --- a/src/iterator.h +++ b/src/iterator.h @@ -79,15 +79,26 @@ extern int git_iterator_for_index(  	const char *start,  	const char *end); +extern int git_iterator_for_workdir_ext( +	git_iterator **out, +	git_repository *repo, +	const char *repo_workdir, +	git_iterator_flag_t flags, +	const char *start, +	const char *end); +  /* workdir iterators will match the ignore_case value from the index of the   * repository, unless you override with a non-zero flag value   */ -extern int git_iterator_for_workdir( +GIT_INLINE(int) git_iterator_for_workdir(  	git_iterator **out,  	git_repository *repo,  	git_iterator_flag_t flags,  	const char *start, -	const char *end); +	const char *end) +{ +	return git_iterator_for_workdir_ext(out, repo, NULL, flags, start, end); +}  /* for filesystem iterators, you have to explicitly pass in the ignore_case   * behavior that you desire diff --git a/tests-clar/checkout/index.c b/tests-clar/checkout/index.c index a3a0f8fda..16584ce22 100644 --- a/tests-clar/checkout/index.c +++ b/tests-clar/checkout/index.c @@ -506,3 +506,31 @@ void test_checkout_index__issue_1397(void)  	check_file_contents("./issue_1397/crlf_file.txt", "first line\r\nsecond line\r\nboth with crlf");  } + +void test_checkout_index__target_directory(void) +{ +	git_checkout_opts opts = GIT_CHECKOUT_OPTS_INIT; +	checkout_counts cts; +	memset(&cts, 0, sizeof(cts)); + +	opts.checkout_strategy = GIT_CHECKOUT_SAFE_CREATE; +	opts.target_directory = "alternative"; + +	opts.notify_flags = GIT_CHECKOUT_NOTIFY_ALL; +	opts.notify_cb = checkout_count_callback; +	opts.notify_payload = &cts; + +	/* create some files that *would* conflict if we were using the wd */ +	cl_git_mkfile("testrepo/README", "I'm in the way!\n"); +	cl_git_mkfile("testrepo/new.txt", "my new file\n"); + +	cl_git_pass(git_checkout_index(g_repo, NULL, &opts)); + +	cl_assert_equal_i(0, cts.n_untracked); +	cl_assert_equal_i(0, cts.n_ignored); +	cl_assert_equal_i(4, cts.n_updates); + +	check_file_contents("./alternative/README", "hey there\n"); +	check_file_contents("./alternative/branch_file.txt", "hi\nbye!\n"); +	check_file_contents("./alternative/new.txt", "my new file\n"); +} | 
