Blame src/sysdir.c

Packit Service 20376f
/*
Packit Service 20376f
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit Service 20376f
 *
Packit Service 20376f
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit Service 20376f
 * a Linking Exception. For full terms see the included COPYING file.
Packit Service 20376f
 */
Packit Service 20376f
Packit Service 20376f
#include "common.h"
Packit Service 20376f
#include "sysdir.h"
Packit Service 20376f
#include "global.h"
Packit Service 20376f
#include "buffer.h"
Packit Service 20376f
#include "path.h"
Packit Service 20376f
#include <ctype.h>
Packit Service 20376f
#if GIT_WIN32
Packit Service 20376f
#include "win32/findfile.h"
Packit Service 20376f
#endif
Packit Service 20376f
Packit Service 20376f
static int git_sysdir_guess_programdata_dirs(git_buf *out)
Packit Service 20376f
{
Packit Service 20376f
#ifdef GIT_WIN32
Packit Service 20376f
	return git_win32__find_programdata_dirs(out);
Packit Service 20376f
#else
Packit Service 20376f
	git_buf_clear(out);
Packit Service 20376f
	return 0;
Packit Service 20376f
#endif
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int git_sysdir_guess_system_dirs(git_buf *out)
Packit Service 20376f
{
Packit Service 20376f
#ifdef GIT_WIN32
Packit Service 20376f
	return git_win32__find_system_dirs(out, L"etc\\");
Packit Service 20376f
#else
Packit Service 20376f
	return git_buf_sets(out, "/etc");
Packit Service 20376f
#endif
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int git_sysdir_guess_global_dirs(git_buf *out)
Packit Service 20376f
{
Packit Service 20376f
#ifdef GIT_WIN32
Packit Service 20376f
	return git_win32__find_global_dirs(out);
Packit Service 20376f
#else
Packit Service 20376f
	int error = git__getenv(out, "HOME");
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_ENOTFOUND) {
Packit Service 20376f
		giterr_clear();
Packit Service 20376f
		error = 0;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
#endif
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int git_sysdir_guess_xdg_dirs(git_buf *out)
Packit Service 20376f
{
Packit Service 20376f
#ifdef GIT_WIN32
Packit Service 20376f
	return git_win32__find_xdg_dirs(out);
Packit Service 20376f
#else
Packit Service 20376f
	git_buf env = GIT_BUF_INIT;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if ((error = git__getenv(&env, "XDG_CONFIG_HOME")) == 0)
Packit Service 20376f
		error = git_buf_joinpath(out, env.ptr, "git");
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_ENOTFOUND && (error = git__getenv(&env, "HOME")) == 0)
Packit Service 20376f
		error = git_buf_joinpath(out, env.ptr, ".config/git");
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_ENOTFOUND) {
Packit Service 20376f
		giterr_clear();
Packit Service 20376f
		error = 0;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	git_buf_free(&env;;
Packit Service 20376f
	return error;
Packit Service 20376f
#endif
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int git_sysdir_guess_template_dirs(git_buf *out)
Packit Service 20376f
{
Packit Service 20376f
#ifdef GIT_WIN32
Packit Service 20376f
	return git_win32__find_system_dirs(out, L"share\\git-core\\templates");
Packit Service 20376f
#else
Packit Service 20376f
	return git_buf_sets(out, "/usr/share/git-core/templates");
Packit Service 20376f
#endif
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
struct git_sysdir__dir {
Packit Service 20376f
	git_buf buf;
Packit Service 20376f
	int (*guess)(git_buf *out);
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
static struct git_sysdir__dir git_sysdir__dirs[] = {
Packit Service 20376f
	{ GIT_BUF_INIT, git_sysdir_guess_system_dirs },
Packit Service 20376f
	{ GIT_BUF_INIT, git_sysdir_guess_global_dirs },
Packit Service 20376f
	{ GIT_BUF_INIT, git_sysdir_guess_xdg_dirs },
Packit Service 20376f
	{ GIT_BUF_INIT, git_sysdir_guess_programdata_dirs },
Packit Service 20376f
	{ GIT_BUF_INIT, git_sysdir_guess_template_dirs },
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
static void git_sysdir_global_shutdown(void)
Packit Service 20376f
{
Packit Service 20376f
	size_t i;
Packit Service 20376f
Packit Service 20376f
	for (i = 0; i < ARRAY_SIZE(git_sysdir__dirs); ++i)
Packit Service 20376f
		git_buf_free(&git_sysdir__dirs[i].buf);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_global_init(void)
Packit Service 20376f
{
Packit Service 20376f
	size_t i;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	for (i = 0; !error && i < ARRAY_SIZE(git_sysdir__dirs); i++)
Packit Service 20376f
		error = git_sysdir__dirs[i].guess(&git_sysdir__dirs[i].buf);
Packit Service 20376f
Packit Service 20376f
	git__on_shutdown(git_sysdir_global_shutdown);
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int git_sysdir_check_selector(git_sysdir_t which)
Packit Service 20376f
{
Packit Service 20376f
	if (which < ARRAY_SIZE(git_sysdir__dirs))
Packit Service 20376f
		return 0;
Packit Service 20376f
Packit Service 20376f
	giterr_set(GITERR_INVALID, "config directory selector out of range");
Packit Service 20376f
	return -1;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
int git_sysdir_get(const git_buf **out, git_sysdir_t which)
Packit Service 20376f
{
Packit Service 20376f
	assert(out);
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
Packit Service 20376f
	GITERR_CHECK_ERROR(git_sysdir_check_selector(which));
Packit Service 20376f
Packit Service 20376f
	*out = &git_sysdir__dirs[which].buf;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_get_str(
Packit Service 20376f
	char *out,
Packit Service 20376f
	size_t outlen,
Packit Service 20376f
	git_sysdir_t which)
Packit Service 20376f
{
Packit Service 20376f
	const git_buf *path = NULL;
Packit Service 20376f
Packit Service 20376f
	GITERR_CHECK_ERROR(git_sysdir_check_selector(which));
Packit Service 20376f
	GITERR_CHECK_ERROR(git_sysdir_get(&path, which));
Packit Service 20376f
Packit Service 20376f
	if (!out || path->size >= outlen) {
Packit Service 20376f
		giterr_set(GITERR_NOMEMORY, "buffer is too short for the path");
Packit Service 20376f
		return GIT_EBUFS;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	git_buf_copy_cstr(out, outlen, path);
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
#define PATH_MAGIC "$PATH"
Packit Service 20376f
Packit Service 20376f
int git_sysdir_set(git_sysdir_t which, const char *search_path)
Packit Service 20376f
{
Packit Service 20376f
	const char *expand_path = NULL;
Packit Service 20376f
	git_buf merge = GIT_BUF_INIT;
Packit Service 20376f
Packit Service 20376f
	GITERR_CHECK_ERROR(git_sysdir_check_selector(which));
Packit Service 20376f
Packit Service 20376f
	if (search_path != NULL)
Packit Service 20376f
		expand_path = strstr(search_path, PATH_MAGIC);
Packit Service 20376f
Packit Service 20376f
	/* reset the default if this path has been cleared */
Packit Service 20376f
	if (!search_path)
Packit Service 20376f
		git_sysdir__dirs[which].guess(&git_sysdir__dirs[which].buf);
Packit Service 20376f
Packit Service 20376f
	/* if $PATH is not referenced, then just set the path */
Packit Service 20376f
	if (!expand_path) {
Packit Service 20376f
		if (search_path)
Packit Service 20376f
			git_buf_sets(&git_sysdir__dirs[which].buf, search_path);
Packit Service 20376f
Packit Service 20376f
		goto done;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* otherwise set to join(before $PATH, old value, after $PATH) */
Packit Service 20376f
	if (expand_path > search_path)
Packit Service 20376f
		git_buf_set(&merge, search_path, expand_path - search_path);
Packit Service 20376f
Packit Service 20376f
	if (git_buf_len(&git_sysdir__dirs[which].buf))
Packit Service 20376f
		git_buf_join(&merge, GIT_PATH_LIST_SEPARATOR,
Packit Service 20376f
			merge.ptr, git_sysdir__dirs[which].buf.ptr);
Packit Service 20376f
Packit Service 20376f
	expand_path += strlen(PATH_MAGIC);
Packit Service 20376f
	if (*expand_path)
Packit Service 20376f
		git_buf_join(&merge, GIT_PATH_LIST_SEPARATOR, merge.ptr, expand_path);
Packit Service 20376f
Packit Service 20376f
	git_buf_swap(&git_sysdir__dirs[which].buf, &merge);
Packit Service 20376f
	git_buf_free(&merge);
Packit Service 20376f
Packit Service 20376f
done:
Packit Service 20376f
	if (git_buf_oom(&git_sysdir__dirs[which].buf))
Packit Service 20376f
		return -1;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int git_sysdir_find_in_dirlist(
Packit Service 20376f
	git_buf *path,
Packit Service 20376f
	const char *name,
Packit Service 20376f
	git_sysdir_t which,
Packit Service 20376f
	const char *label)
Packit Service 20376f
{
Packit Service 20376f
	size_t len;
Packit Service 20376f
	const char *scan, *next = NULL;
Packit Service 20376f
	const git_buf *syspath;
Packit Service 20376f
Packit Service 20376f
	GITERR_CHECK_ERROR(git_sysdir_get(&syspath, which));
Packit Service 20376f
	if (!syspath || !git_buf_len(syspath))
Packit Service 20376f
		goto done;
Packit Service 20376f
Packit Service 20376f
	for (scan = git_buf_cstr(syspath); scan; scan = next) {
Packit Service 20376f
		/* find unescaped separator or end of string */
Packit Service 20376f
		for (next = scan; *next; ++next) {
Packit Service 20376f
			if (*next == GIT_PATH_LIST_SEPARATOR &&
Packit Service 20376f
				(next <= scan || next[-1] != '\\'))
Packit Service 20376f
				break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		len = (size_t)(next - scan);
Packit Service 20376f
		next = (*next ? next + 1 : NULL);
Packit Service 20376f
		if (!len)
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		GITERR_CHECK_ERROR(git_buf_set(path, scan, len));
Packit Service 20376f
		if (name)
Packit Service 20376f
			GITERR_CHECK_ERROR(git_buf_joinpath(path, path->ptr, name));
Packit Service 20376f
Packit Service 20376f
		if (git_path_exists(path->ptr))
Packit Service 20376f
			return 0;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
done:
Packit Service 20376f
	git_buf_free(path);
Packit Service 20376f
	giterr_set(GITERR_OS, "the %s file '%s' doesn't exist", label, name);
Packit Service 20376f
	return GIT_ENOTFOUND;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_find_system_file(git_buf *path, const char *filename)
Packit Service 20376f
{
Packit Service 20376f
	return git_sysdir_find_in_dirlist(
Packit Service 20376f
		path, filename, GIT_SYSDIR_SYSTEM, "system");
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_find_global_file(git_buf *path, const char *filename)
Packit Service 20376f
{
Packit Service 20376f
	return git_sysdir_find_in_dirlist(
Packit Service 20376f
		path, filename, GIT_SYSDIR_GLOBAL, "global");
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_find_xdg_file(git_buf *path, const char *filename)
Packit Service 20376f
{
Packit Service 20376f
	return git_sysdir_find_in_dirlist(
Packit Service 20376f
		path, filename, GIT_SYSDIR_XDG, "global/xdg");
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_find_programdata_file(git_buf *path, const char *filename)
Packit Service 20376f
{
Packit Service 20376f
	return git_sysdir_find_in_dirlist(
Packit Service 20376f
		path, filename, GIT_SYSDIR_PROGRAMDATA, "ProgramData");
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_find_template_dir(git_buf *path)
Packit Service 20376f
{
Packit Service 20376f
	return git_sysdir_find_in_dirlist(
Packit Service 20376f
		path, NULL, GIT_SYSDIR_TEMPLATE, "template");
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_sysdir_expand_global_file(git_buf *path, const char *filename)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_sysdir_find_global_file(path, NULL)) == 0) {
Packit Service 20376f
		if (filename)
Packit Service 20376f
			error = git_buf_joinpath(path, path->ptr, filename);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}