git
/
hook.c
196 строк · 4.3 Кб
1#include "git-compat-util.h"
2#include "abspath.h"
3#include "advice.h"
4#include "gettext.h"
5#include "hook.h"
6#include "path.h"
7#include "run-command.h"
8#include "config.h"
9#include "strbuf.h"
10#include "environment.h"
11#include "setup.h"
12
13const char *find_hook(struct repository *r, const char *name)
14{
15static struct strbuf path = STRBUF_INIT;
16
17int found_hook;
18
19strbuf_reset(&path);
20strbuf_repo_git_path(&path, r, "hooks/%s", name);
21found_hook = access(path.buf, X_OK) >= 0;
22#ifdef STRIP_EXTENSION
23if (!found_hook) {
24int err = errno;
25
26strbuf_addstr(&path, STRIP_EXTENSION);
27found_hook = access(path.buf, X_OK) >= 0;
28if (!found_hook)
29errno = err;
30}
31#endif
32
33if (!found_hook) {
34if (errno == EACCES && advice_enabled(ADVICE_IGNORED_HOOK)) {
35static struct string_list advise_given = STRING_LIST_INIT_DUP;
36
37if (!string_list_lookup(&advise_given, name)) {
38string_list_insert(&advise_given, name);
39advise(_("The '%s' hook was ignored because "
40"it's not set as executable.\n"
41"You can disable this warning with "
42"`git config advice.ignoredHook false`."),
43path.buf);
44}
45}
46return NULL;
47}
48return path.buf;
49}
50
51int hook_exists(struct repository *r, const char *name)
52{
53return !!find_hook(r, name);
54}
55
56static int pick_next_hook(struct child_process *cp,
57struct strbuf *out UNUSED,
58void *pp_cb,
59void **pp_task_cb UNUSED)
60{
61struct hook_cb_data *hook_cb = pp_cb;
62const char *hook_path = hook_cb->hook_path;
63
64if (!hook_path)
65return 0;
66
67cp->no_stdin = 1;
68strvec_pushv(&cp->env, hook_cb->options->env.v);
69/* reopen the file for stdin; run_command closes it. */
70if (hook_cb->options->path_to_stdin) {
71cp->no_stdin = 0;
72cp->in = xopen(hook_cb->options->path_to_stdin, O_RDONLY);
73}
74cp->stdout_to_stderr = 1;
75cp->trace2_hook_name = hook_cb->hook_name;
76cp->dir = hook_cb->options->dir;
77
78strvec_push(&cp->args, hook_path);
79strvec_pushv(&cp->args, hook_cb->options->args.v);
80
81/*
82* This pick_next_hook() will be called again, we're only
83* running one hook, so indicate that no more work will be
84* done.
85*/
86hook_cb->hook_path = NULL;
87
88return 1;
89}
90
91static int notify_start_failure(struct strbuf *out UNUSED,
92void *pp_cb,
93void *pp_task_cp UNUSED)
94{
95struct hook_cb_data *hook_cb = pp_cb;
96
97hook_cb->rc |= 1;
98
99return 1;
100}
101
102static int notify_hook_finished(int result,
103struct strbuf *out UNUSED,
104void *pp_cb,
105void *pp_task_cb UNUSED)
106{
107struct hook_cb_data *hook_cb = pp_cb;
108struct run_hooks_opt *opt = hook_cb->options;
109
110hook_cb->rc |= result;
111
112if (opt->invoked_hook)
113*opt->invoked_hook = 1;
114
115return 0;
116}
117
118static void run_hooks_opt_clear(struct run_hooks_opt *options)
119{
120strvec_clear(&options->env);
121strvec_clear(&options->args);
122}
123
124int run_hooks_opt(struct repository *r, const char *hook_name,
125struct run_hooks_opt *options)
126{
127struct strbuf abs_path = STRBUF_INIT;
128struct hook_cb_data cb_data = {
129.rc = 0,
130.hook_name = hook_name,
131.options = options,
132};
133const char *const hook_path = find_hook(r, hook_name);
134int ret = 0;
135const struct run_process_parallel_opts opts = {
136.tr2_category = "hook",
137.tr2_label = hook_name,
138
139.processes = 1,
140.ungroup = 1,
141
142.get_next_task = pick_next_hook,
143.start_failure = notify_start_failure,
144.task_finished = notify_hook_finished,
145
146.data = &cb_data,
147};
148
149if (!options)
150BUG("a struct run_hooks_opt must be provided to run_hooks");
151
152if (options->invoked_hook)
153*options->invoked_hook = 0;
154
155if (!hook_path && !options->error_if_missing)
156goto cleanup;
157
158if (!hook_path) {
159ret = error("cannot find a hook named %s", hook_name);
160goto cleanup;
161}
162
163cb_data.hook_path = hook_path;
164if (options->dir) {
165strbuf_add_absolute_path(&abs_path, hook_path);
166cb_data.hook_path = abs_path.buf;
167}
168
169run_processes_parallel(&opts);
170ret = cb_data.rc;
171cleanup:
172strbuf_release(&abs_path);
173run_hooks_opt_clear(options);
174return ret;
175}
176
177int run_hooks(struct repository *r, const char *hook_name)
178{
179struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
180
181return run_hooks_opt(r, hook_name, &opt);
182}
183
184int run_hooks_l(struct repository *r, const char *hook_name, ...)
185{
186struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
187va_list ap;
188const char *arg;
189
190va_start(ap, hook_name);
191while ((arg = va_arg(ap, const char *)))
192strvec_push(&opt.args, arg);
193va_end(ap);
194
195return run_hooks_opt(r, hook_name, &opt);
196}
197