1
0
Fork 0
mirror of https://we.phorge.it/source/arcanist.git synced 2025-01-26 06:28:19 +01:00

(stable) Promote 2020 Week 17

This commit is contained in:
epriestley 2020-05-01 12:57:49 -07:00
commit 31c6b56b67
12 changed files with 362 additions and 160 deletions

View file

@ -154,6 +154,8 @@ phutil_register_library_map(array(
'ArcanistDiffParserTestCase' => 'parser/__tests__/ArcanistDiffParserTestCase.php', 'ArcanistDiffParserTestCase' => 'parser/__tests__/ArcanistDiffParserTestCase.php',
'ArcanistDiffUtils' => 'difference/ArcanistDiffUtils.php', 'ArcanistDiffUtils' => 'difference/ArcanistDiffUtils.php',
'ArcanistDiffUtilsTestCase' => 'difference/__tests__/ArcanistDiffUtilsTestCase.php', 'ArcanistDiffUtilsTestCase' => 'difference/__tests__/ArcanistDiffUtilsTestCase.php',
'ArcanistDiffVectorNode' => 'difference/ArcanistDiffVectorNode.php',
'ArcanistDiffVectorTree' => 'difference/ArcanistDiffVectorTree.php',
'ArcanistDiffWorkflow' => 'workflow/ArcanistDiffWorkflow.php', 'ArcanistDiffWorkflow' => 'workflow/ArcanistDiffWorkflow.php',
'ArcanistDifferentialCommitMessage' => 'differential/ArcanistDifferentialCommitMessage.php', 'ArcanistDifferentialCommitMessage' => 'differential/ArcanistDifferentialCommitMessage.php',
'ArcanistDifferentialCommitMessageParserException' => 'differential/ArcanistDifferentialCommitMessageParserException.php', 'ArcanistDifferentialCommitMessageParserException' => 'differential/ArcanistDifferentialCommitMessageParserException.php',
@ -691,7 +693,6 @@ phutil_register_library_map(array(
'PhutilExecutionEnvironment' => 'utils/PhutilExecutionEnvironment.php', 'PhutilExecutionEnvironment' => 'utils/PhutilExecutionEnvironment.php',
'PhutilFileLock' => 'filesystem/PhutilFileLock.php', 'PhutilFileLock' => 'filesystem/PhutilFileLock.php',
'PhutilFileLockTestCase' => 'filesystem/__tests__/PhutilFileLockTestCase.php', 'PhutilFileLockTestCase' => 'filesystem/__tests__/PhutilFileLockTestCase.php',
'PhutilFileTree' => 'filesystem/PhutilFileTree.php',
'PhutilFrenchLocale' => 'internationalization/locales/PhutilFrenchLocale.php', 'PhutilFrenchLocale' => 'internationalization/locales/PhutilFrenchLocale.php',
'PhutilGermanLocale' => 'internationalization/locales/PhutilGermanLocale.php', 'PhutilGermanLocale' => 'internationalization/locales/PhutilGermanLocale.php',
'PhutilGitBinaryAnalyzer' => 'filesystem/binary/PhutilGitBinaryAnalyzer.php', 'PhutilGitBinaryAnalyzer' => 'filesystem/binary/PhutilGitBinaryAnalyzer.php',
@ -1134,6 +1135,8 @@ phutil_register_library_map(array(
'ArcanistDiffParserTestCase' => 'PhutilTestCase', 'ArcanistDiffParserTestCase' => 'PhutilTestCase',
'ArcanistDiffUtils' => 'Phobject', 'ArcanistDiffUtils' => 'Phobject',
'ArcanistDiffUtilsTestCase' => 'PhutilTestCase', 'ArcanistDiffUtilsTestCase' => 'PhutilTestCase',
'ArcanistDiffVectorNode' => 'Phobject',
'ArcanistDiffVectorTree' => 'Phobject',
'ArcanistDiffWorkflow' => 'ArcanistWorkflow', 'ArcanistDiffWorkflow' => 'ArcanistWorkflow',
'ArcanistDifferentialCommitMessage' => 'Phobject', 'ArcanistDifferentialCommitMessage' => 'Phobject',
'ArcanistDifferentialCommitMessageParserException' => 'Exception', 'ArcanistDifferentialCommitMessageParserException' => 'Exception',
@ -1702,7 +1705,6 @@ phutil_register_library_map(array(
'PhutilExecutionEnvironment' => 'Phobject', 'PhutilExecutionEnvironment' => 'Phobject',
'PhutilFileLock' => 'PhutilLock', 'PhutilFileLock' => 'PhutilLock',
'PhutilFileLockTestCase' => 'PhutilTestCase', 'PhutilFileLockTestCase' => 'PhutilTestCase',
'PhutilFileTree' => 'Phobject',
'PhutilFrenchLocale' => 'PhutilLocale', 'PhutilFrenchLocale' => 'PhutilLocale',
'PhutilGermanLocale' => 'PhutilLocale', 'PhutilGermanLocale' => 'PhutilLocale',
'PhutilGitBinaryAnalyzer' => 'PhutilBinaryAnalyzer', 'PhutilGitBinaryAnalyzer' => 'PhutilBinaryAnalyzer',

View file

@ -234,21 +234,8 @@ EOTEXT
$ref_uri = head($ref_uris); $ref_uri = head($ref_uris);
// TODO: "ArcanistRevisionRef", at least, may return a relative URI.
// If we get a relative URI, guess the correct absolute URI based on
// the Conduit URI. This might not be correct for Conduit over SSH.
$raw_uri = $ref_uri->getURI(); $raw_uri = $ref_uri->getURI();
$raw_uri = $this->getAbsoluteURI($raw_uri);
$raw_uri = new PhutilURI($raw_uri);
if (!strlen($raw_uri->getDomain())) {
$base_uri = $this->getConduitEngine()
->getConduitURI();
$raw_uri = id(new PhutilURI($base_uri))
->setPath($raw_uri->getPath());
}
$raw_uri = phutil_string_cast($raw_uri);
$uris[] = $raw_uri; $uris[] = $raw_uri;
} }

View file

@ -0,0 +1,113 @@
<?php
final class ArcanistDiffVectorNode
extends Phobject {
private $vector;
private $children = array();
private $parentNode;
private $displayNode;
private $displayVector;
private $displayDepth;
private $valueNode;
private $attributes = array();
public function setVector(array $vector) {
$this->vector = $vector;
return $this;
}
public function getVector() {
return $this->vector;
}
public function getChildren() {
return $this->children;
}
public function setParentNode(ArcanistDiffVectorNode $parent) {
$this->parentNode = $parent;
return $this;
}
public function getParentNode() {
return $this->parentNode;
}
public function addChild(array $vector, $length, $idx) {
$is_node = ($idx === ($length - 1));
$element = $vector[$idx];
if (!isset($this->children[$element])) {
$this->children[$element] = id(new self())
->setParentNode($this)
->setVector(array_slice($vector, 0, $idx + 1));
}
$child = $this->children[$element];
if ($is_node) {
$child->setValueNode($child);
return;
}
$child->addChild($vector, $length, $idx + 1);
}
public function getDisplayVector() {
return $this->displayVector;
}
public function appendDisplayElement($element) {
if ($this->displayVector === null) {
$this->displayVector = array();
}
$this->displayVector[] = $element;
return $this;
}
public function setDisplayNode(ArcanistDiffVectorNode $display_node) {
$this->displayNode = $display_node;
return $this;
}
public function getDisplayNode() {
return $this->displayNode;
}
public function setDisplayDepth($display_depth) {
$this->displayDepth = $display_depth;
return $this;
}
public function getDisplayDepth() {
return $this->displayDepth;
}
public function setValueNode($value_node) {
$this->valueNode = $value_node;
return $this;
}
public function getValueNode() {
return $this->valueNode;
}
public function setAncestralAttribute($key, $value) {
$this->attributes[$key] = $value;
$parent = $this->getParentNode();
if ($parent) {
$parent->setAncestralAttribute($key, $value);
}
return $this;
}
public function getAttribute($key, $default = null) {
return idx($this->attributes, $key, $default);
}
}

View file

@ -0,0 +1,95 @@
<?php
final class ArcanistDiffVectorTree
extends Phobject {
private $vectors;
public function addVector(array $vector) {
$this->vectors[] = $vector;
return $this;
}
public function newDisplayList() {
$root = new ArcanistDiffVectorNode();
foreach ($this->vectors as $vector) {
$root->addChild($vector, count($vector), 0);
}
foreach ($root->getChildren() as $child) {
$this->compressTree($child);
}
$root->setDisplayDepth(-1);
foreach ($root->getChildren() as $child) {
$this->updateDisplayDepth($child);
}
return $this->getDisplayList($root);
}
private function compressTree(ArcanistDiffVectorNode $node) {
$display_node = $node;
$children = $node->getChildren();
if ($children) {
$parent = $node->getParentNode();
if ($parent) {
$siblings = $parent->getChildren();
if (count($siblings) === 1) {
if (!$parent->getValueNode()) {
$parent_display = $parent->getDisplayNode();
if ($parent_display) {
$display_node = $parent_display;
if ($node->getValueNode()) {
$parent->setValueNode($node->getValueNode());
}
}
}
}
}
}
$node->setDisplayNode($display_node);
$display_element = last($node->getVector());
$display_node->appendDisplayElement($display_element);
foreach ($children as $child) {
$this->compressTree($child);
}
}
private function updateDisplayDepth(ArcanistDiffVectorNode $node) {
$parent_depth = $node->getParentNode()->getDisplayDepth();
if ($node->getDisplayVector() === null) {
$display_depth = $parent_depth;
} else {
$display_depth = $parent_depth + 1;
}
$node->setDisplayDepth($display_depth);
foreach ($node->getChildren() as $child) {
$this->updateDisplayDepth($child);
}
}
private function getDisplayList(ArcanistDiffVectorNode $node) {
$result = array();
foreach ($node->getChildren() as $child) {
if ($child->getDisplayVector() !== null) {
$result[] = $child;
}
foreach ($this->getDisplayList($child) as $item) {
$result[] = $item;
}
}
return $result;
}
}

View file

@ -1,112 +0,0 @@
<?php
/**
* Data structure for representing filesystem directory trees.
*/
final class PhutilFileTree extends Phobject {
private $name;
private $fullPath;
private $data;
private $depth = 0;
private $parentNode;
private $children = array();
public function addPath($path, $data) {
$parts = $this->splitPath($path);
$parts = array_reverse($parts);
$this->insertPath($parts, $data);
return $this;
}
public function destroy() {
$this->parentNode = null;
foreach ($this->children as $child) {
$child->destroy();
}
$this->children = array();
return $this;
}
/**
* Get the next node, iterating in depth-first order.
*/
public function getNextNode() {
if ($this->children) {
return head($this->children);
}
$cursor = $this;
while ($cursor) {
if ($cursor->getNextSibling()) {
return $cursor->getNextSibling();
}
$cursor = $cursor->parentNode;
}
return null;
}
public function getName() {
return $this->name;
}
public function getFullPath() {
return $this->fullPath;
}
public function getDepth() {
return $this->depth;
}
public function getData() {
return $this->data;
}
protected function insertPath(array $parts, $data) {
$part = array_pop($parts);
if ($part === null) {
if ($this->data) {
$full_path = $this->getFullPath();
throw new Exception(
pht("Duplicate insertion for path '%s'.", $full_path));
}
$this->data = $data;
return;
}
if (empty($this->children[$part])) {
$node = new PhutilFileTree();
$node->parentNode = $this;
$node->depth = $this->depth + 1;
$node->name = $part;
$node->fullPath = $this->parentNode ? ($this->fullPath.'/'.$part) : $part;
$this->children[$part] = $node;
}
$this->children[$part]->insertPath($parts, $data);
}
protected function splitPath($path) {
$path = trim($path, '/');
$parts = preg_split('@/+@', $path);
return $parts;
}
protected function getNextSibling() {
if (!$this->parentNode) {
return null;
}
$found = false;
foreach ($this->parentNode->children as $node) {
if ($found) {
return $node;
}
if ($this->name === $node->name) {
$found = true;
}
}
return null;
}
}

View file

@ -41,15 +41,15 @@ abstract class Future extends Phobject {
'timeout.')); 'timeout.'));
} }
if ($this->hasException()) { if (!$this->hasResult() && !$this->hasException()) {
throw $this->getException();
}
if (!$this->hasResult()) {
$graph = new FutureIterator(array($this)); $graph = new FutureIterator(array($this));
$graph->resolveAll(); $graph->resolveAll();
} }
if ($this->hasException()) {
throw $this->getException();
}
return $this->getResult(); return $this->getResult();
} }

View file

@ -713,7 +713,17 @@ final class ExecFuture extends PhutilExecutableFuture {
while (isset($this->stdin) && $this->stdin->getByteLength()) { while (isset($this->stdin) && $this->stdin->getByteLength()) {
$write_segment = $this->stdin->getAnyPrefix(); $write_segment = $this->stdin->getAnyPrefix();
try {
$bytes = fwrite($stdin, $write_segment); $bytes = fwrite($stdin, $write_segment);
} catch (RuntimeException $ex) {
// If the subprocess has exited, we may get a broken pipe error here
// in recent versions of PHP. There does not seem to be any way to
// get the actual error code other than reading the exception string.
// For now, treat this as if writes are blocked.
break;
}
if ($bytes === false) { if ($bytes === false) {
throw new Exception(pht('Unable to write to stdin!')); throw new Exception(pht('Unable to write to stdin!'));
} else if ($bytes) { } else if ($bytes) {

View file

@ -37,6 +37,16 @@ final class ArcanistFileRef
return idxv($this->parameters, array('fields', 'size')); return idxv($this->parameters, array('fields', 'size'));
} }
public function getURI() {
$uri = idxv($this->parameters, array('fields', 'uri'));
if ($uri === null) {
$uri = '/'.$this->getMonogram();
}
return $uri;
}
public function getMonogram() { public function getMonogram() {
return 'F'.$this->getID(); return 'F'.$this->getID();
} }

View file

@ -30,7 +30,13 @@ final class ArcanistPasteRef
} }
public function getURI() { public function getURI() {
return idxv($this->parameters, array('fields', 'uri')); $uri = idxv($this->parameters, array('fields', 'uri'));
if ($uri === null) {
$uri = '/'.$this->getMonogram();
}
return $uri;
} }
public function getContent() { public function getContent() {

View file

@ -9,10 +9,11 @@ final class ArcanistPasteWorkflow
public function getWorkflowInformation() { public function getWorkflowInformation() {
$help = pht(<<<EOTEXT $help = pht(<<<EOTEXT
Share and grab text using the Paste application. To create a paste, Share and grab text using the Paste application. To create a paste, use the
use stdin to provide the text: "--input" flag or provide the text on stdin:
$ cat list_of_ducks.txt | arc paste $ cat list_of_ducks.txt | arc paste
$ arc paste --input list_of_ducks.txt
To retrieve a paste, specify the paste ID: To retrieve a paste, specify the paste ID:
@ -34,8 +35,12 @@ EOTEXT
$this->newWorkflowArgument('lang') $this->newWorkflowArgument('lang')
->setParameter('language') ->setParameter('language')
->setHelp(pht('Language for the paste.')), ->setHelp(pht('Language for the paste.')),
$this->newWorkflowArgument('json') $this->newWorkflowArgument('input')
->setHelp(pht('Output in JSON format.')), ->setParameter('path')
->setIsPathArgument(true)
->setHelp(pht('Create a paste using the content in a file.')),
$this->newWorkflowArgument('browse')
->setHelp(pht('After creating a paste, open it in a web browser.')),
$this->newWorkflowArgument('argv') $this->newWorkflowArgument('argv')
->setWildcard(true), ->setWildcard(true),
); );
@ -44,6 +49,8 @@ EOTEXT
public function runWorkflow() { public function runWorkflow() {
$set_language = $this->getArgument('lang'); $set_language = $this->getArgument('lang');
$set_title = $this->getArgument('title'); $set_title = $this->getArgument('title');
$is_browse = $this->getArgument('browse');
$input_path = $this->getArgument('input');
$argv = $this->getArgument('argv'); $argv = $this->getArgument('argv');
if (count($argv) > 1) { if (count($argv) > 1) {
@ -52,6 +59,8 @@ EOTEXT
'Specify only one paste to retrieve.')); 'Specify only one paste to retrieve.'));
} }
$is_read = (count($argv) === 1);
$symbols = $this->getSymbolEngine(); $symbols = $this->getSymbolEngine();
if (count($argv) === 1) { if (count($argv) === 1) {
@ -67,6 +76,19 @@ EOTEXT
'Flag "--title" is not supported when reading pastes.')); 'Flag "--title" is not supported when reading pastes.'));
} }
if ($is_browse) {
throw new PhutilArgumentUsageException(
pht(
'Flag "--browse" is not supported when reading pastes. Use '.
'"arc browse" to browse known objects.'));
}
if ($input_path !== null) {
throw new PhutilArgumentUsageException(
pht(
'Flag "--input" is not supported when reading pastes.'));
}
$paste_symbol = $argv[0]; $paste_symbol = $argv[0];
$paste_ref = $symbols->loadPasteForSymbol($paste_symbol); $paste_ref = $symbols->loadPasteForSymbol($paste_symbol);
@ -74,7 +96,8 @@ EOTEXT
throw new PhutilArgumentUsageException( throw new PhutilArgumentUsageException(
pht( pht(
'Paste "%s" does not exist, or you do not have access '. 'Paste "%s" does not exist, or you do not have access '.
'to see it.')); 'to see it.',
$paste_symbol));
} }
echo $paste_ref->getContent(); echo $paste_ref->getContent();
@ -82,7 +105,11 @@ EOTEXT
return 0; return 0;
} }
if ($input_path === null || $input_path === '-') {
$content = $this->readStdin(); $content = $this->readStdin();
} else {
$content = Filesystem::readFile($input_path);
}
$xactions = array(); $xactions = array();
@ -121,6 +148,9 @@ EOTEXT
$paste_phid = idxv($result, array('object', 'phid')); $paste_phid = idxv($result, array('object', 'phid'));
$paste_ref = $symbols->loadPasteForSymbol($paste_phid); $paste_ref = $symbols->loadPasteForSymbol($paste_phid);
$uri = $paste_ref->getURI();
$uri = $this->getAbsoluteURI($uri);
$log = $this->getLogEngine(); $log = $this->getLogEngine();
$log->writeSuccess( $log->writeSuccess(
@ -130,7 +160,11 @@ EOTEXT
echo tsprintf( echo tsprintf(
'%s', '%s',
$paste_ref->newDisplayRef() $paste_ref->newDisplayRef()
->setURI($paste_ref->getURI())); ->setURI($uri));
if ($is_browse) {
$this->openURIsInBrowser(array($uri));
}
return 0; return 0;
} }

View file

@ -23,6 +23,10 @@ EOTEXT
return array( return array(
$this->newWorkflowArgument('json') $this->newWorkflowArgument('json')
->setHelp(pht('Output upload information in JSON format.')), ->setHelp(pht('Output upload information in JSON format.')),
$this->newWorkflowArgument('browse')
->setHelp(
pht(
'After the upload completes, open the files in a web browser.')),
$this->newWorkflowArgument('temporary') $this->newWorkflowArgument('temporary')
->setHelp( ->setHelp(
pht( pht(
@ -42,6 +46,7 @@ EOTEXT
$is_temporary = $this->getArgument('temporary'); $is_temporary = $this->getArgument('temporary');
$is_json = $this->getArgument('json'); $is_json = $this->getArgument('json');
$is_browse = $this->getArgument('browse');
$paths = $this->getArgument('paths'); $paths = $this->getArgument('paths');
$conduit = $this->getConduitEngine(); $conduit = $this->getConduitEngine();
@ -65,35 +70,68 @@ EOTEXT
$files = $uploader->uploadFiles(); $files = $uploader->uploadFiles();
$results = array(); $phids = array();
foreach ($files as $file) { foreach ($files as $file) {
// TODO: This could be handled more gracefully; just preserving behavior // TODO: This could be handled more gracefully.
// until we introduce `file.query` and modernize this.
if ($file->getErrors()) { if ($file->getErrors()) {
throw new Exception(implode("\n", $file->getErrors())); throw new Exception(implode("\n", $file->getErrors()));
} }
$phid = $file->getPHID(); $phids[] = $file->getPHID();
$name = $file->getName();
$info = $conduit->resolveCall(
'file.info',
array(
'phid' => $phid,
));
$results[$path] = $info;
if (!$is_json) {
$id = $info['id'];
echo " F{$id} {$name}: ".$info['uri']."\n\n";
} }
$symbols = $this->getSymbolEngine();
$symbol_refs = $symbols->loadFilesForSymbols($phids);
$refs = array();
foreach ($symbol_refs as $symbol_ref) {
$ref = $symbol_ref->getObject();
if ($ref === null) {
throw new Exception(
pht(
'Failed to resolve symbol ref "%s".',
$symbol_ref->getSymbol()));
}
$refs[] = $ref;
} }
if ($is_json) { if ($is_json) {
$output = id(new PhutilJSON())->encodeFormatted($results); $json = array();
echo $output;
foreach ($refs as $key => $ref) {
$uri = $ref->getURI();
$uri = $this->getAbsoluteURI($uri);
$map = array(
'argument' => $paths[$key],
'id' => $ref->getID(),
'phid' => $ref->getPHID(),
'name' => $ref->getName(),
'uri' => $uri,
);
$json[] = $map;
}
echo id(new PhutilJSON())->encodeAsList($json);
} else { } else {
$this->writeStatus(pht('Done.')); foreach ($refs as $ref) {
$uri = $ref->getURI();
$uri = $this->getAbsoluteURI($uri);
echo tsprintf(
'%s',
$ref->newDisplayRef()
->setURI($uri));
}
}
if ($is_browse) {
$uris = array();
foreach ($refs as $ref) {
$uri = $ref->getURI();
$uri = $this->getAbsoluteURI($uri);
$uris[] = $uri;
}
$this->openURIsInBrowser($uris);
} }
return 0; return 0;

View file

@ -2421,4 +2421,23 @@ abstract class ArcanistWorkflow extends Phobject {
return $stdin->read(); return $stdin->read();
} }
protected function getAbsoluteURI($raw_uri) {
// TODO: "ArcanistRevisionRef", at least, may return a relative URI.
// If we get a relative URI, guess the correct absolute URI based on
// the Conduit URI. This might not be correct for Conduit over SSH.
$raw_uri = new PhutilURI($raw_uri);
if (!strlen($raw_uri->getDomain())) {
$base_uri = $this->getConduitEngine()
->getConduitURI();
$raw_uri = id(new PhutilURI($base_uri))
->setPath($raw_uri->getPath());
}
$raw_uri = phutil_string_cast($raw_uri);
return $raw_uri;
}
} }