hook: mark non-parallelizable hooks
Several hooks are known to be inherently non-parallelizable, so initialize them with RUN_HOOKS_OPT_INIT_FORCE_SERIAL. This pins jobs=1 and overrides any hook.jobs or runtime -j flags. These hooks are: applypatch-msg, pre-commit, prepare-commit-msg, commit-msg, post-commit, post-checkout, and push-to-checkout. Signed-off-by: Emily Shaffer <emilyshaffer@google.com> Signed-off-by: Adrian Ratiu <adrian.ratiu@collabora.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>main
parent
f776b77f00
commit
ae25764e50
|
|
@ -36,6 +36,20 @@ hook.<friendly-name>.parallel::
|
||||||
hook.jobs::
|
hook.jobs::
|
||||||
Specifies how many hooks can be run simultaneously during parallelized
|
Specifies how many hooks can be run simultaneously during parallelized
|
||||||
hook execution. If unspecified, defaults to 1 (serial execution).
|
hook execution. If unspecified, defaults to 1 (serial execution).
|
||||||
|
Some hooks always run sequentially regardless of this setting because
|
||||||
|
they operate on shared data and cannot safely be parallelized:
|
||||||
|
+
|
||||||
|
--
|
||||||
|
`applypatch-msg`;;
|
||||||
|
`prepare-commit-msg`;;
|
||||||
|
`commit-msg`;;
|
||||||
|
Receive a commit message file and may rewrite it in place.
|
||||||
|
`pre-commit`;;
|
||||||
|
`post-checkout`;;
|
||||||
|
`push-to-checkout`;;
|
||||||
|
`post-commit`;;
|
||||||
|
Access the working tree, index, or repository state.
|
||||||
|
--
|
||||||
+
|
+
|
||||||
This setting has no effect unless all configured hooks for the event have
|
This setting has no effect unless all configured hooks for the event have
|
||||||
`hook.<friendly-name>.parallel` set to `true`.
|
`hook.<friendly-name>.parallel` set to `true`.
|
||||||
|
|
|
||||||
|
|
@ -490,9 +490,11 @@ static int run_applypatch_msg_hook(struct am_state *state)
|
||||||
|
|
||||||
assert(state->msg);
|
assert(state->msg);
|
||||||
|
|
||||||
if (!state->no_verify)
|
if (!state->no_verify) {
|
||||||
ret = run_hooks_l(the_repository, "applypatch-msg",
|
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT_FORCE_SERIAL;
|
||||||
am_path(state, "final-commit"), NULL);
|
strvec_push(&opt.args, am_path(state, "final-commit"));
|
||||||
|
ret = run_hooks_opt(the_repository, "applypatch-msg", &opt);
|
||||||
|
}
|
||||||
|
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
FREE_AND_NULL(state->msg);
|
FREE_AND_NULL(state->msg);
|
||||||
|
|
|
||||||
|
|
@ -31,6 +31,7 @@
|
||||||
#include "resolve-undo.h"
|
#include "resolve-undo.h"
|
||||||
#include "revision.h"
|
#include "revision.h"
|
||||||
#include "setup.h"
|
#include "setup.h"
|
||||||
|
#include "strvec.h"
|
||||||
#include "submodule.h"
|
#include "submodule.h"
|
||||||
#include "symlinks.h"
|
#include "symlinks.h"
|
||||||
#include "trace2.h"
|
#include "trace2.h"
|
||||||
|
|
@ -123,13 +124,19 @@ static void branch_info_release(struct branch_info *info)
|
||||||
static int post_checkout_hook(struct commit *old_commit, struct commit *new_commit,
|
static int post_checkout_hook(struct commit *old_commit, struct commit *new_commit,
|
||||||
int changed)
|
int changed)
|
||||||
{
|
{
|
||||||
return run_hooks_l(the_repository, "post-checkout",
|
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT_FORCE_SERIAL;
|
||||||
oid_to_hex(old_commit ? &old_commit->object.oid : null_oid(the_hash_algo)),
|
|
||||||
oid_to_hex(new_commit ? &new_commit->object.oid : null_oid(the_hash_algo)),
|
|
||||||
changed ? "1" : "0", NULL);
|
|
||||||
/* "new_commit" can be NULL when checking out from the index before
|
|
||||||
a commit exists. */
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* "new_commit" can be NULL when checking out from the index before
|
||||||
|
* a commit exists.
|
||||||
|
*/
|
||||||
|
strvec_pushl(&opt.args,
|
||||||
|
oid_to_hex(old_commit ? &old_commit->object.oid : null_oid(the_hash_algo)),
|
||||||
|
oid_to_hex(new_commit ? &new_commit->object.oid : null_oid(the_hash_algo)),
|
||||||
|
changed ? "1" : "0",
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
return run_hooks_opt(the_repository, "post-checkout", &opt);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int update_some(const struct object_id *oid, struct strbuf *base,
|
static int update_some(const struct object_id *oid, struct strbuf *base,
|
||||||
|
|
|
||||||
|
|
@ -647,6 +647,7 @@ static int checkout(int submodule_progress,
|
||||||
struct tree *tree;
|
struct tree *tree;
|
||||||
struct tree_desc t;
|
struct tree_desc t;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
struct run_hooks_opt hook_opt = RUN_HOOKS_OPT_INIT_FORCE_SERIAL;
|
||||||
|
|
||||||
if (option_no_checkout)
|
if (option_no_checkout)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
@ -697,8 +698,9 @@ static int checkout(int submodule_progress,
|
||||||
if (write_locked_index(the_repository->index, &lock_file, COMMIT_LOCK))
|
if (write_locked_index(the_repository->index, &lock_file, COMMIT_LOCK))
|
||||||
die(_("unable to write new index file"));
|
die(_("unable to write new index file"));
|
||||||
|
|
||||||
err |= run_hooks_l(the_repository, "post-checkout", oid_to_hex(null_oid(the_hash_algo)),
|
strvec_pushl(&hook_opt.args, oid_to_hex(null_oid(the_hash_algo)),
|
||||||
oid_to_hex(&oid), "1", NULL);
|
oid_to_hex(&oid), "1", NULL);
|
||||||
|
err |= run_hooks_opt(the_repository, "post-checkout", &hook_opt);
|
||||||
|
|
||||||
if (!err && (option_recurse_submodules.nr > 0)) {
|
if (!err && (option_recurse_submodules.nr > 0)) {
|
||||||
struct child_process cmd = CHILD_PROCESS_INIT;
|
struct child_process cmd = CHILD_PROCESS_INIT;
|
||||||
|
|
|
||||||
|
|
@ -1455,7 +1455,8 @@ static const char *push_to_checkout(unsigned char *hash,
|
||||||
struct strvec *env,
|
struct strvec *env,
|
||||||
const char *work_tree)
|
const char *work_tree)
|
||||||
{
|
{
|
||||||
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
|
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT_FORCE_SERIAL;
|
||||||
|
|
||||||
opt.invoked_hook = invoked_hook;
|
opt.invoked_hook = invoked_hook;
|
||||||
|
|
||||||
strvec_pushf(env, "GIT_WORK_TREE=%s", absolute_path(work_tree));
|
strvec_pushf(env, "GIT_WORK_TREE=%s", absolute_path(work_tree));
|
||||||
|
|
|
||||||
|
|
@ -609,7 +609,7 @@ done:
|
||||||
* is_junk is cleared, but do return appropriate code when hook fails.
|
* is_junk is cleared, but do return appropriate code when hook fails.
|
||||||
*/
|
*/
|
||||||
if (!ret && opts->checkout && !opts->orphan) {
|
if (!ret && opts->checkout && !opts->orphan) {
|
||||||
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
|
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT_FORCE_SERIAL;
|
||||||
|
|
||||||
strvec_pushl(&opt.env, "GIT_DIR", "GIT_WORK_TREE", NULL);
|
strvec_pushl(&opt.env, "GIT_DIR", "GIT_WORK_TREE", NULL);
|
||||||
strvec_pushl(&opt.args,
|
strvec_pushl(&opt.args,
|
||||||
|
|
|
||||||
2
commit.c
2
commit.c
|
|
@ -1970,7 +1970,7 @@ size_t ignored_log_message_bytes(const char *buf, size_t len)
|
||||||
int run_commit_hook(int editor_is_used, const char *index_file,
|
int run_commit_hook(int editor_is_used, const char *index_file,
|
||||||
int *invoked_hook, const char *name, ...)
|
int *invoked_hook, const char *name, ...)
|
||||||
{
|
{
|
||||||
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT;
|
struct run_hooks_opt opt = RUN_HOOKS_OPT_INIT_FORCE_SERIAL;
|
||||||
va_list args;
|
va_list args;
|
||||||
const char *arg;
|
const char *arg;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -832,4 +832,20 @@ test_expect_success 'client hooks: pre-push runs in parallel when hook.jobs > 1'
|
||||||
test_cmp expect repo-parallel/hook.order
|
test_cmp expect repo-parallel/hook.order
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'hook.jobs=2 is ignored for force-serial hooks (pre-commit)' '
|
||||||
|
test_when_finished "rm -f sentinel.started sentinel.done hook.order" &&
|
||||||
|
test_config hook.hook-1.event pre-commit &&
|
||||||
|
test_config hook.hook-1.command \
|
||||||
|
"touch sentinel.started; sleep 2; touch sentinel.done" &&
|
||||||
|
test_config hook.hook-1.parallel true &&
|
||||||
|
test_config hook.hook-2.event pre-commit &&
|
||||||
|
test_config hook.hook-2.command \
|
||||||
|
"$(sentinel_detector sentinel hook.order)" &&
|
||||||
|
test_config hook.hook-2.parallel true &&
|
||||||
|
test_config hook.jobs 2 &&
|
||||||
|
git commit --allow-empty -m "test: verify force-serial on pre-commit" &&
|
||||||
|
echo serial >expect &&
|
||||||
|
test_cmp expect hook.order
|
||||||
|
'
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue