190 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			190 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
| #include "git-compat-util.h"
 | |
| #include "abspath.h"
 | |
| #include "advice.h"
 | |
| #include "gettext.h"
 | |
| #include "hook.h"
 | |
| #include "path.h"
 | |
| #include "run-command.h"
 | |
| #include "config.h"
 | |
| #include "strbuf.h"
 | |
| 
 | |
| const char *find_hook(const char *name)
 | |
| {
 | |
| 	static struct strbuf path = STRBUF_INIT;
 | |
| 
 | |
| 	strbuf_reset(&path);
 | |
| 	strbuf_git_path(&path, "hooks/%s", name);
 | |
| 	if (access(path.buf, X_OK) < 0) {
 | |
| 		int err = errno;
 | |
| 
 | |
| #ifdef STRIP_EXTENSION
 | |
| 		strbuf_addstr(&path, STRIP_EXTENSION);
 | |
| 		if (access(path.buf, X_OK) >= 0)
 | |
| 			return path.buf;
 | |
| 		if (errno == EACCES)
 | |
| 			err = errno;
 | |
| #endif
 | |
| 
 | |
| 		if (err == EACCES && advice_enabled(ADVICE_IGNORED_HOOK)) {
 | |
| 			static struct string_list advise_given = STRING_LIST_INIT_DUP;
 | |
| 
 | |
| 			if (!string_list_lookup(&advise_given, name)) {
 | |
| 				string_list_insert(&advise_given, name);
 | |
| 				advise(_("The '%s' hook was ignored because "
 | |
| 					 "it's not set as executable.\n"
 | |
| 					 "You can disable this warning with "
 | |
| 					 "`git config advice.ignoredHook false`."),
 | |
| 				       path.buf);
 | |
| 			}
 | |
| 		}
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	return path.buf;
 | |
| }
 | |
| 
 | |
| int hook_exists(const char *name)
 | |
| {
 | |
| 	return !!find_hook(name);
 | |
| }
 | |
| 
 | |
| static int pick_next_hook(struct child_process *cp,
 | |
| 			  struct strbuf *out UNUSED,
 | |
| 			  void *pp_cb,
 | |
| 			  void **pp_task_cb UNUSED)
 | |
| {
 | |
| 	struct hook_cb_data *hook_cb = pp_cb;
 | |
| 	const char *hook_path = hook_cb->hook_path;
 | |
| 
 | |
| 	if (!hook_path)
 | |
| 		return 0;
 | |
| 
 | |
| 	cp->no_stdin = 1;
 | |
| 	strvec_pushv(&cp->env, hook_cb->options->env.v);
 | |
| 	/* reopen the file for stdin; run_command closes it. */
 | |
| 	if (hook_cb->options->path_to_stdin) {
 | |
| 		cp->no_stdin = 0;
 | |
| 		cp->in = xopen(hook_cb->options->path_to_stdin, O_RDONLY);
 | |
| 	}
 | |
| 	cp->stdout_to_stderr = 1;
 | |
| 	cp->trace2_hook_name = hook_cb->hook_name;
 | |
| 	cp->dir = hook_cb->options->dir;
 | |
| 
 | |
| 	strvec_push(&cp->args, hook_path);
 | |
| 	strvec_pushv(&cp->args, hook_cb->options->args.v);
 | |
| 
 | |
| 	/*
 | |
| 	 * This pick_next_hook() will be called again, we're only
 | |
| 	 * running one hook, so indicate that no more work will be
 | |
| 	 * done.
 | |
| 	 */
 | |
| 	hook_cb->hook_path = NULL;
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static int notify_start_failure(struct strbuf *out UNUSED,
 | |
| 				void *pp_cb,
 | |
| 				void *pp_task_cp UNUSED)
 | |
| {
 | |
| 	struct hook_cb_data *hook_cb = pp_cb;
 | |
| 
 | |
| 	hook_cb->rc |= 1;
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static int notify_hook_finished(int result,
 | |
| 				struct strbuf *out UNUSED,
 | |
| 				void *pp_cb,
 | |
| 				void *pp_task_cb UNUSED)
 | |
| {
 | |
| 	struct hook_cb_data *hook_cb = pp_cb;
 | |
| 	struct run_hooks_opt *opt = hook_cb->options;
 | |
| 
 | |
| 	hook_cb->rc |= result;
 | |
| 
 | |
| 	if (opt->invoked_hook)
 | |
| 		*opt->invoked_hook = 1;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void run_hooks_opt_clear(struct run_hooks_opt *options)
 | |
| {
 | |
| 	strvec_clear(&options->env);
 | |
| 	strvec_clear(&options->args);
 | |
| }
 | |
| 
 | |
| int run_hooks_opt(const char *hook_name, struct run_hooks_opt *options)
 | |
| {
 | |
| 	struct strbuf abs_path = STRBUF_INIT;
 | |
| 	struct hook_cb_data cb_data = {
 | |
| 		.rc = 0,
 | |
| 		.hook_name = hook_name,
 | |
| 		.options = options,
 | |
| 	};
 | |
| 	const char *const hook_path = find_hook(hook_name);
 | |
| 	int ret = 0;
 | |
| 	const struct run_process_parallel_opts opts = {
 | |
| 		.tr2_category = "hook",
 | |
| 		.tr2_label = hook_name,
 | |
| 
 | |
| 		.processes = 1,
 | |
| 		.ungroup = 1,
 | |
| 
 | |
| 		.get_next_task = pick_next_hook,
 | |
| 		.start_failure = notify_start_failure,
 | |
| 		.task_finished = notify_hook_finished,
 | |
| 
 | |
| 		.data = &cb_data,
 | |
| 	};
 | |
| 
 | |
| 	if (!options)
 | |
| 		BUG("a struct run_hooks_opt must be provided to run_hooks");
 | |
| 
 | |
| 	if (options->invoked_hook)
 | |
| 		*options->invoked_hook = 0;
 | |
| 
 | |
| 	if (!hook_path && !options->error_if_missing)
 | |
| 		goto cleanup;
 | |
| 
 | |
| 	if (!hook_path) {
 | |
| 		ret = error("cannot find a hook named %s", hook_name);
 | |
| 		goto cleanup;
 | |
| 	}
 | |
| 
 | |
| 	cb_data.hook_path = hook_path;
 | |
| 	if (options->dir) {
 | |
| 		strbuf_add_absolute_path(&abs_path, hook_path);
 | |
| 		cb_data.hook_path = abs_path.buf;
 | |
| 	}
 | |
| 
 | |
| 	run_processes_parallel(&opts);
 | |
| 	ret = cb_data.rc;
 | |
| cleanup:
 | |
| 	strbuf_release(&abs_path);
 | |
| 	run_hooks_opt_clear(options);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| int run_hooks(const char *hook_name)
 | |
| {
 | |
| 	struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
 | |
| 
 | |
| 	return run_hooks_opt(hook_name, &opt);
 | |
| }
 | |
| 
 | |
| int run_hooks_l(const char *hook_name, ...)
 | |
| {
 | |
| 	struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
 | |
| 	va_list ap;
 | |
| 	const char *arg;
 | |
| 
 | |
| 	va_start(ap, hook_name);
 | |
| 	while ((arg = va_arg(ap, const char *)))
 | |
| 		strvec_push(&opt.args, arg);
 | |
| 	va_end(ap);
 | |
| 
 | |
| 	return run_hooks_opt(hook_name, &opt);
 | |
| }
 |