2011-03-13 01:17:34 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
final class DiffusionGitBranchQuery extends DiffusionBranchQuery {
|
|
|
|
|
|
|
|
protected function executeQuery() {
|
|
|
|
$drequest = $this->getRequest();
|
|
|
|
$repository = $drequest->getRepository();
|
2012-05-10 23:18:49 +02:00
|
|
|
|
|
|
|
// We need to add 1 in case we pick up HEAD.
|
|
|
|
|
|
|
|
$count = $this->getOffset() + $this->getLimit() + 1;
|
2011-03-13 01:17:34 +01:00
|
|
|
|
2011-11-10 00:29:41 +01:00
|
|
|
list($stdout) = $repository->execxLocalCommand(
|
2012-05-10 09:28:19 +02:00
|
|
|
'for-each-ref %C --sort=-creatordate --format=%s refs/remotes',
|
|
|
|
$count ? '--count='.(int)$count : null,
|
|
|
|
'%(refname:short) %(objectname)'
|
|
|
|
);
|
2011-03-13 01:17:34 +01:00
|
|
|
|
2011-12-22 21:24:12 +01:00
|
|
|
$branch_list = self::parseGitRemoteBranchOutput(
|
|
|
|
$stdout,
|
|
|
|
$only_this_remote = DiffusionBranchInformation::DEFAULT_GIT_REMOTE);
|
|
|
|
|
2011-03-13 01:17:34 +01:00
|
|
|
$branches = array();
|
2011-12-22 21:24:12 +01:00
|
|
|
foreach ($branch_list as $name => $head) {
|
Allow Git repositories to track only some branches
Summary:
Some installs use Git as the backbone of a CI framework or use a Git remote to
share patches. The tracker scripts currently recognize associated revisions as
"Committed" when they appear in any branch, even if that branch is
"alincoln-personal-development_test_hack" or whatever.
To address the broadest need here, allow Git repositories to be configured to
track only certain branches instead of all branches.
This doesn't allow you to import a branch into Diffusion but ignore it in
Differential. Supporting that is somewhat technically complicated because the
parser currently goes like this:
- Look at HEAD of all branches.
- For any commits we haven't seen before, follow them back to something we
have seen (or the root).
- "Discover" everything new.
Since this doesn't track <branch, commit> pairs, we currently don't have enough
information to tell when a commit appears in a branch for the first time, so we
don't have anywhere we can put a test for whether that branch is tracked and do
the Differential hook only if it is.
However, I think this cruder patch satisfies most of the need and is simple and
obvious in its implementation.
See also D1263.
Test Plan:
- Updated a Git repository with various filters: "", "master, remote", "derp",
" ,,, master ,,,,,"
- Edited SVN and Mercurial repositories to verify they didn't get caught in
the crossfire.
- Ran daemon in debug mode on libphutil with filter "derp", got exception
about no tracked branches. Ran with filter "master", got tracking. Ran with no
filter, got tracking.
- Looked at Diffusion with "derp" and "master", saw no branches and "master"
respectively.
- Added unit tests to cover filtering logic.
Reviewers: btrahan, jungejason, nh, fratrik
Reviewed By: fratrik
CC: aran, fratrik, epriestley
Maniphest Tasks: T270
Differential Revision: https://secure.phabricator.com/D1290
2011-12-29 18:14:55 +01:00
|
|
|
if (!$repository->shouldTrackBranch($name)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2011-03-21 01:46:02 +01:00
|
|
|
$branch = new DiffusionBranchInformation();
|
|
|
|
$branch->setName($name);
|
|
|
|
$branch->setHeadCommitIdentifier($head);
|
|
|
|
$branches[] = $branch;
|
|
|
|
}
|
|
|
|
|
2012-05-10 09:28:19 +02:00
|
|
|
$offset = $this->getOffset();
|
|
|
|
if ($offset) {
|
|
|
|
$branches = array_slice($branches, $offset);
|
|
|
|
}
|
|
|
|
|
2012-05-10 23:18:49 +02:00
|
|
|
// We might have too many even after offset slicing, if there was no HEAD
|
|
|
|
// for some reason.
|
|
|
|
$limit = $this->getLimit();
|
|
|
|
if ($limit) {
|
|
|
|
$branches = array_slice($branches, 0, $limit);
|
|
|
|
}
|
|
|
|
|
2011-03-21 01:46:02 +01:00
|
|
|
return $branches;
|
|
|
|
}
|
|
|
|
|
2011-12-22 21:24:12 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse the output of 'git branch -r --verbose --no-abbrev' or similar into
|
|
|
|
* a map. For instance:
|
|
|
|
*
|
|
|
|
* array(
|
|
|
|
* 'origin/master' => '99a9c082f9a1b68c7264e26b9e552484a5ae5f25',
|
|
|
|
* );
|
|
|
|
*
|
|
|
|
* If you specify $only_this_remote, branches will be filtered to only those
|
|
|
|
* on the given remote, **and the remote name will be stripped**. For example:
|
|
|
|
*
|
|
|
|
* array(
|
|
|
|
* 'master' => '99a9c082f9a1b68c7264e26b9e552484a5ae5f25',
|
|
|
|
* );
|
|
|
|
*
|
|
|
|
* @param string stdout of git branch command.
|
|
|
|
* @param string Filter branches to those on a specific remote.
|
|
|
|
* @return map Map of 'branch' or 'remote/branch' to hash at HEAD.
|
|
|
|
*/
|
|
|
|
public static function parseGitRemoteBranchOutput(
|
|
|
|
$stdout,
|
|
|
|
$only_this_remote = null) {
|
2011-03-21 01:46:02 +01:00
|
|
|
$map = array();
|
2011-03-13 01:17:34 +01:00
|
|
|
|
|
|
|
$lines = array_filter(explode("\n", $stdout));
|
|
|
|
foreach ($lines as $line) {
|
|
|
|
$matches = null;
|
2011-03-21 01:46:02 +01:00
|
|
|
if (preg_match('/^ (\S+)\s+-> (\S+)$/', $line, $matches)) {
|
2012-05-10 09:28:19 +02:00
|
|
|
// This is a line like:
|
|
|
|
//
|
|
|
|
// origin/HEAD -> origin/master
|
|
|
|
//
|
|
|
|
// ...which we don't currently do anything interesting with, although
|
|
|
|
// in theory we could use it to automatically choose the default
|
|
|
|
// branch.
|
|
|
|
continue;
|
2011-03-21 01:46:02 +01:00
|
|
|
}
|
2012-05-10 09:28:19 +02:00
|
|
|
if (!preg_match('/^ *(\S+)\s+([a-z0-9]{40})/', $line, $matches)) {
|
2011-03-13 01:17:34 +01:00
|
|
|
throw new Exception("Failed to parse {$line}!");
|
|
|
|
}
|
2011-12-22 21:24:12 +01:00
|
|
|
|
|
|
|
$remote_branch = $matches[1];
|
|
|
|
$branch_head = $matches[2];
|
|
|
|
|
2012-05-10 09:28:19 +02:00
|
|
|
if (strpos($remote_branch, 'HEAD') !== false) {
|
|
|
|
// let's assume that no one will call their remote or branch HEAD
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2011-12-22 21:24:12 +01:00
|
|
|
if ($only_this_remote) {
|
|
|
|
$matches = null;
|
|
|
|
if (!preg_match('#^([^/]+)/(.*)$#', $remote_branch, $matches)) {
|
|
|
|
throw new Exception(
|
|
|
|
"Failed to parse remote branch '{$remote_branch}'!");
|
|
|
|
}
|
|
|
|
$remote_name = $matches[1];
|
|
|
|
$branch_name = $matches[2];
|
|
|
|
if ($remote_name != $only_this_remote) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$map[$branch_name] = $branch_head;
|
|
|
|
} else {
|
|
|
|
$map[$remote_branch] = $branch_head;
|
|
|
|
}
|
2011-03-13 01:17:34 +01:00
|
|
|
}
|
|
|
|
|
2011-03-21 01:46:02 +01:00
|
|
|
return $map;
|
2011-03-13 01:17:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|