* Deleting of the specified ref during the post-receive hook @ 2010-09-21 14:40 Joshua Jensen 2010-09-21 14:50 ` Shawn O. Pearce 0 siblings, 1 reply; 4+ messages in thread From: Joshua Jensen @ 2010-09-21 14:40 UTC (permalink / raw) To: git@vger.kernel.org I've recently asked whether there was a public script to act as an automatic 'maintainer', something akin to Gerrit's push always succeeding without having to pull first when using Git as a central repository. I received a number of suggestions, and I have begun trial implementations. My current line of thought has an auto-merging script that monitors the refs/for/ namespace (similar to Gerrit) and then applies --no-ff merges to the appropriate branch. For instance, when the user pushes to refs/for/master, the post-receive hook creates a secondary ref called refs/for/master-SHA1-timestamp and then deletes the refs/for/master ref: #!bin/sh # post-receive hook while read oldrev newrev ref do case $ref in refs/for/*) timestamp=`date +%s` `git update-ref $ref-$newrev-$timestamp $newrev` `git update-ref -d $ref` ;; esac done If you'll pardon my lacking shell script skills (I'm open to learn!), my primary question concerns safety. When receiving a ref via an SSH-based server (which happens to be Gitolite, but I don't think that is relevant here), is the post-receive hook guaranteed to be run in a lockstep manner? That is, if two people push to 'refs/for/master' at the same time, is there a lock to process one user and then the other user? The auto-merging script is just simple at the moment. It runs 'git fetch origin refs/for/*:refs/for/*', sorts the refs/for/ entries by timestamp, and merges into the specified branch, emailing the user on success or failure (not implemented yet... I'm sure Gitolite gives me access to the username, but I haven't looked it up yet). Before I go too much deeper down this path, am I way off base here? Thanks. Josh ^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: Deleting of the specified ref during the post-receive hook 2010-09-21 14:40 Deleting of the specified ref during the post-receive hook Joshua Jensen @ 2010-09-21 14:50 ` Shawn O. Pearce 2010-09-21 15:55 ` Joshua Jensen 0 siblings, 1 reply; 4+ messages in thread From: Shawn O. Pearce @ 2010-09-21 14:50 UTC (permalink / raw) To: Joshua Jensen; +Cc: git@vger.kernel.org Joshua Jensen <jjensen@workspacewhiz.com> wrote: > I've recently asked whether there was a public script to act as an > automatic 'maintainer', something akin to Gerrit's push always > succeeding without having to pull first when using Git as a central > repository. I received a number of suggestions, and I have begun trial > implementations. > > My current line of thought has an auto-merging script that monitors the > refs/for/ namespace (similar to Gerrit) and then applies --no-ff merges > to the appropriate branch. For instance, when the user pushes to > refs/for/master, the post-receive hook creates a secondary ref called > refs/for/master-SHA1-timestamp and then deletes the refs/for/master ref: > > #!bin/sh > # post-receive hook > while read oldrev newrev ref > do > case $ref in > refs/for/*) > timestamp=`date +%s` > `git update-ref $ref-$newrev-$timestamp $newrev` > `git update-ref -d $ref` FWIW, you don't need `` around the git update-ref calls, because you aren't using the output of the command as input to another command. > If you'll pardon my lacking shell script skills (I'm open to learn!), my > primary question concerns safety. When receiving a ref via an SSH-based > server (which happens to be Gitolite, but I don't think that is relevant > here), is the post-receive hook guaranteed to be run in a lockstep > manner? No. If there are two concurrent pushes occurring, the script may execute in parallel. But you'll actually get something much worse. receive-pack will create refs/for/master for the first user... and then might be put to sleep while another user's receive-pack starts. That second user will see refs/for/master existing, and will fail their push because their concept of $newrev doesn't match what is currently at refs/for/master. Then the first user wakes up, runs your post-receive, and the ref is cleared. The better strategy would be to use an update hook that refuses to permit the creation of refs/for/master: #!/bin/sh ref=$1 old=$2 new=$3 case $ref in refs/for/*) timestamp=`date +%s` git update-ref $ref-$new-$timestamp $new echo "Created $ref-$new-$timestamp" exit 1 ;; *) exit 0 ;; esac By exit 1 here we refuse the push attempt, so receive-pack won't create refs/for/master, and another user pushing won't see that false failure. However, unlike with Gerrit, every user is now going to receive a push failure message because the hook has appeared to reject the value, even though it accepted it. > That is, if two people push to 'refs/for/master' at the same > time, is there a lock to process one user and then the other user? There is, but only for the duration of the update hook. So if you used an update hook instead, one user would succeed with the push and the other would get a lock failure, and would need to retry. > Before I go too much deeper down this path, am I way off base here? I'd have to ask why you are using gitolite and trying to abuse git-receive-pack to do something that Gerrit does out of the box... -- Shawn. ^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: Deleting of the specified ref during the post-receive hook 2010-09-21 14:50 ` Shawn O. Pearce @ 2010-09-21 15:55 ` Joshua Jensen 2010-09-21 16:00 ` Shawn O. Pearce 0 siblings, 1 reply; 4+ messages in thread From: Joshua Jensen @ 2010-09-21 15:55 UTC (permalink / raw) To: Shawn O. Pearce; +Cc: git@vger.kernel.org ----- Original Message ----- From: Shawn O. Pearce Date: 9/21/2010 8:50 AM > Joshua Jensen<jjensen@workspacewhiz.com> wrote: >> My current line of thought has an auto-merging script that monitors the >> refs/for/ namespace (similar to Gerrit) and then applies --no-ff merges >> to the appropriate branch. For instance, when the user pushes to >> refs/for/master, the post-receive hook creates a secondary ref called >> refs/for/master-SHA1-timestamp and then deletes the refs/for/master ref >> > FWIW, you don't need `` around the git update-ref calls, because you > aren't using the output of the command as input to another command. You are right. I believe I was early on, and it just got copied and pasted around. > No. If there are two concurrent pushes occurring, the script may > execute in parallel. Okay, so what I'm doing is bad. Got it. > But you'll actually get something much worse. receive-pack will > create refs/for/master for the first user... and then might be put > to sleep while another user's receive-pack starts. That second > user will see refs/for/master existing, and will fail their push > because their concept of $newrev doesn't match what is currently > at refs/for/master. Then the first user wakes up, runs your > post-receive, and the ref is cleared. Yep, this is exactly what I was trying to avoid by deleting the ref. > The better strategy would be to use an update hook that refuses to > permit the creation of refs/for/master: > > #!/bin/sh > ref=$1 > old=$2 > new=$3 > > case $ref in > refs/for/*) > timestamp=`date +%s` > git update-ref $ref-$new-$timestamp $new > echo "Created $ref-$new-$timestamp" > exit 1 > ;; > *) > exit 0 > ;; > esac > > > By exit 1 here we refuse the push attempt, so receive-pack won't > create refs/for/master, and another user pushing won't see that > false failure. However, unlike with Gerrit, every user is now going > to receive a push failure message because the hook has appeared to > reject the value, even though it accepted it. Okay, I'll try this. I can train people in the push failure, if necessary. Thanks! >> Before I go too much deeper down this path, am I way off base here? > I'd have to ask why you are using gitolite and trying to abuse > git-receive-pack to do something that Gerrit does out of the box... > Oh, how I would love to just leave it to Gerrit to handle this. (I *really* like Gerrit.) If you would like to just skip ahead, the actual question related to this email message is in the final paragraph. We don't want to squash our topic branches before pushing them to Gerrit. That means we end up with a range of 'n' number of commits in the Gerrit review screen. Gerrit understands that commit 'n' is dependent on 'n-1' and so on, but unfortunately, it doesn't show them in a tree/group on the main page. That's a minor gripe, but here is the major issue. When I review Change 1, I can press a submit button, and Change 1 goes live right then. Unfortunately, changes 2 through n were left behind. If I review Change 1 through n and then press the Submit button on Change n and then on Change n-1 and then on Change n-2, Gerrit does the 'right' thing (for us) and makes live the entire dependency chain at the same time. Of course, I would prefer to just have a group Submit button, but that's another story. This has come up on the mailing list a few times, and I even think there is an issue tracker item for it. Let's ignore the review+submit portion of Gerrit now. Can Gerrit be coaxed to take a refs/for/master and directly apply it to the master branch WITHOUT the review cycle? If so, then I'm wasting my time trying to right a script. Josh ^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: Deleting of the specified ref during the post-receive hook 2010-09-21 15:55 ` Joshua Jensen @ 2010-09-21 16:00 ` Shawn O. Pearce 0 siblings, 0 replies; 4+ messages in thread From: Shawn O. Pearce @ 2010-09-21 16:00 UTC (permalink / raw) To: Joshua Jensen; +Cc: git@vger.kernel.org Joshua Jensen <jjensen@workspacewhiz.com> wrote: > Let's ignore the review+submit portion of Gerrit now. Can Gerrit be > coaxed to take a refs/for/master and directly apply it to the master > branch WITHOUT the review cycle? If so, then I'm wasting my time trying > to right a script. Nope. But only because nobody has proposed doing that before. Its not impossible to do... -- Shawn. ^ permalink raw reply [flat|nested] 4+ messages in thread
end of thread, other threads:[~2010-09-21 16:00 UTC | newest] Thread overview: 4+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2010-09-21 14:40 Deleting of the specified ref during the post-receive hook Joshua Jensen 2010-09-21 14:50 ` Shawn O. Pearce 2010-09-21 15:55 ` Joshua Jensen 2010-09-21 16:00 ` Shawn O. Pearce
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).