mirror of
https://we.phorge.it/source/phorge.git
synced 2024-12-30 09:20:58 +01:00
Show blame info in diffusion.
Summary: Show blame info. This is part of the task of "Port Diffusion's Browse File view to Phabricator". The color for git repository is not implemented yet. Test Plan: it would work for both git and svn. Reviewed By: epriestley Reviewers: epriestley CC: epriestley Differential Revision: 87
This commit is contained in:
parent
f249ffc882
commit
51a6ce65aa
7 changed files with 252 additions and 82 deletions
|
@ -23,12 +23,11 @@ class DiffusionBrowseFileController extends DiffusionController {
|
||||||
// Build the view selection form.
|
// Build the view selection form.
|
||||||
$select_map = array(
|
$select_map = array(
|
||||||
'highlighted' => 'View as Highlighted Text',
|
'highlighted' => 'View as Highlighted Text',
|
||||||
//'blame' => 'View as Highlighted Text with Blame',
|
'blame' => 'View as Highlighted Text with Blame',
|
||||||
'plain' => 'View as Plain Text',
|
'plain' => 'View as Plain Text',
|
||||||
//'plainblame' => 'View as Plain Text with Blame',
|
'plainblame' => 'View as Plain Text with Blame',
|
||||||
);
|
);
|
||||||
|
|
||||||
$drequest = $this->getDiffusionRequest();
|
|
||||||
$request = $this->getRequest();
|
$request = $this->getRequest();
|
||||||
|
|
||||||
$selected = $request->getStr('view');
|
$selected = $request->getStr('view');
|
||||||
|
@ -58,79 +57,8 @@ class DiffusionBrowseFileController extends DiffusionController {
|
||||||
'<button>view</button>');
|
'<button>view</button>');
|
||||||
$view_select_panel->appendChild($view_select_form);
|
$view_select_panel->appendChild($view_select_form);
|
||||||
|
|
||||||
$file_query = DiffusionFileContentQuery::newFromDiffusionRequest(
|
|
||||||
$this->diffusionRequest);
|
|
||||||
$file_content = $file_query->loadFileContent();
|
|
||||||
|
|
||||||
// Build the content of the file.
|
// Build the content of the file.
|
||||||
// TODO: image
|
$corpus = $this->buildCorpus($selected);
|
||||||
// TODO: blame.
|
|
||||||
switch ($selected) {
|
|
||||||
case 'plain':
|
|
||||||
$style =
|
|
||||||
"margin: 1em 2em; width: 90%; height: 80em; font-family: monospace";
|
|
||||||
$corpus = phutil_render_tag(
|
|
||||||
'textarea',
|
|
||||||
array(
|
|
||||||
'style' => $style,
|
|
||||||
),
|
|
||||||
phutil_escape_html($file_content->getCorpus()));
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'highlighted':
|
|
||||||
default:
|
|
||||||
require_celerity_resource('syntax-highlighting-css');
|
|
||||||
require_celerity_resource('diffusion-source-css');
|
|
||||||
|
|
||||||
$path = $drequest->getPath();
|
|
||||||
$highlightEngine = new PhutilDefaultSyntaxHighlighterEngine();
|
|
||||||
$data = $highlightEngine->highlightSource($path,
|
|
||||||
$file_content->getCorpus());
|
|
||||||
$data = explode("\n", rtrim($data));
|
|
||||||
|
|
||||||
$uri_path = $drequest->getUriPath();
|
|
||||||
$uri_rev = $drequest->getCommit();
|
|
||||||
|
|
||||||
$color = null;
|
|
||||||
$rows = array();
|
|
||||||
$n = 1;
|
|
||||||
foreach ($data as $k => $line) {
|
|
||||||
if ($n == $drequest->getLine()) {
|
|
||||||
$tr = '<tr style="background: #ffff00;">';
|
|
||||||
$targ = '<a id="scroll_target"></a>';
|
|
||||||
Javelin::initBehavior('diffusion-jump-to',
|
|
||||||
array('target' => 'scroll_target'));
|
|
||||||
} else {
|
|
||||||
$tr = '<tr>';
|
|
||||||
$targ = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$l = phutil_render_tag(
|
|
||||||
'a',
|
|
||||||
array(
|
|
||||||
'href' => $uri_path.';'.$uri_rev.'$'.$n,
|
|
||||||
),
|
|
||||||
$n);
|
|
||||||
|
|
||||||
$rows[] = $tr.'<th>'.$l.'</th><td>'.$targ.$line.'</td></tr>';
|
|
||||||
++$n;
|
|
||||||
}
|
|
||||||
|
|
||||||
$corpus_table = phutil_render_tag(
|
|
||||||
'table',
|
|
||||||
array(
|
|
||||||
'class' => "diffusion-source remarkup-code",
|
|
||||||
),
|
|
||||||
implode("\n", $rows));
|
|
||||||
$corpus = phutil_render_tag(
|
|
||||||
'div',
|
|
||||||
array(
|
|
||||||
'style' => 'padding: 0pt 2em;',
|
|
||||||
),
|
|
||||||
$corpus_table);
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Render the page.
|
// Render the page.
|
||||||
$content = array();
|
$content = array();
|
||||||
|
@ -152,4 +80,152 @@ class DiffusionBrowseFileController extends DiffusionController {
|
||||||
'title' => 'Browse',
|
'title' => 'Browse',
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static function renderRevision(
|
||||||
|
DiffusionRequest $drequest,
|
||||||
|
$revision) {
|
||||||
|
|
||||||
|
$callsign = $drequest->getCallsign();
|
||||||
|
|
||||||
|
$name = 'r'.$callsign.$revision;
|
||||||
|
return phutil_render_tag(
|
||||||
|
'a',
|
||||||
|
array(
|
||||||
|
'href' => '/'.$name,
|
||||||
|
),
|
||||||
|
$name
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private function buildCorpus($selected) {
|
||||||
|
$blame = ($selected == 'blame' || $selected == 'plainblame');
|
||||||
|
|
||||||
|
$file_query = DiffusionFileContentQuery::newFromDiffusionRequest(
|
||||||
|
$this->diffusionRequest);
|
||||||
|
$file_query->setNeedsBlame($blame);
|
||||||
|
|
||||||
|
// TODO: image
|
||||||
|
// TODO: blame of blame.
|
||||||
|
switch ($selected) {
|
||||||
|
case 'plain':
|
||||||
|
case 'plainblame':
|
||||||
|
$style =
|
||||||
|
"margin: 1em 2em; width: 90%; height: 80em; font-family: monospace";
|
||||||
|
$corpus = phutil_render_tag(
|
||||||
|
'textarea',
|
||||||
|
array(
|
||||||
|
'style' => $style,
|
||||||
|
),
|
||||||
|
phutil_escape_html($file_query->getRawData()));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'highlighted':
|
||||||
|
case 'blame':
|
||||||
|
default:
|
||||||
|
require_celerity_resource('syntax-highlighting-css');
|
||||||
|
require_celerity_resource('diffusion-source-css');
|
||||||
|
|
||||||
|
list($data, $blamedata, $revs) = $file_query->getTokenizedData();
|
||||||
|
|
||||||
|
$drequest = $this->getDiffusionRequest();
|
||||||
|
$path = $drequest->getPath();
|
||||||
|
$highlightEngine = new PhutilDefaultSyntaxHighlighterEngine();
|
||||||
|
$data = $highlightEngine->highlightSource($path, $data);
|
||||||
|
|
||||||
|
$data = explode("\n", rtrim($data));
|
||||||
|
|
||||||
|
$rows = $this->buildDisplayRows($data, $blame, $blamedata, $drequest,
|
||||||
|
$revs);
|
||||||
|
|
||||||
|
$corpus_table = phutil_render_tag(
|
||||||
|
'table',
|
||||||
|
array(
|
||||||
|
'class' => "diffusion-source remarkup-code",
|
||||||
|
),
|
||||||
|
implode("\n", $rows));
|
||||||
|
$corpus = phutil_render_tag(
|
||||||
|
'div',
|
||||||
|
array(
|
||||||
|
'style' => 'padding: 0pt 2em;',
|
||||||
|
),
|
||||||
|
$corpus_table);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $corpus;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static function buildDisplayRows($data, $blame, $blamedata, $drequest,
|
||||||
|
$revs) {
|
||||||
|
$last = null;
|
||||||
|
$color = null;
|
||||||
|
$rows = array();
|
||||||
|
$n = 1;
|
||||||
|
foreach ($data as $k => $line) {
|
||||||
|
if ($blame) {
|
||||||
|
if ($last == $blamedata[$k][0]) {
|
||||||
|
$blameinfo =
|
||||||
|
'<th style="background: '.$color.'; width: 9em;"></th>'.
|
||||||
|
'<th style="background: '.$color.'"></th>';
|
||||||
|
} else {
|
||||||
|
switch ($drequest->getRepository()->getVersionControlSystem()) {
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_GIT:
|
||||||
|
// TODO: better color for git.
|
||||||
|
$color = '#dddddd';
|
||||||
|
break;
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_SVN:
|
||||||
|
$color = sprintf(
|
||||||
|
'#%02xee%02x',
|
||||||
|
$revs[$blamedata[$k][0]],
|
||||||
|
$revs[$blamedata[$k][0]]);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Exception('repository type not supported');
|
||||||
|
}
|
||||||
|
$revision_link = self::renderRevision(
|
||||||
|
$drequest,
|
||||||
|
$blamedata[$k][0]);
|
||||||
|
|
||||||
|
$author_link = $blamedata[$k][1];
|
||||||
|
$blameinfo =
|
||||||
|
'<th style="background: '.$color.
|
||||||
|
'; width: 9em;">'.$revision_link.'</th>'.
|
||||||
|
'<th style="background: '.$color.
|
||||||
|
'; font-weight: normal; color: #333;">'.$author_link.'</th>';
|
||||||
|
$last = $blamedata[$k][0];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$blameinfo = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($n == $drequest->getLine()) {
|
||||||
|
$tr = '<tr style="background: #ffff00;">';
|
||||||
|
$targ = '<a id="scroll_target"></a>';
|
||||||
|
Javelin::initBehavior('diffusion-jump-to',
|
||||||
|
array('target' => 'scroll_target'));
|
||||||
|
} else {
|
||||||
|
$tr = '<tr>';
|
||||||
|
$targ = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$uri_path = $drequest->getUriPath();
|
||||||
|
$uri_rev = $drequest->getCommit();
|
||||||
|
|
||||||
|
$l = phutil_render_tag(
|
||||||
|
'a',
|
||||||
|
array(
|
||||||
|
'href' => $uri_path.';'.$uri_rev.'$'.$n,
|
||||||
|
),
|
||||||
|
$n);
|
||||||
|
|
||||||
|
$rows[] = $tr.$blameinfo.'<th>'.$l.'</th><td>'.$targ.$line.'</td></tr>';
|
||||||
|
++$n;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $rows;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
|
|
||||||
phutil_require_module('phabricator', 'applications/diffusion/controller/base');
|
phutil_require_module('phabricator', 'applications/diffusion/controller/base');
|
||||||
phutil_require_module('phabricator', 'applications/diffusion/query/filecontent/base');
|
phutil_require_module('phabricator', 'applications/diffusion/query/filecontent/base');
|
||||||
|
phutil_require_module('phabricator', 'applications/repository/constants/repositorytype');
|
||||||
phutil_require_module('phabricator', 'infrastructure/celerity/api');
|
phutil_require_module('phabricator', 'infrastructure/celerity/api');
|
||||||
phutil_require_module('phabricator', 'infrastructure/javelin/api');
|
phutil_require_module('phabricator', 'infrastructure/javelin/api');
|
||||||
phutil_require_module('phabricator', 'view/layout/panel');
|
phutil_require_module('phabricator', 'view/layout/panel');
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
abstract class DiffusionFileContentQuery {
|
abstract class DiffusionFileContentQuery {
|
||||||
|
|
||||||
private $request;
|
private $request;
|
||||||
|
private $needsBlame;
|
||||||
|
|
||||||
final private function __construct() {
|
final private function __construct() {
|
||||||
// <private>
|
// <private>
|
||||||
|
@ -57,4 +58,24 @@ abstract class DiffusionFileContentQuery {
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract protected function executeQuery();
|
abstract protected function executeQuery();
|
||||||
|
|
||||||
|
final public function getRawData() {
|
||||||
|
return $this->loadFileContent()->getCorpus();
|
||||||
|
}
|
||||||
|
|
||||||
|
final public function getTokenizedData() {
|
||||||
|
return $this->tokenizeData($this->getRawData());
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract protected function tokenizeData($data);
|
||||||
|
|
||||||
|
public function setNeedsBlame($needs_blame)
|
||||||
|
{
|
||||||
|
$this->needsBlame = $needs_blame;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getNeedsBlame()
|
||||||
|
{
|
||||||
|
return $this->needsBlame;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,12 +26,19 @@ final class DiffusionGitFileContentQuery extends DiffusionFileContentQuery {
|
||||||
$commit = $drequest->getCommit();
|
$commit = $drequest->getCommit();
|
||||||
|
|
||||||
$local_path = $repository->getDetail('local-path');
|
$local_path = $repository->getDetail('local-path');
|
||||||
|
if ($this->getNeedsBlame()) {
|
||||||
|
list($corpus) = execx(
|
||||||
|
'(cd %s && git --no-pager blame -c --date short %s -- %s)',
|
||||||
|
$local_path,
|
||||||
|
$commit,
|
||||||
|
$path);
|
||||||
|
} else {
|
||||||
list($corpus) = execx(
|
list($corpus) = execx(
|
||||||
'(cd %s && git cat-file blob %s:%s)',
|
'(cd %s && git cat-file blob %s:%s)',
|
||||||
$local_path,
|
$local_path,
|
||||||
$commit,
|
$commit,
|
||||||
$path);
|
$path);
|
||||||
|
}
|
||||||
|
|
||||||
$file_content = new DiffusionFileContent();
|
$file_content = new DiffusionFileContent();
|
||||||
$file_content->setCorpus($corpus);
|
$file_content->setCorpus($corpus);
|
||||||
|
@ -39,4 +46,35 @@ final class DiffusionGitFileContentQuery extends DiffusionFileContentQuery {
|
||||||
return $file_content;
|
return $file_content;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function tokenizeData($data) {
|
||||||
|
$m = array();
|
||||||
|
$blamedata = array();
|
||||||
|
$revs = array();
|
||||||
|
|
||||||
|
if ($this->getNeedsBlame()) {
|
||||||
|
$data = explode("\n", rtrim($data));
|
||||||
|
foreach ($data as $k => $line) {
|
||||||
|
// sample line:
|
||||||
|
// d1b4fcdd ( hzhao 2009-05-01 202)function print();
|
||||||
|
preg_match('/^\s*?(\S+?)\s*\(\s*(\S+)\s+\S+\s+\d+\)(.*)?$/',
|
||||||
|
$line, $m);
|
||||||
|
$data[$k] = idx($m, 3);
|
||||||
|
$blamedata[$k] = array($m[1], $m[2]);
|
||||||
|
|
||||||
|
$revs[$m[1]] = true;
|
||||||
|
}
|
||||||
|
$data = implode("\n", $data);
|
||||||
|
|
||||||
|
krsort($revs);
|
||||||
|
$ii = 0;
|
||||||
|
$len = count($revs);
|
||||||
|
foreach ($revs as $rev => $ignored) {
|
||||||
|
$revs[$rev] = (int)(0xEE * ($ii / $len));
|
||||||
|
++$ii;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array($data, $blamedata, $revs);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ phutil_require_module('phabricator', 'applications/diffusion/data/filecontent');
|
||||||
phutil_require_module('phabricator', 'applications/diffusion/query/filecontent/base');
|
phutil_require_module('phabricator', 'applications/diffusion/query/filecontent/base');
|
||||||
|
|
||||||
phutil_require_module('phutil', 'future/exec');
|
phutil_require_module('phutil', 'future/exec');
|
||||||
|
phutil_require_module('phutil', 'utils');
|
||||||
|
|
||||||
|
|
||||||
phutil_require_source('DiffusionGitFileContentQuery.php');
|
phutil_require_source('DiffusionGitFileContentQuery.php');
|
||||||
|
|
|
@ -29,7 +29,8 @@ final class DiffusionSvnFileContentQuery extends DiffusionFileContentQuery {
|
||||||
|
|
||||||
try {
|
try {
|
||||||
list($corpus) = execx(
|
list($corpus) = execx(
|
||||||
'svn --non-interactive cat %s%s@%s',
|
'svn --non-interactive %s %s%s@%s',
|
||||||
|
$this->getNeedsBlame() ? 'blame' : 'cat',
|
||||||
$remote_uri,
|
$remote_uri,
|
||||||
$path,
|
$path,
|
||||||
$commit);
|
$commit);
|
||||||
|
@ -55,4 +56,35 @@ final class DiffusionSvnFileContentQuery extends DiffusionFileContentQuery {
|
||||||
return $file_content;
|
return $file_content;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function tokenizeData($data)
|
||||||
|
{
|
||||||
|
$m = array();
|
||||||
|
$blamedata = array();
|
||||||
|
$revs = array();
|
||||||
|
|
||||||
|
if ($this->getNeedsBlame()) {
|
||||||
|
$data = explode("\n", rtrim($data));
|
||||||
|
foreach ($data as $k => $line) {
|
||||||
|
// sample line:
|
||||||
|
// 347498 yliu function print();
|
||||||
|
preg_match('/^\s*(\d+)\s+(\S+)(?: (.*))?$/', $line, $m);
|
||||||
|
$data[$k] = idx($m, 3);
|
||||||
|
$blamedata[$k] = array($m[1], $m[2]);
|
||||||
|
|
||||||
|
$revs[$m[1]] = true;
|
||||||
|
}
|
||||||
|
$data = implode("\n", $data);
|
||||||
|
|
||||||
|
krsort($revs);
|
||||||
|
$ii = 0;
|
||||||
|
$len = count($revs);
|
||||||
|
foreach ($revs as $rev => $ignored) {
|
||||||
|
$revs[$rev] = (int)(0xEE * ($ii / $len));
|
||||||
|
++$ii;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array($data, $blamedata, $revs);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ phutil_require_module('phabricator', 'applications/diffusion/data/filecontent');
|
||||||
phutil_require_module('phabricator', 'applications/diffusion/query/filecontent/base');
|
phutil_require_module('phabricator', 'applications/diffusion/query/filecontent/base');
|
||||||
|
|
||||||
phutil_require_module('phutil', 'future/exec');
|
phutil_require_module('phutil', 'future/exec');
|
||||||
|
phutil_require_module('phutil', 'utils');
|
||||||
|
|
||||||
|
|
||||||
phutil_require_source('DiffusionSvnFileContentQuery.php');
|
phutil_require_source('DiffusionSvnFileContentQuery.php');
|
||||||
|
|
Loading…
Reference in a new issue