|
1 #!/bin/sh |
|
2 |
|
3 # An example hook script to update a checked-out tree on a git push. |
|
4 # |
|
5 # This hook is invoked by git-receive-pack(1) when it reacts to git |
|
6 # push and updates reference(s) in its repository, and when the push |
|
7 # tries to update the branch that is currently checked out and the |
|
8 # receive.denyCurrentBranch configuration variable is set to |
|
9 # updateInstead. |
|
10 # |
|
11 # By default, such a push is refused if the working tree and the index |
|
12 # of the remote repository has any difference from the currently |
|
13 # checked out commit; when both the working tree and the index match |
|
14 # the current commit, they are updated to match the newly pushed tip |
|
15 # of the branch. This hook is to be used to override the default |
|
16 # behaviour; however the code below reimplements the default behaviour |
|
17 # as a starting point for convenient modification. |
|
18 # |
|
19 # The hook receives the commit with which the tip of the current |
|
20 # branch is going to be updated: |
|
21 commit=$1 |
|
22 |
|
23 # It can exit with a non-zero status to refuse the push (when it does |
|
24 # so, it must not modify the index or the working tree). |
|
25 die () { |
|
26 echo >&2 "$*" |
|
27 exit 1 |
|
28 } |
|
29 |
|
30 # Or it can make any necessary changes to the working tree and to the |
|
31 # index to bring them to the desired state when the tip of the current |
|
32 # branch is updated to the new commit, and exit with a zero status. |
|
33 # |
|
34 # For example, the hook can simply run git read-tree -u -m HEAD "$1" |
|
35 # in order to emulate git fetch that is run in the reverse direction |
|
36 # with git push, as the two-tree form of git read-tree -u -m is |
|
37 # essentially the same as git switch or git checkout that switches |
|
38 # branches while keeping the local changes in the working tree that do |
|
39 # not interfere with the difference between the branches. |
|
40 |
|
41 # The below is a more-or-less exact translation to shell of the C code |
|
42 # for the default behaviour for git's push-to-checkout hook defined in |
|
43 # the push_to_deploy() function in builtin/receive-pack.c. |
|
44 # |
|
45 # Note that the hook will be executed from the repository directory, |
|
46 # not from the working tree, so if you want to perform operations on |
|
47 # the working tree, you will have to adapt your code accordingly, e.g. |
|
48 # by adding "cd .." or using relative paths. |
|
49 |
|
50 if ! git update-index -q --ignore-submodules --refresh |
|
51 then |
|
52 die "Up-to-date check failed" |
|
53 fi |
|
54 |
|
55 if ! git diff-files --quiet --ignore-submodules -- |
|
56 then |
|
57 die "Working directory has unstaged changes" |
|
58 fi |
|
59 |
|
60 # This is a rough translation of: |
|
61 # |
|
62 # head_has_history() ? "HEAD" : EMPTY_TREE_SHA1_HEX |
|
63 if git cat-file -e HEAD 2>/dev/null |
|
64 then |
|
65 head=HEAD |
|
66 else |
|
67 head=$(git hash-object -t tree --stdin </dev/null) |
|
68 fi |
|
69 |
|
70 if ! git diff-index --quiet --cached --ignore-submodules $head -- |
|
71 then |
|
72 die "Working directory has staged changes" |
|
73 fi |
|
74 |
|
75 if ! git read-tree -u -m "$commit" |
|
76 then |
|
77 die "Could not update working tree to new HEAD" |
|
78 fi |