mirror of
				https://git.proxmox.com/git/libgit2
				synced 2025-10-26 02:22:54 +00:00 
			
		
		
		
	 e069c621bd
			
		
	
	
		e069c621bd
		
	
	
	
	
		
			
			Introduce `git__getenv` which is a UTF-8 aware `getenv` everywhere. Make `cl_getenv` use this to keep consistent memory handling around return values (free everywhere, as opposed to only some platforms).
		
			
				
	
	
		
			301 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			301 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "clar_libgit2.h"
 | |
| #include "fileops.h"
 | |
| #include "sysdir.h"
 | |
| #include "path.h"
 | |
| 
 | |
| #ifdef GIT_WIN32
 | |
| #define NUM_VARS 5
 | |
| static const char *env_vars[NUM_VARS] = {
 | |
| 	"HOME", "HOMEDRIVE", "HOMEPATH", "USERPROFILE", "PROGRAMFILES"
 | |
| };
 | |
| #else
 | |
| #define NUM_VARS 1
 | |
| static const char *env_vars[NUM_VARS] = { "HOME" };
 | |
| #endif
 | |
| 
 | |
| static char *env_save[NUM_VARS];
 | |
| 
 | |
| static char *home_values[] = {
 | |
| 	"fake_home",
 | |
| 	"f\xc3\xa1ke_h\xc3\xb5me", /* all in latin-1 supplement */
 | |
| 	"f\xc4\x80ke_\xc4\xa4ome", /* latin extended */
 | |
| 	"f\xce\xb1\xce\xba\xce\xb5_h\xce\xbfm\xce\xad",  /* having fun with greek */
 | |
| 	"fa\xe0" "\xb8" "\x87" "e_\xe0" "\xb8" "\x99" "ome", /* thai characters */
 | |
| 	"f\xe1\x9c\x80ke_\xe1\x9c\x91ome", /* tagalog characters */
 | |
| 	"\xe1\xb8\x9f\xe1\xba\xa2" "ke_ho" "\xe1" "\xb9" "\x81" "e", /* latin extended additional */
 | |
| 	"\xf0\x9f\x98\x98\xf0\x9f\x98\x82", /* emoticons */
 | |
| 	NULL
 | |
| };
 | |
| 
 | |
| void test_core_env__initialize(void)
 | |
| {
 | |
| 	int i;
 | |
| 	for (i = 0; i < NUM_VARS; ++i)
 | |
| 		env_save[i] = cl_getenv(env_vars[i]);
 | |
| }
 | |
| 
 | |
| static void set_global_search_path_from_env(void)
 | |
| {
 | |
| 	cl_git_pass(git_sysdir_set(GIT_SYSDIR_GLOBAL, NULL));
 | |
| }
 | |
| 
 | |
| static void set_system_search_path_from_env(void)
 | |
| {
 | |
| 	cl_git_pass(git_sysdir_set(GIT_SYSDIR_SYSTEM, NULL));
 | |
| }
 | |
| 
 | |
| void test_core_env__cleanup(void)
 | |
| {
 | |
| 	int i;
 | |
| 	char **val;
 | |
| 
 | |
| 	for (i = 0; i < NUM_VARS; ++i) {
 | |
| 		cl_setenv(env_vars[i], env_save[i]);
 | |
| 		git__free(env_save[i]);
 | |
| 		env_save[i] = NULL;
 | |
| 	}
 | |
| 
 | |
| 	/* these will probably have already been cleaned up, but if a test
 | |
| 	 * fails, then it's probably good to try and clear out these dirs
 | |
| 	 */
 | |
| 	for (val = home_values; *val != NULL; val++) {
 | |
| 		if (**val != '\0')
 | |
| 			(void)p_rmdir(*val);
 | |
| 	}
 | |
| 
 | |
| 	cl_sandbox_set_search_path_defaults();
 | |
| }
 | |
| 
 | |
| static void setenv_and_check(const char *name, const char *value)
 | |
| {
 | |
| 	char *check;
 | |
| 
 | |
| 	cl_git_pass(cl_setenv(name, value));
 | |
| 	check = cl_getenv(name);
 | |
| 
 | |
| 	if (value)
 | |
| 		cl_assert_equal_s(value, check);
 | |
| 	else
 | |
| 		cl_assert(check == NULL);
 | |
| 
 | |
| 	git__free(check);
 | |
| }
 | |
| 
 | |
| void test_core_env__0(void)
 | |
| {
 | |
| 	git_buf path = GIT_BUF_INIT, found = GIT_BUF_INIT;
 | |
| 	char testfile[16], tidx = '0';
 | |
| 	char **val;
 | |
| 	const char *testname = "testfile";
 | |
| 	size_t testlen = strlen(testname);
 | |
| 
 | |
| 	strncpy(testfile, testname, sizeof(testfile));
 | |
| 	cl_assert_equal_s(testname, testfile);
 | |
| 
 | |
| 	for (val = home_values; *val != NULL; val++) {
 | |
| 
 | |
| 		/* if we can't make the directory, let's just assume
 | |
| 		 * we are on a filesystem that doesn't support the
 | |
| 		 * characters in question and skip this test...
 | |
| 		 */
 | |
| 		if (p_mkdir(*val, 0777) != 0) {
 | |
| 			*val = ""; /* mark as not created */
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		cl_git_pass(git_path_prettify(&path, *val, NULL));
 | |
| 
 | |
| 		/* vary testfile name in each directory so accidentally leaving
 | |
| 		 * an environment variable set from a previous iteration won't
 | |
| 		 * accidentally make this test pass...
 | |
| 		 */
 | |
| 		testfile[testlen] = tidx++;
 | |
| 		cl_git_pass(git_buf_joinpath(&path, path.ptr, testfile));
 | |
| 		cl_git_mkfile(path.ptr, "find me");
 | |
| 		git_buf_rtruncate_at_char(&path, '/');
 | |
| 
 | |
| 		cl_assert_equal_i(
 | |
| 			GIT_ENOTFOUND, git_sysdir_find_global_file(&found, testfile));
 | |
| 
 | |
| 		setenv_and_check("HOME", path.ptr);
 | |
| 		set_global_search_path_from_env();
 | |
| 
 | |
| 		cl_git_pass(git_sysdir_find_global_file(&found, testfile));
 | |
| 
 | |
| 		cl_setenv("HOME", env_save[0]);
 | |
| 		set_global_search_path_from_env();
 | |
| 
 | |
| 		cl_assert_equal_i(
 | |
| 			GIT_ENOTFOUND, git_sysdir_find_global_file(&found, testfile));
 | |
| 
 | |
| #ifdef GIT_WIN32
 | |
| 		setenv_and_check("HOMEDRIVE", NULL);
 | |
| 		setenv_and_check("HOMEPATH", NULL);
 | |
| 		setenv_and_check("USERPROFILE", path.ptr);
 | |
| 		set_global_search_path_from_env();
 | |
| 
 | |
| 		cl_git_pass(git_sysdir_find_global_file(&found, testfile));
 | |
| 
 | |
| 		{
 | |
| 			int root = git_path_root(path.ptr);
 | |
| 			char old;
 | |
| 
 | |
| 			if (root >= 0) {
 | |
| 				setenv_and_check("USERPROFILE", NULL);
 | |
| 				set_global_search_path_from_env();
 | |
| 
 | |
| 				cl_assert_equal_i(
 | |
| 					GIT_ENOTFOUND, git_sysdir_find_global_file(&found, testfile));
 | |
| 
 | |
| 				old = path.ptr[root];
 | |
| 				path.ptr[root] = '\0';
 | |
| 				setenv_and_check("HOMEDRIVE", path.ptr);
 | |
| 				path.ptr[root] = old;
 | |
| 				setenv_and_check("HOMEPATH", &path.ptr[root]);
 | |
| 				set_global_search_path_from_env();
 | |
| 
 | |
| 				cl_git_pass(git_sysdir_find_global_file(&found, testfile));
 | |
| 			}
 | |
| 		}
 | |
| #endif
 | |
| 
 | |
| 		(void)p_rmdir(*val);
 | |
| 	}
 | |
| 
 | |
| 	git_buf_free(&path);
 | |
| 	git_buf_free(&found);
 | |
| }
 | |
| 
 | |
| 
 | |
| void test_core_env__1(void)
 | |
| {
 | |
| 	git_buf path = GIT_BUF_INIT;
 | |
| 
 | |
| 	cl_assert_equal_i(
 | |
| 		GIT_ENOTFOUND, git_sysdir_find_global_file(&path, "nonexistentfile"));
 | |
| 
 | |
| 	cl_git_pass(cl_setenv("HOME", "doesnotexist"));
 | |
| #ifdef GIT_WIN32
 | |
| 	cl_git_pass(cl_setenv("HOMEPATH", "doesnotexist"));
 | |
| 	cl_git_pass(cl_setenv("USERPROFILE", "doesnotexist"));
 | |
| #endif
 | |
| 	set_global_search_path_from_env();
 | |
| 
 | |
| 	cl_assert_equal_i(
 | |
| 		GIT_ENOTFOUND, git_sysdir_find_global_file(&path, "nonexistentfile"));
 | |
| 
 | |
| 	cl_git_pass(cl_setenv("HOME", NULL));
 | |
| #ifdef GIT_WIN32
 | |
| 	cl_git_pass(cl_setenv("HOMEPATH", NULL));
 | |
| 	cl_git_pass(cl_setenv("USERPROFILE", NULL));
 | |
| #endif
 | |
| 	set_global_search_path_from_env();
 | |
| 	set_system_search_path_from_env();
 | |
| 
 | |
| 	cl_assert_equal_i(
 | |
| 		GIT_ENOTFOUND, git_sysdir_find_global_file(&path, "nonexistentfile"));
 | |
| 
 | |
| 	cl_assert_equal_i(
 | |
| 		GIT_ENOTFOUND, git_sysdir_find_system_file(&path, "nonexistentfile"));
 | |
| 
 | |
| #ifdef GIT_WIN32
 | |
| 	cl_git_pass(cl_setenv("PROGRAMFILES", NULL));
 | |
| 	set_system_search_path_from_env();
 | |
| 
 | |
| 	cl_assert_equal_i(
 | |
| 		GIT_ENOTFOUND, git_sysdir_find_system_file(&path, "nonexistentfile"));
 | |
| #endif
 | |
| 
 | |
| 	git_buf_free(&path);
 | |
| }
 | |
| 
 | |
| static void check_global_searchpath(
 | |
| 	const char *path, int position, const char *file, git_buf *temp)
 | |
| {
 | |
| 	git_buf out = GIT_BUF_INIT;
 | |
| 
 | |
| 	/* build and set new path */
 | |
| 	if (position < 0)
 | |
| 		cl_git_pass(git_buf_join(temp, GIT_PATH_LIST_SEPARATOR, path, "$PATH"));
 | |
| 	else if (position > 0)
 | |
| 		cl_git_pass(git_buf_join(temp, GIT_PATH_LIST_SEPARATOR, "$PATH", path));
 | |
| 	else
 | |
| 		cl_git_pass(git_buf_sets(temp, path));
 | |
| 
 | |
| 	cl_git_pass(git_libgit2_opts(
 | |
| 		GIT_OPT_SET_SEARCH_PATH, GIT_CONFIG_LEVEL_GLOBAL, temp->ptr));
 | |
| 
 | |
| 	/* get path and make sure $PATH expansion worked */
 | |
| 	cl_git_pass(git_libgit2_opts(
 | |
| 		GIT_OPT_GET_SEARCH_PATH, GIT_CONFIG_LEVEL_GLOBAL, &out));
 | |
| 
 | |
| 	if (position < 0)
 | |
| 		cl_assert(git__prefixcmp(out.ptr, path) == 0);
 | |
| 	else if (position > 0)
 | |
| 		cl_assert(git__suffixcmp(out.ptr, path) == 0);
 | |
| 	else
 | |
| 		cl_assert_equal_s(out.ptr, path);
 | |
| 
 | |
| 	/* find file using new path */
 | |
| 	cl_git_pass(git_sysdir_find_global_file(temp, file));
 | |
| 
 | |
| 	/* reset path and confirm file not found */
 | |
| 	cl_git_pass(git_libgit2_opts(
 | |
| 		GIT_OPT_SET_SEARCH_PATH, GIT_CONFIG_LEVEL_GLOBAL, NULL));
 | |
| 	cl_assert_equal_i(
 | |
| 		GIT_ENOTFOUND, git_sysdir_find_global_file(temp, file));
 | |
| 
 | |
| 	git_buf_free(&out);
 | |
| }
 | |
| 
 | |
| void test_core_env__2(void)
 | |
| {
 | |
| 	git_buf path = GIT_BUF_INIT, found = GIT_BUF_INIT;
 | |
| 	char testfile[16], tidx = '0';
 | |
| 	char **val;
 | |
| 	const char *testname = "alternate";
 | |
| 	size_t testlen = strlen(testname);
 | |
| 
 | |
| 	strncpy(testfile, testname, sizeof(testfile));
 | |
| 	cl_assert_equal_s(testname, testfile);
 | |
| 
 | |
| 	for (val = home_values; *val != NULL; val++) {
 | |
| 
 | |
| 		/* if we can't make the directory, let's just assume
 | |
| 		 * we are on a filesystem that doesn't support the
 | |
| 		 * characters in question and skip this test...
 | |
| 		 */
 | |
| 		if (p_mkdir(*val, 0777) != 0 && errno != EEXIST) {
 | |
| 			*val = ""; /* mark as not created */
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		cl_git_pass(git_path_prettify(&path, *val, NULL));
 | |
| 
 | |
| 		/* vary testfile name so any sloppiness is resetting variables or
 | |
| 		 * deleting files won't accidentally make a test pass.
 | |
| 		 */
 | |
| 		testfile[testlen] = tidx++;
 | |
| 		cl_git_pass(git_buf_joinpath(&path, path.ptr, testfile));
 | |
| 		cl_git_mkfile(path.ptr, "find me");
 | |
| 		git_buf_rtruncate_at_char(&path, '/');
 | |
| 
 | |
| 		/* default should be NOTFOUND */
 | |
| 		cl_assert_equal_i(
 | |
| 			GIT_ENOTFOUND, git_sysdir_find_global_file(&found, testfile));
 | |
| 
 | |
| 		/* try plain, append $PATH, and prepend $PATH */
 | |
| 		check_global_searchpath(path.ptr,  0, testfile, &found);
 | |
| 		check_global_searchpath(path.ptr, -1, testfile, &found);
 | |
| 		check_global_searchpath(path.ptr,  1, testfile, &found);
 | |
| 
 | |
| 		/* cleanup */
 | |
| 		cl_git_pass(git_buf_joinpath(&path, path.ptr, testfile));
 | |
| 		(void)p_unlink(path.ptr);
 | |
| 		(void)p_rmdir(*val);
 | |
| 	}
 | |
| 
 | |
| 	git_buf_free(&path);
 | |
| 	git_buf_free(&found);
 | |
| }
 |