blob: 1a8483186349b00608d40bc3d8702aa39ffb79c8 [file] [log] [blame]
Ævar Arnfjörð Bjarmason5e3aba32021-09-26 21:03:26 +02001#include "cache.h"
2#include "hook.h"
3#include "run-command.h"
Emily Shaffer96e72252021-12-22 04:59:27 +01004#include "config.h"
Ævar Arnfjörð Bjarmason5e3aba32021-09-26 21:03:26 +02005
6const char *find_hook(const char *name)
7{
8 static struct strbuf path = STRBUF_INIT;
9
10 strbuf_reset(&path);
11 strbuf_git_path(&path, "hooks/%s", name);
12 if (access(path.buf, X_OK) < 0) {
13 int err = errno;
14
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
22
23 if (err == EACCES && advice_enabled(ADVICE_IGNORED_HOOK)) {
24 static struct string_list advise_given = STRING_LIST_INIT_DUP;
25
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);
33 }
34 }
35 return NULL;
36 }
37 return path.buf;
38}
Emily Shaffer330155e2021-09-26 21:03:27 +020039
40int hook_exists(const char *name)
41{
42 return !!find_hook(name);
43}
Emily Shaffer96e72252021-12-22 04:59:27 +010044
45static int pick_next_hook(struct child_process *cp,
46 struct strbuf *out,
47 void *pp_cb,
48 void **pp_task_cb)
49{
50 struct hook_cb_data *hook_cb = pp_cb;
51 const char *hook_path = hook_cb->hook_path;
52
53 if (!hook_path)
54 return 0;
55
56 cp->no_stdin = 1;
Ævar Arnfjörð Bjarmason29fda242022-06-02 11:09:50 +020057 strvec_pushv(&cp->env, hook_cb->options->env.v);
Emily Shaffer917e0802023-02-08 20:21:13 +010058 /* reopen the file for stdin; run_command closes it. */
59 if (hook_cb->options->path_to_stdin) {
60 cp->no_stdin = 0;
61 cp->in = xopen(hook_cb->options->path_to_stdin, O_RDONLY);
62 }
Emily Shaffer96e72252021-12-22 04:59:27 +010063 cp->stdout_to_stderr = 1;
64 cp->trace2_hook_name = hook_cb->hook_name;
Emily Shaffer1a3017d2021-12-22 04:59:36 +010065 cp->dir = hook_cb->options->dir;
Emily Shaffer96e72252021-12-22 04:59:27 +010066
67 strvec_push(&cp->args, hook_path);
68 strvec_pushv(&cp->args, hook_cb->options->args.v);
69
Emily Shaffer96e72252021-12-22 04:59:27 +010070 /*
71 * This pick_next_hook() will be called again, we're only
72 * running one hook, so indicate that no more work will be
73 * done.
74 */
75 hook_cb->hook_path = NULL;
76
77 return 1;
78}
79
80static int notify_start_failure(struct strbuf *out,
81 void *pp_cb,
82 void *pp_task_cp)
83{
84 struct hook_cb_data *hook_cb = pp_cb;
Emily Shaffer96e72252021-12-22 04:59:27 +010085
86 hook_cb->rc |= 1;
87
Emily Shaffer96e72252021-12-22 04:59:27 +010088 return 1;
89}
90
91static int notify_hook_finished(int result,
92 struct strbuf *out,
93 void *pp_cb,
94 void *pp_task_cb)
95{
96 struct hook_cb_data *hook_cb = pp_cb;
Ævar Arnfjörð Bjarmasona8cc5942022-03-07 13:33:46 +010097 struct run_hooks_opt *opt = hook_cb->options;
Emily Shaffer96e72252021-12-22 04:59:27 +010098
99 hook_cb->rc |= result;
100
Ævar Arnfjörð Bjarmasona8cc5942022-03-07 13:33:46 +0100101 if (opt->invoked_hook)
102 *opt->invoked_hook = 1;
103
Emily Shaffer96e72252021-12-22 04:59:27 +0100104 return 0;
105}
106
107static void run_hooks_opt_clear(struct run_hooks_opt *options)
108{
109 strvec_clear(&options->env);
110 strvec_clear(&options->args);
111}
112
113int run_hooks_opt(const char *hook_name, struct run_hooks_opt *options)
114{
Emily Shaffer1a3017d2021-12-22 04:59:36 +0100115 struct strbuf abs_path = STRBUF_INIT;
Emily Shaffer96e72252021-12-22 04:59:27 +0100116 struct hook_cb_data cb_data = {
117 .rc = 0,
118 .hook_name = hook_name,
119 .options = options,
120 };
121 const char *const hook_path = find_hook(hook_name);
Emily Shaffer96e72252021-12-22 04:59:27 +0100122 int ret = 0;
Ævar Arnfjörð Bjarmason6e5ba0b2022-10-12 23:02:26 +0200123 const struct run_process_parallel_opts opts = {
124 .tr2_category = "hook",
125 .tr2_label = hook_name,
126
127 .processes = 1,
128 .ungroup = 1,
129
130 .get_next_task = pick_next_hook,
131 .start_failure = notify_start_failure,
132 .task_finished = notify_hook_finished,
133
134 .data = &cb_data,
135 };
Emily Shaffer96e72252021-12-22 04:59:27 +0100136
137 if (!options)
138 BUG("a struct run_hooks_opt must be provided to run_hooks");
139
Ævar Arnfjörð Bjarmasona8cc5942022-03-07 13:33:46 +0100140 if (options->invoked_hook)
141 *options->invoked_hook = 0;
142
Emily Shaffer96e72252021-12-22 04:59:27 +0100143 if (!hook_path && !options->error_if_missing)
144 goto cleanup;
145
146 if (!hook_path) {
147 ret = error("cannot find a hook named %s", hook_name);
148 goto cleanup;
149 }
150
151 cb_data.hook_path = hook_path;
Emily Shaffer1a3017d2021-12-22 04:59:36 +0100152 if (options->dir) {
153 strbuf_add_absolute_path(&abs_path, hook_path);
154 cb_data.hook_path = abs_path.buf;
155 }
156
Ævar Arnfjörð Bjarmason6e5ba0b2022-10-12 23:02:26 +0200157 run_processes_parallel(&opts);
Emily Shaffer96e72252021-12-22 04:59:27 +0100158 ret = cb_data.rc;
159cleanup:
Emily Shaffer1a3017d2021-12-22 04:59:36 +0100160 strbuf_release(&abs_path);
Emily Shaffer96e72252021-12-22 04:59:27 +0100161 run_hooks_opt_clear(options);
162 return ret;
163}
Ævar Arnfjörð Bjarmason474c1192021-12-22 04:59:28 +0100164
165int run_hooks(const char *hook_name)
166{
167 struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
168
169 return run_hooks_opt(hook_name, &opt);
170}
Ævar Arnfjörð Bjarmasonab81cf22021-12-22 04:59:31 +0100171
172int run_hooks_l(const char *hook_name, ...)
173{
174 struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
175 va_list ap;
176 const char *arg;
177
178 va_start(ap, hook_name);
179 while ((arg = va_arg(ap, const char *)))
180 strvec_push(&opt.args, arg);
181 va_end(ap);
182
183 return run_hooks_opt(hook_name, &opt);
184}