rebase: add coverage of other incompatible options
[alt-git.git] / hook.c
bloba4fa1031f287652eb734b276d9d6c98df40d9a08
1 #include "cache.h"
2 #include "hook.h"
3 #include "run-command.h"
4 #include "config.h"
6 const char *find_hook(const char *name)
8 static struct strbuf path = STRBUF_INIT;
10 strbuf_reset(&path);
11 strbuf_git_path(&path, "hooks/%s", name);
12 if (access(path.buf, X_OK) < 0) {
13 int err = errno;
15 #ifdef STRIP_EXTENSION
16 strbuf_addstr(&path, STRIP_EXTENSION);
17 if (access(path.buf, X_OK) >= 0)
18 return path.buf;
19 if (errno == EACCES)
20 err = errno;
21 #endif
23 if (err == EACCES && advice_enabled(ADVICE_IGNORED_HOOK)) {
24 static struct string_list advise_given = STRING_LIST_INIT_DUP;
26 if (!string_list_lookup(&advise_given, name)) {
27 string_list_insert(&advise_given, name);
28 advise(_("The '%s' hook was ignored because "
29 "it's not set as executable.\n"
30 "You can disable this warning with "
31 "`git config advice.ignoredHook false`."),
32 path.buf);
35 return NULL;
37 return path.buf;
40 int hook_exists(const char *name)
42 return !!find_hook(name);
45 static int pick_next_hook(struct child_process *cp,
46 struct strbuf *out,
47 void *pp_cb,
48 void **pp_task_cb)
50 struct hook_cb_data *hook_cb = pp_cb;
51 const char *hook_path = hook_cb->hook_path;
53 if (!hook_path)
54 return 0;
56 cp->no_stdin = 1;
57 strvec_pushv(&cp->env, hook_cb->options->env.v);
58 cp->stdout_to_stderr = 1;
59 cp->trace2_hook_name = hook_cb->hook_name;
60 cp->dir = hook_cb->options->dir;
62 strvec_push(&cp->args, hook_path);
63 strvec_pushv(&cp->args, hook_cb->options->args.v);
66 * This pick_next_hook() will be called again, we're only
67 * running one hook, so indicate that no more work will be
68 * done.
70 hook_cb->hook_path = NULL;
72 return 1;
75 static int notify_start_failure(struct strbuf *out,
76 void *pp_cb,
77 void *pp_task_cp)
79 struct hook_cb_data *hook_cb = pp_cb;
81 hook_cb->rc |= 1;
83 return 1;
86 static int notify_hook_finished(int result,
87 struct strbuf *out,
88 void *pp_cb,
89 void *pp_task_cb)
91 struct hook_cb_data *hook_cb = pp_cb;
92 struct run_hooks_opt *opt = hook_cb->options;
94 hook_cb->rc |= result;
96 if (opt->invoked_hook)
97 *opt->invoked_hook = 1;
99 return 0;
102 static void run_hooks_opt_clear(struct run_hooks_opt *options)
104 strvec_clear(&options->env);
105 strvec_clear(&options->args);
108 int run_hooks_opt(const char *hook_name, struct run_hooks_opt *options)
110 struct strbuf abs_path = STRBUF_INIT;
111 struct hook_cb_data cb_data = {
112 .rc = 0,
113 .hook_name = hook_name,
114 .options = options,
116 const char *const hook_path = find_hook(hook_name);
117 int ret = 0;
118 const struct run_process_parallel_opts opts = {
119 .tr2_category = "hook",
120 .tr2_label = hook_name,
122 .processes = 1,
123 .ungroup = 1,
125 .get_next_task = pick_next_hook,
126 .start_failure = notify_start_failure,
127 .task_finished = notify_hook_finished,
129 .data = &cb_data,
132 if (!options)
133 BUG("a struct run_hooks_opt must be provided to run_hooks");
135 if (options->invoked_hook)
136 *options->invoked_hook = 0;
138 if (!hook_path && !options->error_if_missing)
139 goto cleanup;
141 if (!hook_path) {
142 ret = error("cannot find a hook named %s", hook_name);
143 goto cleanup;
146 cb_data.hook_path = hook_path;
147 if (options->dir) {
148 strbuf_add_absolute_path(&abs_path, hook_path);
149 cb_data.hook_path = abs_path.buf;
152 run_processes_parallel(&opts);
153 ret = cb_data.rc;
154 cleanup:
155 strbuf_release(&abs_path);
156 run_hooks_opt_clear(options);
157 return ret;
160 int run_hooks(const char *hook_name)
162 struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
164 return run_hooks_opt(hook_name, &opt);
167 int run_hooks_l(const char *hook_name, ...)
169 struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
170 va_list ap;
171 const char *arg;
173 va_start(ap, hook_name);
174 while ((arg = va_arg(ap, const char *)))
175 strvec_push(&opt.args, arg);
176 va_end(ap);
178 return run_hooks_opt(hook_name, &opt);