我的post-receive,一个git hook,有两个命令不起作用

问题描述 投票:2回答:1

我有三个git存储库:

  1. 我的git服务器上的裸存储库
  2. 我的笔记本电脑上的本地存储库(A)(我在这台机器上开发网页。)
  3. 托管服务器上的本地存储库(B)(我将其用作http服务器。)

我在裸存储库上设置了一个post-receive,一个服务器端钩子,让存储库(B)从裸机中拉出来,当我提交并从存储库(A)推送到裸机时让git服务器发送通知邮件库。我在裸存储库上创建的post-receive文件如下:

#!/bin/sh
ssh -l MYUSER -i PRIVKEY www.HOSTINGSERVER.com "cd GITDIRECTORY;git pull"
. $(dirname $0)/post-receive-email

我还在裸存储库上设置了环境变量:

git config hooks.mailinglist "MyMailAddress"
git config hooks.announcelist "MyMailAddress"
git config hooks.emailprefix "FooBar"
echo FooBar > description

我预计存储库(B)被拉出,git服务器在我提交并推送到裸存储库时通过post-receive-email发送通知电子邮件,但是post-receive没有像我希望的那样工作。它有存储库(B)拉,但没有git服务器发送任何电子邮件。我没有在git服务器上的日志中看到任何电子邮件错误。

当我评论收到后的ssh部分时,我收到了一封电子邮件。当我注释掉收件后电子邮件部分时,存储库(B)正确拉出。当ssh部分和post-receive-email部分都处于活动状态时,post-receive仅具有存储库(B)pull并且不发送任何电子邮件。

我不知道为什么当ssh部分和post-receive-email部分同时存在时,post-receive无法正常工作。

任何提示和评论表示赞赏。先感谢您。

git githooks git-post-receive post-receive-email
1个回答
4
投票

TL; DR:ssh -n

背景事实:

  1. 后接收挂钩在其标准输入上获得输入。读取标准输入后,每个更新的参考文献将获得一行,格式为the githooks documentation
  2. 在stdin上接收输入的所有Git钩子都在管道上执行。 (我认为这是一个坏主意 - 他们应该接收他们作为文件描述符在临时文件上打开的输入 - 但这是一个单独的战斗。即使不是这种情况,你仍然需要另一个技巧。)
  3. 读取管道上可用的输入会消耗该输入。一旦阅读,它就消失了。 (顺便说一下,这就是为什么shell中内置的read效率低下的原因:一次读取输入一个字节是必要的,这样只会消耗read读取的数量。)

考虑到这一点,请考虑ssh命令,以及其documentation的摘录:

-n从/ dev / null重定向stdin(实际上,阻止从stdin读取)。

那么,如果你在没有ssh的情况下运行-n会发生什么?当然,它会读取标准输入。但是......它会读多少钱?好吧,严格来说,这取决于:ssh运行多个进程和/或线程,并且如果存在大量stdin输入并且远程端上运行的命令执行得非常快,则存在竞争,但通常, ssh读取是stdin的全部内容。

现在咨询背景事实#3,并回答问题:当ssh完成时stdin上还剩下什么?

现在,考虑一下发送电子邮件的post-receive挂钩。必须读取其标准输入以查看哪些引用已更新。它的stdin会得到什么数据?

如果你阻止ssh读取(从而消耗)所有输入,后接收钩子将能够读取(并消耗,但现在无关紧要)所有输入,因此正确运行。

© www.soinside.com 2019 - 2024. All rights reserved.