mirror of
https://we.phorge.it/source/arcanist.git
synced 2025-01-10 14:51:05 +01:00
266a4f4c75
Test Plan: Threw in `didRunLinters()` of one linter, still saw the result of other linters and "Some linters failed" at the end. Reviewers: epriestley Reviewed By: epriestley CC: aran, Korvin Differential Revision: https://secure.phabricator.com/D5124
491 lines
14 KiB
PHP
491 lines
14 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Manages lint execution. When you run 'arc lint' or 'arc diff', Arcanist
|
|
* checks your .arcconfig to see if you have specified a lint engine in the
|
|
* key "lint.engine". The engine must extend this class. For example:
|
|
*
|
|
* lang=js
|
|
* {
|
|
* // ...
|
|
* "lint.engine" : "ExampleLintEngine",
|
|
* // ...
|
|
* }
|
|
*
|
|
* The lint engine is given a list of paths (generally, the paths that you
|
|
* modified in your change) and determines which linters to run on them. The
|
|
* linters themselves are responsible for actually analyzing file text and
|
|
* finding warnings and errors. For example, if the modified paths include some
|
|
* JS files and some Python files, you might want to run JSLint on the JS files
|
|
* and PyLint on the Python files.
|
|
*
|
|
* You can also run multiple linters on a single file. For instance, you might
|
|
* run one linter on all text files to make sure they don't have trailing
|
|
* whitespace, or enforce tab vs space rules, or make sure there are enough
|
|
* curse words in them.
|
|
*
|
|
* Because lint engines are pretty custom to the rules of a project, you will
|
|
* generally need to build your own. Fortunately, it's pretty easy (and you
|
|
* can use the prebuilt //linters//, you just need to write a little glue code
|
|
* to tell Arcanist which linters to run). For a simple example of how to build
|
|
* a lint engine, see @{class:ExampleLintEngine}.
|
|
*
|
|
* You can test an engine like this:
|
|
*
|
|
* arc lint --engine ExampleLintEngine --lintall some_file.py
|
|
*
|
|
* ...which will show you all the lint issues raised in the file.
|
|
*
|
|
* See @{article@phabricator:Arcanist User Guide: Customizing Lint, Unit Tests
|
|
* and Workflows} for more information about configuring lint engines.
|
|
*
|
|
* @group lint
|
|
* @stable
|
|
*/
|
|
abstract class ArcanistLintEngine {
|
|
|
|
protected $workingCopy;
|
|
protected $paths = array();
|
|
protected $fileData = array();
|
|
|
|
protected $charToLine = array();
|
|
protected $lineToFirstChar = array();
|
|
private $cachedResults;
|
|
private $cacheVersion;
|
|
private $repositoryVersion;
|
|
private $results = array();
|
|
private $stopped = array();
|
|
private $minimumSeverity = ArcanistLintSeverity::SEVERITY_DISABLED;
|
|
|
|
private $changedLines = array();
|
|
private $commitHookMode = false;
|
|
private $hookAPI;
|
|
|
|
private $enableAsyncLint = false;
|
|
private $postponedLinters = array();
|
|
|
|
public function __construct() {
|
|
|
|
}
|
|
|
|
public function setWorkingCopy(ArcanistWorkingCopyIdentity $working_copy) {
|
|
$this->workingCopy = $working_copy;
|
|
return $this;
|
|
}
|
|
|
|
public function getWorkingCopy() {
|
|
return $this->workingCopy;
|
|
}
|
|
|
|
public function setPaths($paths) {
|
|
$this->paths = $paths;
|
|
return $this;
|
|
}
|
|
|
|
public function getPaths() {
|
|
return $this->paths;
|
|
}
|
|
|
|
public function setPathChangedLines($path, $changed) {
|
|
if ($changed === null) {
|
|
$this->changedLines[$path] = null;
|
|
} else {
|
|
$this->changedLines[$path] = array_fill_keys($changed, true);
|
|
}
|
|
return $this;
|
|
}
|
|
|
|
public function getPathChangedLines($path) {
|
|
return idx($this->changedLines, $path);
|
|
}
|
|
|
|
public function setFileData($data) {
|
|
$this->fileData = $data + $this->fileData;
|
|
return $this;
|
|
}
|
|
|
|
public function setCommitHookMode($mode) {
|
|
$this->commitHookMode = $mode;
|
|
return $this;
|
|
}
|
|
|
|
public function setHookAPI(ArcanistHookAPI $hook_api) {
|
|
$this->hookAPI = $hook_api;
|
|
return $this;
|
|
}
|
|
|
|
public function getHookAPI() {
|
|
return $this->hookAPI;
|
|
}
|
|
|
|
public function setEnableAsyncLint($enable_async_lint) {
|
|
$this->enableAsyncLint = $enable_async_lint;
|
|
return $this;
|
|
}
|
|
|
|
public function getEnableAsyncLint() {
|
|
return $this->enableAsyncLint;
|
|
}
|
|
|
|
public function loadData($path) {
|
|
if (!isset($this->fileData[$path])) {
|
|
if ($this->getCommitHookMode()) {
|
|
$this->fileData[$path] = $this->getHookAPI()
|
|
->getCurrentFileData($path);
|
|
} else {
|
|
$disk_path = $this->getFilePathOnDisk($path);
|
|
$this->fileData[$path] = Filesystem::readFile($disk_path);
|
|
}
|
|
}
|
|
return $this->fileData[$path];
|
|
}
|
|
|
|
public function pathExists($path) {
|
|
if ($this->getCommitHookMode()) {
|
|
$file_data = $this->loadData($path);
|
|
return ($file_data !== null);
|
|
} else {
|
|
$disk_path = $this->getFilePathOnDisk($path);
|
|
return Filesystem::pathExists($disk_path);
|
|
}
|
|
}
|
|
|
|
public function getFilePathOnDisk($path) {
|
|
return Filesystem::resolvePath(
|
|
$path,
|
|
$this->getWorkingCopy()->getProjectRoot());
|
|
}
|
|
|
|
public function setMinimumSeverity($severity) {
|
|
$this->minimumSeverity = $severity;
|
|
return $this;
|
|
}
|
|
|
|
public function getCommitHookMode() {
|
|
return $this->commitHookMode;
|
|
}
|
|
|
|
public function run() {
|
|
$linters = $this->buildLinters();
|
|
|
|
if (!$linters) {
|
|
throw new ArcanistNoEffectException("No linters to run.");
|
|
}
|
|
|
|
$have_paths = false;
|
|
foreach ($linters as $linter) {
|
|
if ($linter->getPaths()) {
|
|
$have_paths = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!$have_paths) {
|
|
throw new ArcanistNoEffectException("No paths are lintable.");
|
|
}
|
|
|
|
$versions = array($this->getCacheVersion());
|
|
foreach ($linters as $linter) {
|
|
$linter->setEngine($this);
|
|
$versions[] = get_class($linter).':'.$linter->getCacheVersion();
|
|
}
|
|
$this->cacheVersion = crc32(implode("\n", $versions));
|
|
|
|
$this->stopped = array();
|
|
$exceptions = array();
|
|
foreach ($linters as $linter_name => $linter) {
|
|
if (!is_string($linter_name)) {
|
|
$linter_name = get_class($linter);
|
|
}
|
|
try {
|
|
if (!$linter->canRun()) {
|
|
continue;
|
|
}
|
|
$paths = $linter->getPaths();
|
|
|
|
foreach ($paths as $key => $path) {
|
|
// Make sure each path has a result generated, even if it is empty
|
|
// (i.e., the file has no lint messages).
|
|
$result = $this->getResultForPath($path);
|
|
if (isset($this->stopped[$path])) {
|
|
unset($paths[$key]);
|
|
}
|
|
if (isset($this->cachedResults[$path][$this->cacheVersion])) {
|
|
$cached_result = $this->cachedResults[$path][$this->cacheVersion];
|
|
|
|
$use_cache = $this->shouldUseCache(
|
|
$linter->getCacheGranularity(),
|
|
idx($cached_result, 'repository_version'));
|
|
|
|
if ($use_cache) {
|
|
unset($paths[$key]);
|
|
|
|
if (idx($cached_result, 'stopped') == $linter_name) {
|
|
$this->stopped[$path] = $linter_name;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
$paths = array_values($paths);
|
|
|
|
if ($paths) {
|
|
$profiler = PhutilServiceProfiler::getInstance();
|
|
$call_id = $profiler->beginServiceCall(array(
|
|
'type' => 'lint',
|
|
'linter' => $linter_name,
|
|
'paths' => $paths,
|
|
));
|
|
|
|
try {
|
|
$linter->willLintPaths($paths);
|
|
foreach ($paths as $path) {
|
|
$linter->willLintPath($path);
|
|
$linter->lintPath($path);
|
|
if ($linter->didStopAllLinters()) {
|
|
$this->stopped[$path] = $linter_name;
|
|
}
|
|
}
|
|
} catch (Exception $ex) {
|
|
$profiler->endServiceCall($call_id, array());
|
|
throw $ex;
|
|
}
|
|
$profiler->endServiceCall($call_id, array());
|
|
}
|
|
|
|
} catch (Exception $ex) {
|
|
$exceptions[$linter_name] = $ex;
|
|
}
|
|
}
|
|
|
|
$exceptions += $this->didRunLinters($linters);
|
|
|
|
foreach ($linters as $linter) {
|
|
foreach ($linter->getLintMessages() as $message) {
|
|
if (!$this->isSeverityEnabled($message->getSeverity())) {
|
|
continue;
|
|
}
|
|
if (!$this->isRelevantMessage($message)) {
|
|
continue;
|
|
}
|
|
$message->setGranularity($linter->getCacheGranularity());
|
|
$result = $this->getResultForPath($message->getPath());
|
|
$result->addMessage($message);
|
|
}
|
|
}
|
|
|
|
if ($this->cachedResults) {
|
|
foreach ($this->cachedResults as $path => $messages) {
|
|
$messages = idx($messages, $this->cacheVersion, array());
|
|
$repository_version = idx($messages, 'repository_version');
|
|
unset($messages['stopped']);
|
|
unset($messages['repository_version']);
|
|
foreach ($messages as $message) {
|
|
$use_cache = $this->shouldUseCache(
|
|
idx($message, 'granularity'),
|
|
$repository_version);
|
|
if ($use_cache) {
|
|
$this->getResultForPath($path)->addMessage(
|
|
ArcanistLintMessage::newFromDictionary($message));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
foreach ($this->results as $path => $result) {
|
|
$disk_path = $this->getFilePathOnDisk($path);
|
|
$result->setFilePathOnDisk($disk_path);
|
|
if (isset($this->fileData[$path])) {
|
|
$result->setData($this->fileData[$path]);
|
|
} else if ($disk_path && Filesystem::pathExists($disk_path)) {
|
|
// TODO: this may cause us to, e.g., load a large binary when we only
|
|
// raised an error about its filename. We could refine this by looking
|
|
// through the lint messages and doing this load only if any of them
|
|
// have original/replacement text or something like that.
|
|
try {
|
|
$this->fileData[$path] = Filesystem::readFile($disk_path);
|
|
$result->setData($this->fileData[$path]);
|
|
} catch (FilesystemException $ex) {
|
|
// Ignore this, it's noncritical that we access this data and it
|
|
// might be unreadable or a directory or whatever else for plenty
|
|
// of legitimate reasons.
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($exceptions) {
|
|
throw new PhutilAggregateException('Some linters failed:', $exceptions);
|
|
}
|
|
|
|
return $this->results;
|
|
}
|
|
|
|
public function isSeverityEnabled($severity) {
|
|
$minimum = $this->minimumSeverity;
|
|
return ArcanistLintSeverity::isAtLeastAsSevere($severity, $minimum);
|
|
}
|
|
|
|
private function shouldUseCache($cache_granularity, $repository_version) {
|
|
switch ($cache_granularity) {
|
|
case ArcanistLinter::GRANULARITY_FILE:
|
|
return true;
|
|
case ArcanistLinter::GRANULARITY_DIRECTORY:
|
|
case ArcanistLinter::GRANULARITY_REPOSITORY:
|
|
return ($this->repositoryVersion == $repository_version);
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param dict<string path, dict<string version, list<dict message>>>
|
|
* @return this
|
|
*/
|
|
public function setCachedResults(array $results) {
|
|
$this->cachedResults = $results;
|
|
return $this;
|
|
}
|
|
|
|
public function getResults() {
|
|
return $this->results;
|
|
}
|
|
|
|
public function getStoppedPaths() {
|
|
return $this->stopped;
|
|
}
|
|
|
|
abstract protected function buildLinters();
|
|
|
|
protected function didRunLinters(array $linters) {
|
|
assert_instances_of($linters, 'ArcanistLinter');
|
|
|
|
$exceptions = array();
|
|
$profiler = PhutilServiceProfiler::getInstance();
|
|
|
|
foreach ($linters as $linter_name => $linter) {
|
|
if (!is_string($linter_name)) {
|
|
$linter_name = get_class($linter);
|
|
}
|
|
|
|
$call_id = $profiler->beginServiceCall(array(
|
|
'type' => 'lint',
|
|
'linter' => $linter_name,
|
|
));
|
|
|
|
try {
|
|
$linter->didRunLinters();
|
|
} catch (Exception $ex) {
|
|
$exceptions[$linter_name] = $ex;
|
|
}
|
|
$profiler->endServiceCall($call_id, array());
|
|
}
|
|
|
|
return $exceptions;
|
|
}
|
|
|
|
public function setRepositoryVersion($version) {
|
|
$this->repositoryVersion = $version;
|
|
return $this;
|
|
}
|
|
|
|
private function isRelevantMessage(ArcanistLintMessage $message) {
|
|
// When a user runs "arc lint", we default to raising only warnings on
|
|
// lines they have changed (errors are still raised anywhere in the
|
|
// file). The list of $changed lines may be null, to indicate that the
|
|
// path is a directory or a binary file so we should not exclude
|
|
// warnings.
|
|
|
|
if (!$this->changedLines || $message->isError()) {
|
|
return true;
|
|
}
|
|
|
|
$locations = $message->getOtherLocations();
|
|
$locations[] = $message->toDictionary();
|
|
|
|
foreach ($locations as $location) {
|
|
$path = idx($location, 'path', $message->getPath());
|
|
|
|
if (!array_key_exists($path, $this->changedLines)) {
|
|
continue;
|
|
}
|
|
|
|
$changed = $this->getPathChangedLines($path);
|
|
|
|
if ($changed === null || !$location['line']) {
|
|
return true;
|
|
}
|
|
|
|
$last_line = $location['line'];
|
|
if (isset($location['original'])) {
|
|
$last_line += substr_count($location['original'], "\n");
|
|
}
|
|
|
|
for ($l = $location['line']; $l <= $last_line; $l++) {
|
|
if (!empty($changed[$l])) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
protected function getResultForPath($path) {
|
|
if (empty($this->results[$path])) {
|
|
$result = new ArcanistLintResult();
|
|
$result->setPath($path);
|
|
$result->setCacheVersion($this->cacheVersion);
|
|
$this->results[$path] = $result;
|
|
}
|
|
return $this->results[$path];
|
|
}
|
|
|
|
public function getLineAndCharFromOffset($path, $offset) {
|
|
if (!isset($this->charToLine[$path])) {
|
|
$char_to_line = array();
|
|
$line_to_first_char = array();
|
|
|
|
$lines = explode("\n", $this->loadData($path));
|
|
$line_number = 0;
|
|
$line_start = 0;
|
|
foreach ($lines as $line) {
|
|
$len = strlen($line) + 1; // Account for "\n".
|
|
$line_to_first_char[] = $line_start;
|
|
$line_start += $len;
|
|
for ($ii = 0; $ii < $len; $ii++) {
|
|
$char_to_line[] = $line_number;
|
|
}
|
|
$line_number++;
|
|
}
|
|
$this->charToLine[$path] = $char_to_line;
|
|
$this->lineToFirstChar[$path] = $line_to_first_char;
|
|
}
|
|
|
|
$line = $this->charToLine[$path][$offset];
|
|
$char = $offset - $this->lineToFirstChar[$path][$line];
|
|
|
|
return array($line, $char);
|
|
}
|
|
|
|
public function getPostponedLinters() {
|
|
return $this->postponedLinters;
|
|
}
|
|
|
|
public function setPostponedLinters(array $linters) {
|
|
$this->postponedLinters = $linters;
|
|
return $this;
|
|
}
|
|
|
|
protected function getCacheVersion() {
|
|
return 1;
|
|
}
|
|
|
|
protected function getPEP8WithTextOptions() {
|
|
// E101 is subset of TXT2 (Tab Literal).
|
|
// E501 is same as TXT3 (Line Too Long).
|
|
// W291 is same as TXT6 (Trailing Whitespace).
|
|
// W292 is same as TXT4 (File Does Not End in Newline).
|
|
// W293 is same as TXT6 (Trailing Whitespace).
|
|
return '--ignore=E101,E501,W291,W292,W293';
|
|
}
|
|
|
|
}
|