从PHP读取Git提交消息

时间:2012-01-31 22:21:02

标签: php git

我正在寻找一种用PHP读取Git提交消息的方法。我怀疑我需要使用Git钩子,但我以前从未使用它们,所以我需要在正确的方向上按 push 。具体来说,我想实现以下过程:

  • 每次提交后都会自动执行PHP脚本
  • 该脚本捕获Git用户名,提交时间和提交内容

如果可能的话,我想坚持使用纯PHP。如果有可以指出的教程或参考资料,那将是一个巨大的帮助。

4 个答案:

答案 0 :(得分:22)

要获取提交哈希,您可以使用

git rev-parse --verify HEAD 2> /dev/null

来自php:

exec('git rev-parse --verify HEAD 2> /dev/null', $output);
$hash = $output[0];

您可以获取提交消息,作者和时间(但是 - 如果它作为提交后挂钩的一部分运行,时间将只是“现在”):

exec("git show $hash", $output);

如果不是很明显,那么无论你使用php做什么都只是围绕你在cli上用git做的事情的包装 - 即。任何“我怎么能用PHP的git做x”只是exec('the git answer', $output)

答案 1 :(得分:7)

就使用PHP提取正确的提交而言:

Indefero

有一个名为Indefero的项目,它是一个具有SCM connector for git的PHP forge工具。您可以轻松地将their git class用作自己的API。您可以抓住git classSCM class

例如,我已经从下面的课程中提取了两种方法,我认为这两种方法与您最相关,因此您可以看到它们是如何工作的。

获取更改日志列表:getChangeLog()

/**
 * Get latest changes.
 *
 * @param string Commit ('HEAD').
 * @param int Number of changes (10).
 * @return array Changes.
 */
public function getChangeLog($commit='HEAD', $n=10)
{
    if ($n === null) $n = '';
    else $n = ' -'.$n;
    $cmd = sprintf('GIT_DIR=%s '.Pluf::f('git_path', 'git').' log%s --date=iso --pretty=format:\'%s\' %s',
                   escapeshellarg($this->repo), $n, $this->mediumtree_fmt,
                   escapeshellarg($commit));
    $out = array();
    $cmd = Pluf::f('idf_exec_cmd_prefix', '').$cmd;
    self::exec('IDF_Scm_Git::getChangeLog', $cmd, $out);
    return self::parseLog($out);
}

获取特定提交:getCommit()

/**
 * Get commit details.
 *
 * @param string Commit
 * @param bool Get commit diff (false)
 * @return array Changes
 */
public function getCommit($commit, $getdiff=false)
{
    if ($getdiff) {
        $cmd = sprintf('GIT_DIR=%s '.Pluf::f('git_path', 'git').' show --date=iso --pretty=format:%s %s',
                       escapeshellarg($this->repo),
                       "'".$this->mediumtree_fmt."'",
                       escapeshellarg($commit));
    } else {
        $cmd = sprintf('GIT_DIR=%s '.Pluf::f('git_path', 'git').' log -1 --date=iso --pretty=format:%s %s',
                       escapeshellarg($this->repo),
                       "'".$this->mediumtree_fmt."'",
                       escapeshellarg($commit));
    }
    $out = array();
    $cmd = Pluf::f('idf_exec_cmd_prefix', '').$cmd;
    self::exec('IDF_Scm_Git::getCommit', $cmd, $out, $ret);
    if ($ret != 0 or count($out) == 0) {
        return false;
    }
    if ($getdiff) {
        $log = array();
        $change = array();
        $inchange = false;
        foreach ($out as $line) {
            if (!$inchange and 0 === strpos($line, 'diff --git a')) {
                $inchange = true;
            }
            if ($inchange) {
                $change[] = $line;
            } else {
                $log[] = $line;
            }
        }
        $out = self::parseLog($log);
        $out[0]->diff = implode("\n", $change);
    } else {
        $out = self::parseLog($out);
        $out[0]->diff = '';
    }

    $out[0]->branch = implode(', ', $this->inBranches($commit, null));
    return $out[0];
}
来自PEAR的

VersionControl_Git

在PEAR中还有一个名为VersionControl_Git的图书馆,在这种情况下会很有用,并且是documented

答案 2 :(得分:4)

正如@Pawel所提到的,你将会想要使用钩子:

  

在localhost上,您可以导航到/.git/hooks并重命名   post-commit.sample to post-commit然后放入#!/ usr / bin / php    还有其他钩子可能更多   适合你。

Git会在您提交后查找post-commit挂钩并自动运行任何内容。

你要在这里做什么取决于任务,但我建议curl脚本 - 这里的事情变得有趣。

为了提取您正在寻找的信息,您将要使用git log -1 - 这应该撤回最新的提交。

更具体地说,您需要使用--pretty=format切换来构建提交,这可以使用您需要的信息输出最新提交。看看这个字符串:

git log -1 --pretty=format:'%h - %cn (%ce) - %s (%ci)'

这将返回您正在寻找的大部分内容。查看git-log页面,查看可与%一起使用的所有不同--pretty=format:变量。一旦你创建了你想要的字符串,你可以通过cURL POST那些PHP脚本,或运行PHP脚本并使用shell_exec(git log -1 --pretty=format:'%h - %cn (%ce) - %s (%ci)')来处理内联提交消息。

答案 3 :(得分:1)

我正在挖掘同样的问题,并找到了一种更快更轻松的方法。

要获得可以使用的提交消息

git rev-list --format=%B --max-count=1 HEAD

显然HEAD可以替换为任何提交哈希。

它将输出类似

的内容
commit 4152601a42270440ad52680ac7c66ba87a506174
Improved migrations and models relations

第二行是你需要的。