我正在寻找一种用PHP读取Git提交消息的方法。我怀疑我需要使用Git钩子,但我以前从未使用它们,所以我需要在正确的方向上按 push 。具体来说,我想实现以下过程:
如果可能的话,我想坚持使用纯PHP。如果有可以指出的教程或参考资料,那将是一个巨大的帮助。
答案 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的项目,它是一个具有SCM connector for git的PHP forge工具。您可以轻松地将their git class用作自己的API。您可以抓住git class和SCM 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的在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
第二行是你需要的。