mirror of
https://we.phorge.it/source/arcanist.git
synced 2024-11-22 06:42:41 +01:00
'arc liberate', convenience wrapper for various libphutil operations
Summary: The story for creating and maintaining libphutil libraries and modules is pretty terrible right now: you need to know a bunch of secret scripts and dark magic. Provide 'arc liberate' which endeavors to always do the right thing and put a library in the correct state. Test Plan: Ran liberate on libphutil, arcanist, phabricator; created new libphutil libraries, added classes to them, liberated everything, introduced errors etc and liberated that stuff, nothing was obviously broken in a terrible way..? Reviewed By: aran Reviewers: jungejason, tuomaspelkonen, aran CC: aran, epriestley Differential Revision: 269
This commit is contained in:
parent
ade681e0db
commit
3a559ddd13
10 changed files with 461 additions and 11 deletions
|
@ -33,6 +33,7 @@ $builtin = array(
|
||||||
'print' => true,
|
'print' => true,
|
||||||
'exit' => true,
|
'exit' => true,
|
||||||
'die' => true,
|
'die' => true,
|
||||||
|
'phutil_load_library' => true,
|
||||||
),
|
),
|
||||||
'interface' => array_fill_keys($builtin_interfaces, true),
|
'interface' => array_fill_keys($builtin_interfaces, true),
|
||||||
);
|
);
|
||||||
|
|
|
@ -19,6 +19,16 @@
|
||||||
|
|
||||||
require_once dirname(__FILE__).'/__init_script__.php';
|
require_once dirname(__FILE__).'/__init_script__.php';
|
||||||
|
|
||||||
|
$liberate_mode = false;
|
||||||
|
for ($ii = 0; $ii < $argc; $ii++) {
|
||||||
|
if ($argv[$ii] == '--find-paths-for-liberate') {
|
||||||
|
$liberate_mode = true;
|
||||||
|
unset($argv[$ii]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$argv = array_values($argv);
|
||||||
|
$argc = count($argv);
|
||||||
|
|
||||||
if ($argc != 2) {
|
if ($argc != 2) {
|
||||||
$self = basename($argv[0]);
|
$self = basename($argv[0]);
|
||||||
echo "usage: {$self} <phutil_library_root>\n";
|
echo "usage: {$self} <phutil_library_root>\n";
|
||||||
|
@ -35,6 +45,10 @@ if (!@file_exists($root.'/__phutil_library_init__.php')) {
|
||||||
throw new Exception("Provided path is not a phutil library.");
|
throw new Exception("Provided path is not a phutil library.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($liberate_mode) {
|
||||||
|
ob_start();
|
||||||
|
}
|
||||||
|
|
||||||
echo "Finding phutil modules...\n";
|
echo "Finding phutil modules...\n";
|
||||||
$files = id(new FileFinder($root))
|
$files = id(new FileFinder($root))
|
||||||
->withType('f')
|
->withType('f')
|
||||||
|
@ -81,19 +95,24 @@ if ($cache) {
|
||||||
|
|
||||||
$specs = array();
|
$specs = array();
|
||||||
|
|
||||||
$futures = array();
|
$need_update = array();
|
||||||
foreach ($signatures as $module => $signature) {
|
foreach ($signatures as $module => $signature) {
|
||||||
if (isset($signature_cache[$module]) &&
|
if (isset($signature_cache[$module]) &&
|
||||||
$signature_cache[$module]['signature'] == $signature) {
|
$signature_cache[$module]['signature'] == $signature) {
|
||||||
$specs[$module] = $signature_cache[$module];
|
$specs[$module] = $signature_cache[$module];
|
||||||
} else {
|
} else {
|
||||||
$futures[$module] = new ExecFuture(
|
$need_update[$module] = true;
|
||||||
'%s %s',
|
|
||||||
dirname(__FILE__).'/phutil_analyzer.php',
|
|
||||||
$root.'/'.$module);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$futures = array();
|
||||||
|
foreach ($need_update as $module => $ignored) {
|
||||||
|
$futures[$module] = new ExecFuture(
|
||||||
|
'%s %s',
|
||||||
|
dirname(__FILE__).'/phutil_analyzer.php',
|
||||||
|
$root.'/'.$module);
|
||||||
|
}
|
||||||
|
|
||||||
if ($futures) {
|
if ($futures) {
|
||||||
echo "Found ".count($specs)." modules in cache; ".
|
echo "Found ".count($specs)." modules in cache; ".
|
||||||
"analyzing ".count($futures)." modified modules";
|
"analyzing ".count($futures)." modified modules";
|
||||||
|
@ -180,6 +199,12 @@ echo "Writing library map file...\n";
|
||||||
|
|
||||||
Filesystem::writeFile($root.'/__phutil_library_map__.php', $map_file);
|
Filesystem::writeFile($root.'/__phutil_library_map__.php', $map_file);
|
||||||
|
|
||||||
|
if ($liberate_mode) {
|
||||||
|
ob_get_clean();
|
||||||
|
echo json_encode(array_keys($need_update))."\n";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
echo "Writing module cache...\n";
|
echo "Writing module cache...\n";
|
||||||
|
|
||||||
Filesystem::writeFile(
|
Filesystem::writeFile(
|
||||||
|
|
|
@ -36,6 +36,8 @@ phutil_register_library_map(array(
|
||||||
'ArcanistGitAPI' => 'repository/api/git',
|
'ArcanistGitAPI' => 'repository/api/git',
|
||||||
'ArcanistGitHookPreReceiveWorkflow' => 'workflow/git-hook-pre-receive',
|
'ArcanistGitHookPreReceiveWorkflow' => 'workflow/git-hook-pre-receive',
|
||||||
'ArcanistHelpWorkflow' => 'workflow/help',
|
'ArcanistHelpWorkflow' => 'workflow/help',
|
||||||
|
'ArcanistLiberateLintEngine' => 'lint/engine/liberate',
|
||||||
|
'ArcanistLiberateWorkflow' => 'workflow/liberate',
|
||||||
'ArcanistLicenseLinter' => 'lint/linter/license',
|
'ArcanistLicenseLinter' => 'lint/linter/license',
|
||||||
'ArcanistLintEngine' => 'lint/engine/base',
|
'ArcanistLintEngine' => 'lint/engine/base',
|
||||||
'ArcanistLintJSONRenderer' => 'lint/renderer',
|
'ArcanistLintJSONRenderer' => 'lint/renderer',
|
||||||
|
@ -97,6 +99,8 @@ phutil_register_library_map(array(
|
||||||
'ArcanistGitAPI' => 'ArcanistRepositoryAPI',
|
'ArcanistGitAPI' => 'ArcanistRepositoryAPI',
|
||||||
'ArcanistGitHookPreReceiveWorkflow' => 'ArcanistBaseWorkflow',
|
'ArcanistGitHookPreReceiveWorkflow' => 'ArcanistBaseWorkflow',
|
||||||
'ArcanistHelpWorkflow' => 'ArcanistBaseWorkflow',
|
'ArcanistHelpWorkflow' => 'ArcanistBaseWorkflow',
|
||||||
|
'ArcanistLiberateLintEngine' => 'ArcanistLintEngine',
|
||||||
|
'ArcanistLiberateWorkflow' => 'ArcanistBaseWorkflow',
|
||||||
'ArcanistLicenseLinter' => 'ArcanistLinter',
|
'ArcanistLicenseLinter' => 'ArcanistLinter',
|
||||||
'ArcanistLintWorkflow' => 'ArcanistBaseWorkflow',
|
'ArcanistLintWorkflow' => 'ArcanistBaseWorkflow',
|
||||||
'ArcanistLinterTestCase' => 'ArcanistPhutilTestCase',
|
'ArcanistLinterTestCase' => 'ArcanistPhutilTestCase',
|
||||||
|
|
|
@ -182,7 +182,7 @@ abstract class ArcanistLintEngine {
|
||||||
// through the lint messages and doing this load only if any of them
|
// through the lint messages and doing this load only if any of them
|
||||||
// have original/replacement text or something like that.
|
// have original/replacement text or something like that.
|
||||||
try {
|
try {
|
||||||
$this->fileData[$path] = Filesystem::readFile($path);
|
$this->fileData[$path] = Filesystem::readFile($disk_path);
|
||||||
$result->setData($this->fileData[$path]);
|
$result->setData($this->fileData[$path]);
|
||||||
} catch (FilesystemException $ex) {
|
} catch (FilesystemException $ex) {
|
||||||
// Ignore this, it's noncritical that we access this data and it
|
// Ignore this, it's noncritical that we access this data and it
|
||||||
|
|
38
src/lint/engine/liberate/ArcanistLiberateLintEngine.php
Normal file
38
src/lint/engine/liberate/ArcanistLiberateLintEngine.php
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright 2011 Facebook, Inc.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lint engine which powers 'arc liberate'.
|
||||||
|
*
|
||||||
|
* @group linter
|
||||||
|
*/
|
||||||
|
class ArcanistLiberateLintEngine extends ArcanistLintEngine {
|
||||||
|
|
||||||
|
public function buildLinters() {
|
||||||
|
// We just run the module linter, 'arc liberate' is only interested in
|
||||||
|
// building __init__.php files.
|
||||||
|
|
||||||
|
$module_linter = new ArcanistPhutilModuleLinter();
|
||||||
|
foreach ($this->getPaths() as $path) {
|
||||||
|
$module_linter->addPath($path);
|
||||||
|
}
|
||||||
|
|
||||||
|
return array($module_linter);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
13
src/lint/engine/liberate/__init__.php
Normal file
13
src/lint/engine/liberate/__init__.php
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* This file is automatically generated. Lint this module to rebuild it.
|
||||||
|
* @generated
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
phutil_require_module('arcanist', 'lint/engine/base');
|
||||||
|
phutil_require_module('arcanist', 'lint/linter/phutilmodule');
|
||||||
|
|
||||||
|
|
||||||
|
phutil_require_source('ArcanistLiberateLintEngine.php');
|
|
@ -159,7 +159,7 @@ class ArcanistPhutilModuleLinter extends ArcanistLinter {
|
||||||
}
|
}
|
||||||
|
|
||||||
$requirements = array();
|
$requirements = array();
|
||||||
foreach (Futures($futures) as $key => $future) {
|
foreach (Futures($futures)->limit(16) as $key => $future) {
|
||||||
$requirements[$key] = $future->resolveJSON();
|
$requirements[$key] = $future->resolveJSON();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -203,7 +203,7 @@ class ArcanistPhutilModuleLinter extends ArcanistLinter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (Futures($futures) as $key => $future) {
|
foreach (Futures($futures)->limit(16) as $key => $future) {
|
||||||
$dependencies[$key] = $future->resolveJSON();
|
$dependencies[$key] = $future->resolveJSON();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -378,13 +378,17 @@ class ArcanistPhutilModuleLinter extends ArcanistLinter {
|
||||||
$need_functions,
|
$need_functions,
|
||||||
$drop_modules);
|
$drop_modules);
|
||||||
$init_path = $this->getModulePathOnDisk($key).'/__init__.php';
|
$init_path = $this->getModulePathOnDisk($key).'/__init__.php';
|
||||||
$try_path = Filesystem::readablePath($init_path);
|
|
||||||
if (Filesystem::pathExists($try_path)) {
|
$root = $this->getEngine()->getWorkingCopy()->getProjectRoot();
|
||||||
|
$try_path = Filesystem::readablePath($init_path, $root);
|
||||||
|
$full_path = Filesystem::resolvePath($try_path, $root);
|
||||||
|
if (Filesystem::pathExists($full_path)) {
|
||||||
$init_path = $try_path;
|
$init_path = $try_path;
|
||||||
$old_file = Filesystem::readFile($init_path);
|
$old_file = Filesystem::readFile($full_path);
|
||||||
} else {
|
} else {
|
||||||
$old_file = '';
|
$old_file = '';
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->willLintPath($init_path);
|
$this->willLintPath($init_path);
|
||||||
$message = $this->raiseLintAtOffset(
|
$message = $this->raiseLintAtOffset(
|
||||||
null,
|
null,
|
||||||
|
|
|
@ -78,6 +78,9 @@ EOTEXT
|
||||||
if ($argument == '*') {
|
if ($argument == '*') {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
if (!empty($spec['hide'])) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
if (isset($spec['param'])) {
|
if (isset($spec['param'])) {
|
||||||
if (isset($spec['short'])) {
|
if (isset($spec['short'])) {
|
||||||
$optref[] = phutil_console_format(
|
$optref[] = phutil_console_format(
|
||||||
|
|
336
src/workflow/liberate/ArcanistLiberateWorkflow.php
Normal file
336
src/workflow/liberate/ArcanistLiberateWorkflow.php
Normal file
|
@ -0,0 +1,336 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright 2011 Facebook, Inc.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create and update libphutil libraries.
|
||||||
|
*
|
||||||
|
* This workflow is unusual and involves reexecuting 'arc liberate' as a
|
||||||
|
* subprocess with "--remap" and "--verify". This is because there is no way
|
||||||
|
* to unload or reload a library, so every process is stuck with the library
|
||||||
|
* definition it had when it first loaded. This is normally fine, but
|
||||||
|
* problematic in this case because 'arc liberate' modifies library definitions.
|
||||||
|
*
|
||||||
|
* @group workflow
|
||||||
|
*/
|
||||||
|
class ArcanistLiberateWorkflow extends ArcanistBaseWorkflow {
|
||||||
|
|
||||||
|
public function getCommandHelp() {
|
||||||
|
return phutil_console_format(<<<EOTEXT
|
||||||
|
**liberate** [__path__]
|
||||||
|
Supports: libphutil
|
||||||
|
Create or update a libphutil library, generating required metadata
|
||||||
|
files like __init__.php.
|
||||||
|
EOTEXT
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getArguments() {
|
||||||
|
return array(
|
||||||
|
'all' => array(
|
||||||
|
'help' =>
|
||||||
|
"Drop the module cache before liberating. This will completely ".
|
||||||
|
"reanalyze the entire library. Thorough, but slow!",
|
||||||
|
),
|
||||||
|
'force-update' => array(
|
||||||
|
'help' =>
|
||||||
|
"Force the library map to be updated, even in the presence of ".
|
||||||
|
"lint errors.",
|
||||||
|
),
|
||||||
|
'remap' => array(
|
||||||
|
'hide' => true,
|
||||||
|
'help' =>
|
||||||
|
"Internal. Run the remap step of liberation. You do not need to ".
|
||||||
|
"run this unless you are debugging the workflow.",
|
||||||
|
),
|
||||||
|
'verify' => array(
|
||||||
|
'hide' => true,
|
||||||
|
'help' =>
|
||||||
|
"Internal. Run the verify step of liberation. You do not need to ".
|
||||||
|
"run this unless you are debugging the workflow.",
|
||||||
|
),
|
||||||
|
'*' => 'argv',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function run() {
|
||||||
|
$argv = $this->getArgument('argv');
|
||||||
|
if (count($argv) > 1) {
|
||||||
|
throw new ArcanistUsageException(
|
||||||
|
"Provide only one path to 'arc liberate'. The path should be a ".
|
||||||
|
"directory where you want to create or update a libphutil library.");
|
||||||
|
} else if (count($argv) == 0) {
|
||||||
|
$path = getcwd();
|
||||||
|
} else {
|
||||||
|
$path = reset($argv);
|
||||||
|
}
|
||||||
|
|
||||||
|
$is_remap = $this->getArgument('remap');
|
||||||
|
$is_verify = $this->getArgument('verify');
|
||||||
|
|
||||||
|
$path = Filesystem::resolvePath($path);
|
||||||
|
|
||||||
|
if (Filesystem::pathExists($path) && is_dir($path)) {
|
||||||
|
$init = id(new FileFinder($path))
|
||||||
|
->withPath('*/__phutil_library_init__.php')
|
||||||
|
->find();
|
||||||
|
} else {
|
||||||
|
$init = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($init) {
|
||||||
|
if (count($init) > 1) {
|
||||||
|
throw new ArcanistUsageException(
|
||||||
|
"Specified directory contains more than one libphutil library. Use ".
|
||||||
|
"a more specific path.");
|
||||||
|
}
|
||||||
|
$path = Filesystem::resolvePath(dirname(reset($init)), $path);
|
||||||
|
} else {
|
||||||
|
$found = false;
|
||||||
|
foreach (Filesystem::walkToRoot($path) as $dir) {
|
||||||
|
if (Filesystem::pathExists($dir.'/__phutil_library_init__.php')) {
|
||||||
|
$path = $dir;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!$found) {
|
||||||
|
echo "No library currently exists at that path...\n";
|
||||||
|
$this->liberateCreateDirectory($path);
|
||||||
|
$this->liberateCreateLibrary($path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->getArgument('remap')) {
|
||||||
|
return $this->liberateRunRemap($path);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->getArgument('verify')) {
|
||||||
|
return $this->liberateRunVerify($path);
|
||||||
|
}
|
||||||
|
|
||||||
|
$readable = Filesystem::readablePath($path);
|
||||||
|
echo "Using library root at '{$readable}'...\n";
|
||||||
|
|
||||||
|
if ($this->getArgument('all')) {
|
||||||
|
echo "Dropping module cache...\n";
|
||||||
|
Filesystem::remove($path.'/.phutil_module_cache');
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "Mapping library...\n";
|
||||||
|
|
||||||
|
// Force a rebuild of the library map before running lint. The remap
|
||||||
|
// operation will load the map before regenerating it, so if a class has
|
||||||
|
// been renamed (say, from OldClass to NewClass) this rebuild will
|
||||||
|
// cause the initial remap to see NewClass and correctly remove includes
|
||||||
|
// caused by use of OldClass.
|
||||||
|
$this->liberateGetChangedPaths($path);
|
||||||
|
|
||||||
|
$arc_bin = $this->getScriptPath('bin/arc');
|
||||||
|
|
||||||
|
do {
|
||||||
|
$future = new ExecFuture(
|
||||||
|
'%s liberate --remap -- %s',
|
||||||
|
$arc_bin,
|
||||||
|
$path);
|
||||||
|
$wrote = $future->resolveJSON();
|
||||||
|
foreach ($wrote as $wrote_path) {
|
||||||
|
echo "Updated '{$wrote_path}'...\n";
|
||||||
|
}
|
||||||
|
} while ($wrote);
|
||||||
|
|
||||||
|
echo "Verifying library...\n";
|
||||||
|
|
||||||
|
$err = 0;
|
||||||
|
$cmd = csprintf('%s liberate --verify -- %s', $arc_bin, $path);
|
||||||
|
passthru($cmd, $err);
|
||||||
|
|
||||||
|
$do_update = (!$err || $this->getArgument('force-update'));
|
||||||
|
|
||||||
|
if ($do_update) {
|
||||||
|
echo "Finalizing library map...\n";
|
||||||
|
execx('%s %s', $this->getPhutilMapperLocation(), $path);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($err) {
|
||||||
|
if ($do_update) {
|
||||||
|
echo phutil_console_format(
|
||||||
|
"<bg:yellow>** WARNING **</bg> Library update forced, but lint ".
|
||||||
|
"failures remain.\n");
|
||||||
|
} else {
|
||||||
|
echo phutil_console_format(
|
||||||
|
"<bg:red>** UNRESOLVED LINT ERRORS **</bg> This library has ".
|
||||||
|
"unresolved lint failures. The library map was not updated. Use ".
|
||||||
|
"--force-update to force an update.\n");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
echo phutil_console_format(
|
||||||
|
"<bg:green>** OKAY **</bg> Library updated.\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return $err;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function liberateLintModules($path, array $changed) {
|
||||||
|
$engine = $this->liberateBuildLintEngine($path, $changed);
|
||||||
|
if ($engine) {
|
||||||
|
return $engine->run();
|
||||||
|
} else {
|
||||||
|
return array();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function liberateWritePatches(array $results) {
|
||||||
|
$wrote = array();
|
||||||
|
|
||||||
|
foreach ($results as $result) {
|
||||||
|
if ($result->isPatchable()) {
|
||||||
|
$patcher = ArcanistLintPatcher::newFromArcanistLintResult($result);
|
||||||
|
$patcher->writePatchToDisk();
|
||||||
|
$wrote[] = $result->getPath();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $wrote;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function liberateBuildLintEngine($path, array $changed) {
|
||||||
|
$lint_map = array();
|
||||||
|
foreach ($changed as $module) {
|
||||||
|
$module_path = $path.'/'.$module;
|
||||||
|
$files = Filesystem::listDirectory($module_path);
|
||||||
|
$lint_map[$module] = $files;
|
||||||
|
}
|
||||||
|
|
||||||
|
$working_copy = ArcanistWorkingCopyIdentity::newFromRootAndConfigFile(
|
||||||
|
$path,
|
||||||
|
json_encode(
|
||||||
|
array(
|
||||||
|
'project_id' => '__arcliberate__',
|
||||||
|
)),
|
||||||
|
'arc liberate');
|
||||||
|
|
||||||
|
$engine = new ArcanistLiberateLintEngine();
|
||||||
|
$engine->setWorkingCopy($working_copy);
|
||||||
|
|
||||||
|
$lint_paths = array();
|
||||||
|
foreach ($lint_map as $module => $files) {
|
||||||
|
foreach ($files as $file) {
|
||||||
|
$lint_paths[] = $module.'/'.$file;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$lint_paths) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$engine->setPaths($lint_paths);
|
||||||
|
$engine->setMinimumSeverity(ArcanistLintSeverity::SEVERITY_ERROR);
|
||||||
|
|
||||||
|
return $engine;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private function liberateCreateDirectory($path) {
|
||||||
|
if (Filesystem::pathExists($path)) {
|
||||||
|
if (!is_dir($path)) {
|
||||||
|
throw new ArcanistUsageException(
|
||||||
|
"Provide a directory to create or update a libphutil library in.");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "The directory '{$path}' does not exist.";
|
||||||
|
if (!phutil_console_confirm('Do you want to create it?')) {
|
||||||
|
throw new ArcanistUsageException("Cancelled.");
|
||||||
|
}
|
||||||
|
|
||||||
|
execx('mkdir -p %s', $path);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function liberateCreateLibrary($path) {
|
||||||
|
$init_path = $path.'/__phutil_library_init__.php';
|
||||||
|
if (Filesystem::pathExists($init_path)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "Creating new libphutil library in '{$path}'.\n";
|
||||||
|
echo "Choose a name for the new library.\n";
|
||||||
|
do {
|
||||||
|
$name = phutil_console_prompt('What do you want to name this library?');
|
||||||
|
if (preg_match('/^[a-z]+$/', $name)) {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
echo "Library name should contain only lowercase letters.\n";
|
||||||
|
}
|
||||||
|
} while (true);
|
||||||
|
|
||||||
|
$template =
|
||||||
|
"<?php\n\n".
|
||||||
|
"phutil_register_library('{$name}', __FILE__);\n";
|
||||||
|
|
||||||
|
echo "Writing '__phutil_library_init__.php' to '{$init_path}'...\n";
|
||||||
|
Filesystem::writeFile($init_path, $template);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function liberateGetChangedPaths($path) {
|
||||||
|
$mapper = $this->getPhutilMapperLocation();
|
||||||
|
$future = new ExecFuture('%s %s --find-paths-for-liberate', $mapper, $path);
|
||||||
|
return $future->resolveJSON();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getScriptPath($script) {
|
||||||
|
$root = dirname(phutil_get_library_root('arcanist'));
|
||||||
|
return $root.'/'.$script;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getPhutilMapperLocation() {
|
||||||
|
return $this->getScriptPath('scripts/phutil_mapper.php');
|
||||||
|
}
|
||||||
|
|
||||||
|
private function liberateRunRemap($path) {
|
||||||
|
phutil_load_library($path);
|
||||||
|
|
||||||
|
$paths = $this->liberateGetChangedPaths($path);
|
||||||
|
$results = $this->liberateLintModules($path, $paths);
|
||||||
|
$wrote = $this->liberateWritePatches($results);
|
||||||
|
|
||||||
|
echo json_encode($wrote, true);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function liberateRunVerify($path) {
|
||||||
|
phutil_load_library($path);
|
||||||
|
|
||||||
|
$paths = $this->liberateGetChangedPaths($path);
|
||||||
|
$results = $this->liberateLintModules($path, $paths);
|
||||||
|
|
||||||
|
$renderer = new ArcanistLintRenderer();
|
||||||
|
|
||||||
|
$unresolved = false;
|
||||||
|
foreach ($results as $result) {
|
||||||
|
foreach ($result->getMessages() as $message) {
|
||||||
|
echo $renderer->renderLintResult($result);
|
||||||
|
$unresolved = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (int)$unresolved;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
26
src/workflow/liberate/__init__.php
Normal file
26
src/workflow/liberate/__init__.php
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* This file is automatically generated. Lint this module to rebuild it.
|
||||||
|
* @generated
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
phutil_require_module('arcanist', 'exception/usage');
|
||||||
|
phutil_require_module('arcanist', 'lint/engine/liberate');
|
||||||
|
phutil_require_module('arcanist', 'lint/patcher');
|
||||||
|
phutil_require_module('arcanist', 'lint/renderer');
|
||||||
|
phutil_require_module('arcanist', 'lint/severity');
|
||||||
|
phutil_require_module('arcanist', 'workflow/base');
|
||||||
|
phutil_require_module('arcanist', 'workingcopyidentity');
|
||||||
|
|
||||||
|
phutil_require_module('phutil', 'console');
|
||||||
|
phutil_require_module('phutil', 'filesystem');
|
||||||
|
phutil_require_module('phutil', 'filesystem/filefinder');
|
||||||
|
phutil_require_module('phutil', 'future/exec');
|
||||||
|
phutil_require_module('phutil', 'moduleutils');
|
||||||
|
phutil_require_module('phutil', 'utils');
|
||||||
|
phutil_require_module('phutil', 'xsprintf/csprintf');
|
||||||
|
|
||||||
|
|
||||||
|
phutil_require_source('ArcanistLiberateWorkflow.php');
|
Loading…
Reference in a new issue