2012-12-09 23:09:35 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
2014-08-04 21:02:54 +02:00
|
|
|
* Browse files or objects in the Phabricator web interface.
|
2012-12-09 23:09:35 +01:00
|
|
|
*/
|
2014-07-21 23:49:15 +02:00
|
|
|
final class ArcanistBrowseWorkflow extends ArcanistWorkflow {
|
2012-12-09 23:09:35 +01:00
|
|
|
|
|
|
|
public function getWorkflowName() {
|
|
|
|
return 'browse';
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getCommandSynopses() {
|
|
|
|
return phutil_console_format(<<<EOTEXT
|
2014-02-09 21:19:57 +01:00
|
|
|
**browse** [__options__] __path__ ...
|
2014-08-04 21:02:54 +02:00
|
|
|
**browse** [__options__] __object__ ...
|
2012-12-09 23:09:35 +01:00
|
|
|
EOTEXT
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getCommandHelp() {
|
|
|
|
return phutil_console_format(<<<EOTEXT
|
2014-02-09 21:19:57 +01:00
|
|
|
Supports: git, hg, svn
|
2014-08-04 21:02:54 +02:00
|
|
|
Open a file or object (like a task or revision) in your web browser.
|
|
|
|
|
|
|
|
$ arc browse README # Open a file in Diffusion.
|
|
|
|
$ arc browse T123 # View a task.
|
2012-12-09 23:09:35 +01:00
|
|
|
|
|
|
|
Set the 'browser' value using 'arc set-config' to select a browser. If
|
|
|
|
no browser is set, the command will try to guess which browser to use.
|
|
|
|
EOTEXT
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getArguments() {
|
|
|
|
return array(
|
|
|
|
'branch' => array(
|
|
|
|
'param' => 'branch_name',
|
2014-02-09 21:19:57 +01:00
|
|
|
'help' => pht(
|
|
|
|
'Default branch name to view on server. Defaults to "master".'),
|
2012-12-09 23:09:35 +01:00
|
|
|
),
|
2014-08-04 21:02:54 +02:00
|
|
|
'force' => array(
|
|
|
|
'help' => pht(
|
|
|
|
'Open arguments as paths, even if they do not exist in the '.
|
|
|
|
'working copy.'),
|
|
|
|
),
|
2012-12-09 23:09:35 +01:00
|
|
|
'*' => 'paths',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2014-08-04 21:02:54 +02:00
|
|
|
public function desiresWorkingCopy() {
|
2012-12-09 23:09:35 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function requiresConduit() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function requiresAuthentication() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-08-04 21:02:54 +02:00
|
|
|
public function desiresRepositoryAPI() {
|
2012-12-09 23:09:35 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function run() {
|
2014-08-04 21:02:54 +02:00
|
|
|
$console = PhutilConsole::getConsole();
|
2012-12-09 23:09:35 +01:00
|
|
|
|
2014-08-04 21:02:54 +02:00
|
|
|
$is_force = $this->getArgument('force');
|
|
|
|
|
|
|
|
$things = $this->getArgument('paths');
|
|
|
|
if (!$things) {
|
2014-02-09 21:19:57 +01:00
|
|
|
throw new ArcanistUsageException(
|
|
|
|
pht(
|
2014-08-04 21:02:54 +02:00
|
|
|
'Specify one or more paths or objects to browse. Use the command '.
|
2014-02-09 21:19:57 +01:00
|
|
|
'"arc browse ." if you want to browse this directory.'));
|
|
|
|
}
|
2014-08-04 21:02:54 +02:00
|
|
|
$things = array_fuse($things);
|
|
|
|
|
|
|
|
$objects = $this->getConduit()->callMethodSynchronous(
|
|
|
|
'phid.lookup',
|
|
|
|
array(
|
|
|
|
'names' => array_keys($things),
|
|
|
|
));
|
|
|
|
|
|
|
|
$uris = array();
|
|
|
|
foreach ($objects as $name => $object) {
|
|
|
|
$uris[] = $object['uri'];
|
|
|
|
|
|
|
|
$console->writeOut(
|
|
|
|
pht(
|
|
|
|
'Opening **%s** as an object.',
|
|
|
|
$name)."\n");
|
|
|
|
|
|
|
|
unset($things[$name]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->hasRepositoryAPI()) {
|
|
|
|
$repository_api = $this->getRepositoryAPI();
|
|
|
|
$project_root = $this->getWorkingCopy()->getProjectRoot();
|
|
|
|
|
|
|
|
foreach ($things as $key => $path) {
|
|
|
|
$path = preg_replace('/:([0-9]+)$/', '$\1', $path);
|
|
|
|
$full_path = Filesystem::resolvePath($path);
|
|
|
|
|
|
|
|
if (!$is_force && !Filesystem::pathExists($full_path)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$console->writeOut(
|
|
|
|
pht(
|
|
|
|
'Opening **%s** as a repository path.',
|
|
|
|
$key)."\n");
|
2014-02-09 21:19:57 +01:00
|
|
|
|
2014-08-04 21:02:54 +02:00
|
|
|
unset($things[$key]);
|
2012-12-09 23:09:35 +01:00
|
|
|
|
2014-08-04 21:02:54 +02:00
|
|
|
if ($full_path == $project_root) {
|
|
|
|
$path = '';
|
|
|
|
} else {
|
|
|
|
$path = Filesystem::readablePath($full_path, $project_root);
|
|
|
|
}
|
|
|
|
|
|
|
|
$base_uri = $this->getBaseURI();
|
|
|
|
$uris[] = $base_uri.$path;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ($things) {
|
|
|
|
$console->writeOut(
|
|
|
|
pht(
|
|
|
|
"The current working directory is not a repository working ".
|
|
|
|
"copy, so remaining arguments can not be resolved as paths. ".
|
|
|
|
"To browse paths in Diffusion, run 'arc browse' from inside ".
|
|
|
|
"a working copy.")."\n");
|
2014-02-09 18:00:54 +01:00
|
|
|
}
|
2012-12-09 23:09:35 +01:00
|
|
|
}
|
|
|
|
|
2014-08-04 21:02:54 +02:00
|
|
|
foreach ($things as $thing) {
|
|
|
|
$console->writeOut(
|
|
|
|
pht(
|
|
|
|
'Unable to find an object named **%s**, and no such path exists '.
|
|
|
|
'in the working copy. Use __--force__ to treat this as a path '.
|
|
|
|
'anyway.',
|
|
|
|
$thing)."\n");
|
|
|
|
}
|
2012-12-09 23:09:35 +01:00
|
|
|
|
2014-08-04 21:02:54 +02:00
|
|
|
if ($uris) {
|
|
|
|
$browser = $this->getBrowserCommand();
|
|
|
|
foreach ($uris as $uri) {
|
|
|
|
$err = phutil_passthru('%s %s', $browser, $uri);
|
|
|
|
if ($err) {
|
|
|
|
throw new ArcanistUsageException(
|
|
|
|
pht(
|
|
|
|
"Failed to execute browser ('%s'). Check your 'browser' config ".
|
|
|
|
"option."));
|
|
|
|
}
|
2012-12-09 23:09:35 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getBaseURI() {
|
2014-02-09 17:55:16 +01:00
|
|
|
$repo_uri = $this->getRepositoryURI();
|
2014-02-09 21:19:57 +01:00
|
|
|
if ($repo_uri === null) {
|
|
|
|
throw new ArcanistUsageException(
|
|
|
|
pht(
|
|
|
|
'arc is unable to determine which repository in Diffusion '.
|
|
|
|
'this working copy belongs to. Use "arc which" to understand how '.
|
|
|
|
'arc looks for a repository.'));
|
|
|
|
}
|
|
|
|
|
2012-12-09 23:09:35 +01:00
|
|
|
$branch = $this->getArgument('branch', 'master');
|
|
|
|
|
2014-02-09 17:55:16 +01:00
|
|
|
return $repo_uri.'browse/'.$branch.'/';
|
2012-12-09 23:09:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private function getBrowserCommand() {
|
2013-10-19 01:10:06 +02:00
|
|
|
$config = $this->getConfigFromAnySource('browser');
|
2012-12-09 23:09:35 +01:00
|
|
|
if ($config) {
|
|
|
|
return $config;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (phutil_is_windows()) {
|
2014-02-09 21:19:57 +01:00
|
|
|
return 'start';
|
2012-12-09 23:09:35 +01:00
|
|
|
}
|
|
|
|
|
2014-02-09 21:19:57 +01:00
|
|
|
$candidates = array('sensible-browser', 'xdg-open', 'open');
|
|
|
|
|
|
|
|
// NOTE: The "open" command works well on OS X, but on many Linuxes "open"
|
|
|
|
// exists and is not a browser. For now, we're just looking for other
|
|
|
|
// commands first, but we might want to be smarter about selecting "open"
|
|
|
|
// only on OS X.
|
2012-12-09 23:09:35 +01:00
|
|
|
|
|
|
|
foreach ($candidates as $cmd) {
|
2014-02-09 21:19:57 +01:00
|
|
|
if (Filesystem::binaryExists($cmd)) {
|
2012-12-09 23:09:35 +01:00
|
|
|
return $cmd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new ArcanistUsageException(
|
2014-02-09 21:19:57 +01:00
|
|
|
pht(
|
|
|
|
"Unable to find a browser command to run. Set 'browser' in your ".
|
|
|
|
"arc config to specify one."));
|
2012-12-09 23:09:35 +01:00
|
|
|
}
|
2014-07-09 01:12:13 +02:00
|
|
|
|
2012-12-09 23:09:35 +01:00
|
|
|
}
|