array( 'help' => 'Include closed and abandoned revisions', ), 'by-status' => array( 'help' => 'Sort branches by status instead of time.', ), ); } public function run() { $repository_api = $this->getRepositoryAPI(); if (!($repository_api instanceof ArcanistGitAPI)) { throw new ArcanistUsageException( 'arc branch is only supported under git.'); } $branches = $repository_api->getAllBranches(); if (!$branches) { throw new ArcanistUsageException('No branches in this working copy.'); } $commit_map = $this->loadCommitInfo($branches, $repository_api); foreach ($branches as $key => $branch) { $branches[$key] += $commit_map[$branch['hash']]; } $revisions = $this->loadRevisions($branches); $this->printBranches($branches, $revisions); return 0; } private function loadCommitInfo( array $branches, ArcanistRepositoryAPI $repository_api) { $commits = ipull($branches, 'hash'); list($info) = $repository_api->execxLocal( 'log --format=%C %Ls --', '%H%x01%ct%x01%T%x01%s%n%b%x02', $commits); $commit_map = array(); $info = array_filter(explode("\2", trim($info))); foreach ($info as $line) { list($hash, $epoch, $tree, $text) = explode("\1", trim($line), 4); $message = ArcanistDifferentialCommitMessage::newFromRawCorpus($text); $id = $message->getRevisionID(); $commit_map[$hash] = array( 'epoch' => (int)$epoch, 'tree' => $tree, 'revisionID' => $id, ); } return $commit_map; } private function loadRevisions(array $branches) { $ids = array(); $hashes = array(); foreach ($branches as $branch) { if ($branch['revisionID']) { $ids[] = $branch['revisionID']; } $hashes[] = array('gtcm', $branch['hash']); $hashes[] = array('gttr', $branch['tree']); } $calls = array(); if ($ids) { $calls[] = $this->getConduit()->callMethod( 'differential.query', array( 'ids' => $ids, )); } if ($hashes) { $calls[] = $this->getConduit()->callMethod( 'differential.query', array( 'commitHashes' => $hashes, )); } $results = array(); foreach (Futures($calls) as $call) { $results[] = $call->resolve(); } return array_mergev($results); } private function printBranches(array $branches, array $revisions) { $revisions = ipull($revisions, null, 'id'); static $color_map = array( 'Closed' => 'cyan', 'Needs Review' => 'magenta', 'Needs Revision' => 'red', 'Accepted' => 'green', 'No Revision' => 'blue', 'Abandoned' => 'default', ); static $ssort_map = array( 'Closed' => 1, 'No Revision' => 2, 'Needs Review' => 3, 'Needs Revision' => 4, 'Accepted' => 5, ); $out = array(); foreach ($branches as $branch) { $revision = idx($revisions, idx($branch, 'revisionID')); // If we haven't identified a revision by ID, try to identify it by hash. if (!$revision) { foreach ($revisions as $rev) { $hashes = idx($rev, 'hashes', array()); foreach ($hashes as $hash) { if (($hash[0] == 'gtcm' && $hash[1] == $branch['hash']) || ($hash[0] == 'gttr' && $hash[1] == $branch['tree'])) { $revision = $rev; break; } } } } if ($revision) { $desc = 'D'.$revision['id'].': '.$revision['title']; $status = $revision['statusName']; } else { $desc = $branch['desc']; $status = 'No Revision'; } if (!$this->getArgument('view-all') && !$branch['current']) { if ($status == 'Closed' || $status == 'Abandoned') { continue; } } $epoch = $branch['epoch']; $color = idx($color_map, $status, 'default'); $ssort = sprintf('%d%012d', idx($ssort_map, $status, 0), $epoch); $out[] = array( 'name' => $branch['name'], 'current' => $branch['current'], 'status' => $status, 'desc' => $desc, 'color' => $color, 'esort' => $epoch, 'ssort' => $ssort, ); } $len_name = max(array_map('strlen', ipull($out, 'name'))) + 2; $len_status = max(array_map('strlen', ipull($out, 'status'))) + 2; if ($this->getArgument('by-status')) { $out = isort($out, 'ssort'); } else { $out = isort($out, 'esort'); } $console = PhutilConsole::getConsole(); foreach ($out as $line) { $color = $line['color']; $console->writeOut( "%s **%s** %s %s\n", $line['current'] ? '* ' : ' ', str_pad($line['name'], $len_name), str_pad($line['status'], $len_status), $line['desc']); } } }