Browse Source
Add a sample 'push-to-checkout' hook, that performs the same as what the built-in default action does. * as/sample-push-to-checkout-hook: hook: add sample template for push-to-checkoutmaint
![gitster@pobox.com](/assets/img/avatar_default.png)
1 changed files with 78 additions and 0 deletions
@ -0,0 +1,78 @@
@@ -0,0 +1,78 @@
|
||||
#!/bin/sh |
||||
|
||||
# An example hook script to update a checked-out tree on a git push. |
||||
# |
||||
# This hook is invoked by git-receive-pack(1) when it reacts to git |
||||
# push and updates reference(s) in its repository, and when the push |
||||
# tries to update the branch that is currently checked out and the |
||||
# receive.denyCurrentBranch configuration variable is set to |
||||
# updateInstead. |
||||
# |
||||
# By default, such a push is refused if the working tree and the index |
||||
# of the remote repository has any difference from the currently |
||||
# checked out commit; when both the working tree and the index match |
||||
# the current commit, they are updated to match the newly pushed tip |
||||
# of the branch. This hook is to be used to override the default |
||||
# behaviour; however the code below reimplements the default behaviour |
||||
# as a starting point for convenient modification. |
||||
# |
||||
# The hook receives the commit with which the tip of the current |
||||
# branch is going to be updated: |
||||
commit=$1 |
||||
|
||||
# It can exit with a non-zero status to refuse the push (when it does |
||||
# so, it must not modify the index or the working tree). |
||||
die () { |
||||
echo >&2 "$*" |
||||
exit 1 |
||||
} |
||||
|
||||
# Or it can make any necessary changes to the working tree and to the |
||||
# index to bring them to the desired state when the tip of the current |
||||
# branch is updated to the new commit, and exit with a zero status. |
||||
# |
||||
# For example, the hook can simply run git read-tree -u -m HEAD "$1" |
||||
# in order to emulate git fetch that is run in the reverse direction |
||||
# with git push, as the two-tree form of git read-tree -u -m is |
||||
# essentially the same as git switch or git checkout that switches |
||||
# branches while keeping the local changes in the working tree that do |
||||
# not interfere with the difference between the branches. |
||||
|
||||
# The below is a more-or-less exact translation to shell of the C code |
||||
# for the default behaviour for git's push-to-checkout hook defined in |
||||
# the push_to_deploy() function in builtin/receive-pack.c. |
||||
# |
||||
# Note that the hook will be executed from the repository directory, |
||||
# not from the working tree, so if you want to perform operations on |
||||
# the working tree, you will have to adapt your code accordingly, e.g. |
||||
# by adding "cd .." or using relative paths. |
||||
|
||||
if ! git update-index -q --ignore-submodules --refresh |
||||
then |
||||
die "Up-to-date check failed" |
||||
fi |
||||
|
||||
if ! git diff-files --quiet --ignore-submodules -- |
||||
then |
||||
die "Working directory has unstaged changes" |
||||
fi |
||||
|
||||
# This is a rough translation of: |
||||
# |
||||
# head_has_history() ? "HEAD" : EMPTY_TREE_SHA1_HEX |
||||
if git cat-file -e HEAD 2>/dev/null |
||||
then |
||||
head=HEAD |
||||
else |
||||
head=$(git hash-object -t tree --stdin </dev/null) |
||||
fi |
||||
|
||||
if ! git diff-index --quiet --cached --ignore-submodules $head -- |
||||
then |
||||
die "Working directory has staged changes" |
||||
fi |
||||
|
||||
if ! git read-tree -u -m "$commit" |
||||
then |
||||
die "Could not update working tree to new HEAD" |
||||
fi |
Loading…
Reference in new issue