* [PATCH] post-update hook: update working copy @ 2007-11-02 0:45 Sam Vilain 2007-11-02 1:02 ` Junio C Hamano 0 siblings, 1 reply; 6+ messages in thread From: Sam Vilain @ 2007-11-02 0:45 UTC (permalink / raw) To: Junio C Hamano; +Cc: git, Sam Vilain Now that git-stash is available, it is not so unsafe to push to a non-bare repository, but care needs to be taken to preserve any dirty working copy or index state. This hook script does that, using git-stash. Signed-off-by: Sam Vilain <sam.vilain@catalyst.net.nz> --- templates/hooks--post-update | 76 ++++++++++++++++++++++++++++++++++++++++-- 1 files changed, 73 insertions(+), 3 deletions(-) mode change 100644 => 100755 templates/hooks--post-update diff --git a/templates/hooks--post-update b/templates/hooks--post-update old mode 100644 new mode 100755 index bcba893..352a432 --- a/templates/hooks--post-update +++ b/templates/hooks--post-update @@ -1,8 +1,78 @@ #!/bin/sh # -# An example hook script to prepare a packed repository for use over -# dumb transports. +# This hook does two things: +# +# 1. update the "info" files that allow the list of references to be +# queries over dumb transports such as http +# +# 2. if this repository looks like it is a non-bare repository, and +# the checked-out branch is pushed to, then update the working copy. +# This makes "push" function somewhat similarly to darcs and bzr. # # To enable this hook, make this file executable by "chmod +x post-update". -exec git-update-server-info +git-update-server-info + +is_bare=$(git-config --get --bool core.bare) + +if [ -z "$is_bare" ] +then + # for compatibility's sake, guess + git_dir_full=$(cd $GIT_DIR; pwd) + case $git_dir_full in */.git) is_bare=false;; *) is_bare=true;; esac +fi + +update_wc() { + ref=$1 + echo "Push to checked out branch $ref" >&2 + if (cd $GIT_WORK_TREE; git-diff-files -q --exit-code >/dev/null) + then + wc_dirty=0 + else + echo "W:unstaged changes found in working copy" >&2 + wc_dirty=1 + desc="working copy" + fi + if git diff-index HEAD@{1} >/dev/null + then + index_dirty=0 + else + echo "W:uncommitted, staged changes found" >&2 + index_dirty=1 + if [ -n "$desc" ] + then + desc="$desc and index" + else + desc="index" + fi + fi + if [ "$wc_dirty" -ne 0 -o "$index_dirty" -ne 0 ] + then + new=$(git rev-parse HEAD) + git-update-ref --no-deref HEAD HEAD@{1} + echo "W:stashing dirty $desc - see git-stash(1)" >&2 + (cd $GIT_WORK_TREE + git stash save "dirty $desc before update to $new") + git-symbolic-ref HEAD "$ref" + fi + + # eye candy - show the WC updates :) + echo "Updating working copy" >&2 + (cd $GIT_WORK_TREE + git-diff-index -R --name-status HEAD >&2 + git-reset --hard HEAD) +} + +if [ "$is_bare" = "false" ] +then + active_branch=`git-symbolic-ref HEAD` + export GIT_DIR=$(cd $GIT_DIR; pwd) + GIT_WORK_TREE=${GIT_WORK_TREE-..} + for ref + do + if [ "$ref" = "$active_branch" ] + then + update_wc $ref + fi + done +fi -- 1.5.3.2.3.g2f2dcc-dirty ^ permalink raw reply related [flat|nested] 6+ messages in thread
* Re: [PATCH] post-update hook: update working copy 2007-11-02 0:45 [PATCH] post-update hook: update working copy Sam Vilain @ 2007-11-02 1:02 ` Junio C Hamano 2007-11-02 3:36 ` Sam Vilain 2007-11-02 10:34 ` Andreas Ericsson 0 siblings, 2 replies; 6+ messages in thread From: Junio C Hamano @ 2007-11-02 1:02 UTC (permalink / raw) To: Sam Vilain; +Cc: git Sam Vilain <sam.vilain@catalyst.net.nz> writes: > Now that git-stash is available, it is not so unsafe to push to a > non-bare repository, but care needs to be taken to preserve any dirty > working copy or index state. This hook script does that, using > git-stash. Honestly, I am reluctant to do things that _encourages_ pushing into a live tree. - Who guarantees that the reflog is enabled for the HEAD? - Who guarantees that a human user is not actively editing the work tree files without saving? You would not see "dirty state", the editor would notice "the file was modified since you started editing it" and tell so to the user, but the user cannot recover from the situation without knowing to do the three-way merge between HEAD@{1}, HEAD and the index _anyway_. > +update_wc() { > + ref=$1 > + echo "Push to checked out branch $ref" >&2 > + if (cd $GIT_WORK_TREE; git-diff-files -q --exit-code >/dev/null) > + then > + wc_dirty=0 > + else > + echo "W:unstaged changes found in working copy" >&2 > + wc_dirty=1 > + desc="working copy" > + fi > + if git diff-index HEAD@{1} >/dev/null Are you missing "--cached" here? > + if [ "$wc_dirty" -ne 0 -o "$index_dirty" -ne 0 ] > + then > + new=$(git rev-parse HEAD) > + git-update-ref --no-deref HEAD HEAD@{1} > + echo "W:stashing dirty $desc - see git-stash(1)" >&2 > + (cd $GIT_WORK_TREE > + git stash save "dirty $desc before update to $new") > + git-symbolic-ref HEAD "$ref" This part feels somewhat dangerous. What happens if we are interrupted in the middle of these commands? > + fi > + > + # eye candy - show the WC updates :) > + echo "Updating working copy" >&2 > + (cd $GIT_WORK_TREE > + git-diff-index -R --name-status HEAD >&2 > + git-reset --hard HEAD) > +} And I would have expected you would unstash the dirty state here. Are there any reason not to? ^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [PATCH] post-update hook: update working copy 2007-11-02 1:02 ` Junio C Hamano @ 2007-11-02 3:36 ` Sam Vilain 2007-11-02 8:49 ` Junio C Hamano 2007-11-02 10:34 ` Andreas Ericsson 1 sibling, 1 reply; 6+ messages in thread From: Sam Vilain @ 2007-11-02 3:36 UTC (permalink / raw) To: Junio C Hamano; +Cc: Sam Vilain, git Junio C Hamano wrote: >> Now that git-stash is available, it is not so unsafe to push to a >> non-bare repository, but care needs to be taken to preserve any dirty >> working copy or index state. This hook script does that, using >> git-stash. > > Honestly, I am reluctant to do things that _encourages_ pushing > into a live tree. I think a lot of people want this, and explaining why it doesn't work to people is especially tiresome given that I know it can work, and the caveats get smaller and smaller each time. I still think that the operation can be made safe enough for general use. > - Who guarantees that the reflog is enabled for the HEAD? Ok, so I guess if that's the case then the old behaviour is OK. However, this would not be required if implemented in receive-pack, as the only thing I'm using it for is to get the revision that the current index is based on. The other option, which was writing two hooks, both of which need to be enabled, seemed far too ugly! > - Who guarantees that a human user is not actively editing the > work tree files without saving? You would not see "dirty > state", the editor would notice "the file was modified since > you started editing it" and tell so to the user, but the user > cannot recover from the situation without knowing to do the > three-way merge between HEAD@{1}, HEAD and the index _anyway_. There seems to be a lot of focus on this. However I don't think it is a showstopper; in this instance that the person who has their life ruined by the incoming push can blame the person who did it, blame themselves for giving that stupid user access to their working directory, and then accept that the tool is doing the best that it can. >> + if git diff-index HEAD@{1} >/dev/null > > Are you missing "--cached" here? Ah, yes, good catch. >> + if [ "$wc_dirty" -ne 0 -o "$index_dirty" -ne 0 ] >> + then >> + new=$(git rev-parse HEAD) >> + git-update-ref --no-deref HEAD HEAD@{1} >> + echo "W:stashing dirty $desc - see git-stash(1)" >&2 >> + (cd $GIT_WORK_TREE >> + git stash save "dirty $desc before update to $new") >> + git-symbolic-ref HEAD "$ref" > > This part feels somewhat dangerous. What happens if we are > interrupted in the middle of these commands? Heh. What seems to happen is that the local command is aborted and the remote one continues. Nonetheless I'll add a trap statement, but again if implemented in receive-pack it could probably be more resilient. >> + fi >> + >> + # eye candy - show the WC updates :) >> + echo "Updating working copy" >&2 >> + (cd $GIT_WORK_TREE >> + git-diff-index -R --name-status HEAD >&2 >> + git-reset --hard HEAD) >> +} > > And I would have expected you would unstash the dirty state here. > Are there any reason not to? If git-stash could support stashing conflicted merges, I don't think so. However, if the user is a git-shell user, then they won't be able to resolve the conflict and they won't be able to re-push as the stash will fail (a condition not visited by the above). Sam. changes as you suggested are below; diff --git a/templates/hooks--post-update b/templates/hooks--post-update index 352a432..b15c711 100755 --- a/templates/hooks--post-update +++ b/templates/hooks--post-update @@ -25,6 +25,11 @@ fi update_wc() { ref=$1 echo "Push to checked out branch $ref" >&2 + if [ ! -f $GIT_DIR/logs/HEAD ] + then + echo "E:push to non-bare repository requires a HEAD reflog" >&2 + exit 1 + fi if (cd $GIT_WORK_TREE; git-diff-files -q --exit-code >/dev/null) then wc_dirty=0 @@ -33,7 +38,7 @@ update_wc() { wc_dirty=1 desc="working copy" fi - if git diff-index HEAD@{1} >/dev/null + if git diff-index --cached HEAD@{1} >/dev/null then index_dirty=0 else @@ -49,11 +54,13 @@ update_wc() { if [ "$wc_dirty" -ne 0 -o "$index_dirty" -ne 0 ] then new=$(git rev-parse HEAD) - git-update-ref --no-deref HEAD HEAD@{1} echo "W:stashing dirty $desc - see git-stash(1)" >&2 - (cd $GIT_WORK_TREE - git stash save "dirty $desc before update to $new") + ( trap 'echo trapped $$; git symbolic-ref HEAD "'"$ref"'"' 2 3 13 15 ERR EXIT + git-update-ref --no-deref HEAD HEAD@{1} + cd $GIT_WORK_TREE + git stash save "dirty $desc before update to $new"; git-symbolic-ref HEAD "$ref" + ) fi # eye candy - show the WC updates :) ^ permalink raw reply related [flat|nested] 6+ messages in thread
* Re: [PATCH] post-update hook: update working copy 2007-11-02 3:36 ` Sam Vilain @ 2007-11-02 8:49 ` Junio C Hamano 0 siblings, 0 replies; 6+ messages in thread From: Junio C Hamano @ 2007-11-02 8:49 UTC (permalink / raw) To: Sam Vilain; +Cc: Sam Vilain, git Sam Vilain <sam@vilain.net> writes: > changes as you suggested are below; I squashed these and looked the result over, but I see it is contradicting with itself. You do not want to "publish" a live git repository so running git-update-server-info in a repository that is served by this script feels very wrong. How about having this in contrib/hooks/post-update-worktre? By the way, there are quite a few careless places that use shell variables without quoting. ^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [PATCH] post-update hook: update working copy 2007-11-02 1:02 ` Junio C Hamano 2007-11-02 3:36 ` Sam Vilain @ 2007-11-02 10:34 ` Andreas Ericsson 2007-11-02 17:28 ` Steven Grimm 1 sibling, 1 reply; 6+ messages in thread From: Andreas Ericsson @ 2007-11-02 10:34 UTC (permalink / raw) To: Junio C Hamano; +Cc: Sam Vilain, git Junio C Hamano wrote: > Sam Vilain <sam.vilain@catalyst.net.nz> writes: > >> Now that git-stash is available, it is not so unsafe to push to a >> non-bare repository, but care needs to be taken to preserve any dirty >> working copy or index state. This hook script does that, using >> git-stash. > > Honestly, I am reluctant to do things that _encourages_ pushing > into a live tree. > "Live" and "living" are perhaps two different things here. I for one have something similar, but only for repositories residing on certain servers, where there really must be zero local changes to the working tree. > - Who guarantees that the reflog is enabled for the HEAD? > I disable reflogs on that server. There's (hardly ever) any human interaction with the scripts in that repo, so I really, really don't care about reflogs. > - Who guarantees that a human user is not actively editing the > work tree files without saving? There are times when one simply doesn't care. I realize that for my situation, a much simpler script can (and is) used, so I agree with your concerns. The idea that every git repo has a human hacking on it isn't true though, so doing things like this are sometimes useful, timesaving and a real help. -- Andreas Ericsson andreas.ericsson@op5.se OP5 AB www.op5.se Tel: +46 8-230225 Fax: +46 8-230231 ^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [PATCH] post-update hook: update working copy 2007-11-02 10:34 ` Andreas Ericsson @ 2007-11-02 17:28 ` Steven Grimm 0 siblings, 0 replies; 6+ messages in thread From: Steven Grimm @ 2007-11-02 17:28 UTC (permalink / raw) To: Andreas Ericsson; +Cc: Junio C Hamano, Sam Vilain, git Andreas Ericsson wrote: >> - Who guarantees that a human user is not actively editing the >> work tree files without saving? > There are times when one simply doesn't care. > > I realize that for my situation, a much simpler script can (and is) > used, so > I agree with your concerns. The idea that every git repo has a human > hacking > on it isn't true though, so doing things like this are sometimes useful, > timesaving and a real help. Yeah, that's absolutely true. My use cases would be twofold. First, a public reference tree on a shared development server where people can look over my corner of the code base without having to check the whole thing out for themselves. Second, a Web server with a bunch of static text/image files and PHP scripts. If I can deploy by just pushing to a "current release" branch, that saves me from having to first push then ssh to the machine and do "git reset --hard". Neither one of those things is impossible to do with vanilla git. They just require extra busywork steps at the moment if you don't use an "update the working copy on push" hook. -Steve ^ permalink raw reply [flat|nested] 6+ messages in thread
end of thread, other threads:[~2007-11-02 17:28 UTC | newest] Thread overview: 6+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2007-11-02 0:45 [PATCH] post-update hook: update working copy Sam Vilain 2007-11-02 1:02 ` Junio C Hamano 2007-11-02 3:36 ` Sam Vilain 2007-11-02 8:49 ` Junio C Hamano 2007-11-02 10:34 ` Andreas Ericsson 2007-11-02 17:28 ` Steven Grimm
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).