2011-01-09 15:22:25 -08:00
|
|
|
<?php
|
|
|
|
|
2011-02-19 11:36:08 -08:00
|
|
|
/**
|
|
|
|
* Implements lint rules, like syntax checks for a specific language.
|
|
|
|
*
|
2014-05-11 13:42:56 -07:00
|
|
|
* @task info Human Readable Information
|
Split large path lists into blocks when linting
Summary:
Fixes T5097. When linting a large list of paths (e.g., with `--everything`), do this internally:
$chunks = array_chunk($paths, 32);
foreach ($chunks as $chunk) {
$this->lintChunk($chunk);
}
This keeps the advantages of parallelism and artifact sharing for future-based linters, without having memory usage grow in an unbounded way.
These callbacks changed:
- `willLintPath()`: Useless, no meaningful implementations. Internalized the required side effect and broke the hook.
- `didRunLinters()`: Now useless, with no meaningful implementations. Broke the hook.
- `didLintPaths()`: New hook which executes opposite `willLintPaths()`.
- `lintPath()`: Linters no longer need to implement this method.
XHPAST now has an explicit way to release shared futures.
These minor changes also happened:
- Formalized the "linter ID", which is a semi-durable identifier for the cache.
- Removed linter -> exception explicit mapping, which was unused. We now just collect exceptions.
- We do the `canRun()` checks first (and separately) now.
- Share more service call profiling code.
- Fix an issue where the test harness would use the path on disk, even if configuration set a different path.
Test Plan:
- Ran `arc lint --everything` in `arcanist/`.
- With no chunking, saw **unstable** memory usage with a peak at 941 MB.
- With chunk size 32, saw **stable** memory usage with a peak at 269 MB.
- With chunk size 8, saw **stable** memory usage with a peak at 180 MB.
- Ran with `--trace` and saw profiling information.
- Created this diff.
Reviewers: joshuaspence
Reviewed By: joshuaspence
Subscribers: epriestley
Maniphest Tasks: T5097
Differential Revision: https://secure.phabricator.com/D12501
2015-04-22 05:15:57 -07:00
|
|
|
* @task state Runtime State
|
|
|
|
* @task exec Executing Linters
|
2012-01-31 12:07:05 -08:00
|
|
|
* @stable
|
2011-02-19 11:36:08 -08:00
|
|
|
*/
|
2011-01-09 15:22:25 -08:00
|
|
|
abstract class ArcanistLinter {
|
|
|
|
|
2012-11-21 18:38:24 -08:00
|
|
|
const GRANULARITY_FILE = 1;
|
|
|
|
const GRANULARITY_DIRECTORY = 2;
|
|
|
|
const GRANULARITY_REPOSITORY = 3;
|
|
|
|
const GRANULARITY_GLOBAL = 4;
|
|
|
|
|
Split large path lists into blocks when linting
Summary:
Fixes T5097. When linting a large list of paths (e.g., with `--everything`), do this internally:
$chunks = array_chunk($paths, 32);
foreach ($chunks as $chunk) {
$this->lintChunk($chunk);
}
This keeps the advantages of parallelism and artifact sharing for future-based linters, without having memory usage grow in an unbounded way.
These callbacks changed:
- `willLintPath()`: Useless, no meaningful implementations. Internalized the required side effect and broke the hook.
- `didRunLinters()`: Now useless, with no meaningful implementations. Broke the hook.
- `didLintPaths()`: New hook which executes opposite `willLintPaths()`.
- `lintPath()`: Linters no longer need to implement this method.
XHPAST now has an explicit way to release shared futures.
These minor changes also happened:
- Formalized the "linter ID", which is a semi-durable identifier for the cache.
- Removed linter -> exception explicit mapping, which was unused. We now just collect exceptions.
- We do the `canRun()` checks first (and separately) now.
- Share more service call profiling code.
- Fix an issue where the test harness would use the path on disk, even if configuration set a different path.
Test Plan:
- Ran `arc lint --everything` in `arcanist/`.
- With no chunking, saw **unstable** memory usage with a peak at 941 MB.
- With chunk size 32, saw **stable** memory usage with a peak at 269 MB.
- With chunk size 8, saw **stable** memory usage with a peak at 180 MB.
- Ran with `--trace` and saw profiling information.
- Created this diff.
Reviewers: joshuaspence
Reviewed By: joshuaspence
Subscribers: epriestley
Maniphest Tasks: T5097
Differential Revision: https://secure.phabricator.com/D12501
2015-04-22 05:15:57 -07:00
|
|
|
private $id;
|
2015-04-27 23:20:31 +10:00
|
|
|
protected $paths = array();
|
|
|
|
private $filteredPaths = null;
|
|
|
|
protected $data = array();
|
2011-01-09 15:22:25 -08:00
|
|
|
protected $engine;
|
|
|
|
protected $activePath;
|
|
|
|
protected $messages = array();
|
|
|
|
|
|
|
|
protected $stopAllLinters = false;
|
|
|
|
|
|
|
|
private $customSeverityMap = array();
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
private $customSeverityRules = array();
|
2011-01-09 15:22:25 -08:00
|
|
|
|
2014-05-11 13:42:56 -07:00
|
|
|
|
|
|
|
/* -( Human Readable Information )---------------------------------------- */
|
|
|
|
|
Split large path lists into blocks when linting
Summary:
Fixes T5097. When linting a large list of paths (e.g., with `--everything`), do this internally:
$chunks = array_chunk($paths, 32);
foreach ($chunks as $chunk) {
$this->lintChunk($chunk);
}
This keeps the advantages of parallelism and artifact sharing for future-based linters, without having memory usage grow in an unbounded way.
These callbacks changed:
- `willLintPath()`: Useless, no meaningful implementations. Internalized the required side effect and broke the hook.
- `didRunLinters()`: Now useless, with no meaningful implementations. Broke the hook.
- `didLintPaths()`: New hook which executes opposite `willLintPaths()`.
- `lintPath()`: Linters no longer need to implement this method.
XHPAST now has an explicit way to release shared futures.
These minor changes also happened:
- Formalized the "linter ID", which is a semi-durable identifier for the cache.
- Removed linter -> exception explicit mapping, which was unused. We now just collect exceptions.
- We do the `canRun()` checks first (and separately) now.
- Share more service call profiling code.
- Fix an issue where the test harness would use the path on disk, even if configuration set a different path.
Test Plan:
- Ran `arc lint --everything` in `arcanist/`.
- With no chunking, saw **unstable** memory usage with a peak at 941 MB.
- With chunk size 32, saw **stable** memory usage with a peak at 269 MB.
- With chunk size 8, saw **stable** memory usage with a peak at 180 MB.
- Ran with `--trace` and saw profiling information.
- Created this diff.
Reviewers: joshuaspence
Reviewed By: joshuaspence
Subscribers: epriestley
Maniphest Tasks: T5097
Differential Revision: https://secure.phabricator.com/D12501
2015-04-22 05:15:57 -07:00
|
|
|
|
2014-05-11 13:42:56 -07:00
|
|
|
/**
|
|
|
|
* Return an optional informative URI where humans can learn more about this
|
|
|
|
* linter.
|
|
|
|
*
|
|
|
|
* For most linters, this should return a link to the project home page. This
|
|
|
|
* is shown on `arc linters`.
|
|
|
|
*
|
|
|
|
* @return string|null Optionally, return an informative URI.
|
|
|
|
* @task info
|
|
|
|
*/
|
|
|
|
public function getInfoURI() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a brief human-readable description of the linter.
|
|
|
|
*
|
|
|
|
* These should be a line or two, and are shown on `arc linters`.
|
|
|
|
*
|
|
|
|
* @return string|null Optionally, return a brief human-readable description.
|
|
|
|
* @task info
|
|
|
|
*/
|
|
|
|
public function getInfoDescription() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a human-readable linter name.
|
|
|
|
*
|
|
|
|
* These are used by `arc linters`, and can let you give a linter a more
|
|
|
|
* presentable name.
|
|
|
|
*
|
|
|
|
* @return string Human-readable linter name.
|
|
|
|
* @task info
|
|
|
|
*/
|
|
|
|
public function getInfoName() {
|
|
|
|
return nonempty(
|
|
|
|
$this->getLinterName(),
|
|
|
|
$this->getLinterConfigurationName(),
|
|
|
|
get_class($this));
|
|
|
|
}
|
|
|
|
|
Split large path lists into blocks when linting
Summary:
Fixes T5097. When linting a large list of paths (e.g., with `--everything`), do this internally:
$chunks = array_chunk($paths, 32);
foreach ($chunks as $chunk) {
$this->lintChunk($chunk);
}
This keeps the advantages of parallelism and artifact sharing for future-based linters, without having memory usage grow in an unbounded way.
These callbacks changed:
- `willLintPath()`: Useless, no meaningful implementations. Internalized the required side effect and broke the hook.
- `didRunLinters()`: Now useless, with no meaningful implementations. Broke the hook.
- `didLintPaths()`: New hook which executes opposite `willLintPaths()`.
- `lintPath()`: Linters no longer need to implement this method.
XHPAST now has an explicit way to release shared futures.
These minor changes also happened:
- Formalized the "linter ID", which is a semi-durable identifier for the cache.
- Removed linter -> exception explicit mapping, which was unused. We now just collect exceptions.
- We do the `canRun()` checks first (and separately) now.
- Share more service call profiling code.
- Fix an issue where the test harness would use the path on disk, even if configuration set a different path.
Test Plan:
- Ran `arc lint --everything` in `arcanist/`.
- With no chunking, saw **unstable** memory usage with a peak at 941 MB.
- With chunk size 32, saw **stable** memory usage with a peak at 269 MB.
- With chunk size 8, saw **stable** memory usage with a peak at 180 MB.
- Ran with `--trace` and saw profiling information.
- Created this diff.
Reviewers: joshuaspence
Reviewed By: joshuaspence
Subscribers: epriestley
Maniphest Tasks: T5097
Differential Revision: https://secure.phabricator.com/D12501
2015-04-22 05:15:57 -07:00
|
|
|
|
|
|
|
/* -( Runtime State )------------------------------------------------------ */
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @task state
|
|
|
|
*/
|
|
|
|
final public function getActivePath() {
|
|
|
|
return $this->activePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @task state
|
|
|
|
*/
|
|
|
|
final public function setActivePath($path) {
|
|
|
|
$this->stopAllLinters = false;
|
|
|
|
$this->activePath = $path;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @task state
|
|
|
|
*/
|
|
|
|
final public function setEngine(ArcanistLintEngine $engine) {
|
|
|
|
$this->engine = $engine;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @task state
|
|
|
|
*/
|
|
|
|
final protected function getEngine() {
|
|
|
|
return $this->engine;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the internal ID for this linter.
|
|
|
|
*
|
|
|
|
* This ID is assigned automatically by the @{class:ArcanistLintEngine}.
|
|
|
|
*
|
|
|
|
* @param string Unique linter ID.
|
|
|
|
* @return this
|
|
|
|
* @task state
|
|
|
|
*/
|
|
|
|
final public function setLinterID($id) {
|
|
|
|
$this->id = $id;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the internal ID for this linter.
|
|
|
|
*
|
|
|
|
* Retrieves an internal linter ID managed by the @{class:ArcanistLintEngine}.
|
|
|
|
* This ID is a unique scalar which distinguishes linters in a list.
|
|
|
|
*
|
|
|
|
* @return string Unique linter ID.
|
|
|
|
* @task state
|
|
|
|
*/
|
|
|
|
final public function getLinterID() {
|
|
|
|
return $this->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* -( Executing Linters )-------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hook called before a list of paths are linted.
|
|
|
|
*
|
|
|
|
* Parallelizable linters can start multiple requests in parallel here,
|
|
|
|
* to improve performance. They can implement @{method:didLintPaths} to
|
|
|
|
* collect results.
|
|
|
|
*
|
|
|
|
* Linters which are not parallelizable should normally ignore this callback
|
|
|
|
* and implement @{method:lintPath} instead.
|
|
|
|
*
|
|
|
|
* @param list<string> A list of paths to be linted
|
|
|
|
* @return void
|
|
|
|
* @task exec
|
|
|
|
*/
|
|
|
|
public function willLintPaths(array $paths) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hook called for each path to be linted.
|
|
|
|
*
|
|
|
|
* Linters which are not parallelizable can do work here.
|
|
|
|
*
|
|
|
|
* Linters which are parallelizable may want to ignore this callback and
|
|
|
|
* implement @{method:willLintPaths} and @{method:didLintPaths} instead.
|
|
|
|
*
|
|
|
|
* @param string Path to lint.
|
|
|
|
* @return void
|
|
|
|
* @task exec
|
|
|
|
*/
|
|
|
|
public function lintPath($path) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hook called after a list of paths are linted.
|
|
|
|
*
|
|
|
|
* Parallelizable linters can collect results here.
|
|
|
|
*
|
|
|
|
* Linters which are not paralleizable should normally ignore this callback
|
|
|
|
* and implement @{method:lintPath} instead.
|
|
|
|
*
|
|
|
|
* @param list<string> A list of paths which were linted.
|
|
|
|
* @return void
|
|
|
|
* @task exec
|
|
|
|
*/
|
|
|
|
public function didLintPaths(array $paths) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Obsolete hook which was invoked before a path was linted.
|
|
|
|
*
|
|
|
|
* WARNING: This is an obsolete hook which is not called. If you maintain
|
|
|
|
* a linter which relies on it, update to use @{method:lintPath} instead.
|
|
|
|
*
|
|
|
|
* @task exec
|
|
|
|
*/
|
|
|
|
final public function willLintPath($path) {
|
|
|
|
// TODO: Remove this method after some time. In the meantime, the "final"
|
|
|
|
// will fatal subclasses which implement this hook and point at the API
|
|
|
|
// change so maintainers get fewer surprises.
|
|
|
|
throw new PhutilMethodNotImplementedException();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Obsolete hook which was invoked after linters ran.
|
|
|
|
*
|
|
|
|
* WARNING: This is an obsolete hook which is not called. If you maintain
|
|
|
|
* a linter which relies on it, update to use @{method:didLintPaths} instead.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
* @task exec
|
|
|
|
*/
|
|
|
|
final public function didRunLinters() {
|
|
|
|
// TODO: Remove this method after some time. In the meantime, the "final"
|
|
|
|
// will fatal subclasses which implement this hook and point at the API
|
|
|
|
// change so maintainers get fewer surprises.
|
|
|
|
throw new PhutilMethodNotImplementedException();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
public function getLinterPriority() {
|
|
|
|
return 1.0;
|
|
|
|
}
|
|
|
|
|
2014-10-08 00:07:39 +11:00
|
|
|
/**
|
|
|
|
* TODO: This should be `final`.
|
|
|
|
*/
|
2011-01-09 15:22:25 -08:00
|
|
|
public function setCustomSeverityMap(array $map) {
|
|
|
|
$this->customSeverityMap = $map;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-10-08 00:07:39 +11:00
|
|
|
final public function setCustomSeverityRules(array $rules) {
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
$this->customSeverityRules = $rules;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function getOtherLocation($offset, $path = null) {
|
2013-02-08 15:04:32 -08:00
|
|
|
if ($path === null) {
|
|
|
|
$path = $this->getActivePath();
|
|
|
|
}
|
|
|
|
|
|
|
|
list($line, $char) = $this->getEngine()->getLineAndCharFromOffset(
|
|
|
|
$path,
|
|
|
|
$offset);
|
|
|
|
|
|
|
|
return array(
|
|
|
|
'path' => $path,
|
|
|
|
'line' => $line + 1,
|
|
|
|
'char' => $char,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function stopAllLinters() {
|
2011-01-09 15:22:25 -08:00
|
|
|
$this->stopAllLinters = true;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function didStopAllLinters() {
|
2011-01-09 15:22:25 -08:00
|
|
|
return $this->stopAllLinters;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function addPath($path) {
|
2011-01-09 15:22:25 -08:00
|
|
|
$this->paths[$path] = $path;
|
2015-04-27 23:20:31 +10:00
|
|
|
$this->filteredPaths = null;
|
2011-01-09 15:22:25 -08:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function setPaths(array $paths) {
|
2012-10-20 05:45:22 -07:00
|
|
|
$this->paths = $paths;
|
2015-04-27 23:20:31 +10:00
|
|
|
$this->filteredPaths = null;
|
2012-10-20 05:45:22 -07:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
/**
|
|
|
|
* Filter out paths which this linter doesn't act on (for example, because
|
|
|
|
* they are binaries and the linter doesn't apply to binaries).
|
2015-04-27 23:20:31 +10:00
|
|
|
*
|
|
|
|
* @param list<string>
|
|
|
|
* @return list<string>
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
*/
|
2015-04-27 23:20:31 +10:00
|
|
|
final private function filterPaths(array $paths) {
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
$engine = $this->getEngine();
|
|
|
|
|
|
|
|
$keep = array();
|
|
|
|
foreach ($paths as $path) {
|
|
|
|
if (!$this->shouldLintDeletedFiles() && !$engine->pathExists($path)) {
|
|
|
|
continue;
|
|
|
|
}
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
|
|
|
|
if (!$this->shouldLintDirectories() && $engine->isDirectory($path)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$this->shouldLintBinaryFiles() && $engine->isBinaryFile($path)) {
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
continue;
|
|
|
|
}
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
|
2014-06-10 18:20:59 -07:00
|
|
|
if (!$this->shouldLintSymbolicLinks() && $engine->isSymbolicLink($path)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
$keep[] = $path;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $keep;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function getPaths() {
|
2015-04-27 23:20:31 +10:00
|
|
|
if ($this->filteredPaths === null) {
|
|
|
|
$this->filteredPaths = $this->filterPaths(array_values($this->paths));
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->filteredPaths;
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function addData($path, $data) {
|
2011-01-09 15:22:25 -08:00
|
|
|
$this->data[$path] = $data;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final protected function getData($path) {
|
2011-01-09 15:22:25 -08:00
|
|
|
if (!array_key_exists($path, $this->data)) {
|
[arc svn-hook-pre-commit] Access working copy
Summary:
Creates a new hook API that can be used to interface with
SVN/Git/Mercurial in the context of a commit hook. Currently only adds a
function to read the modified file data in a Subversion commit hook.
An object of this API is created in the SvnHookPreCommitWorkflow and
passed on the Lint Engine which then uses it to access current file
data, of the way the APIs seem to be structured); linters use the
getData function which is essentially a wrapper around the engine's
call, with another layer of caching.
Task ID: #770556
Blame Rev:
Test Plan:
- Create a local svn repository and add a minimal hook to run the local
version of arc to test commits
(http://phabricator.com/docs/arcanist/article/Installing_Arcanist_SVN_Hooks.html)
- Create a temporary repository that can trigger any of the linters
available, and test against a temporary linter by committing against
the test repository: the linter should be able to access all required
files by using loadData/getData in the LintEngine and Linter.
Revert Plan:
Tags: lint, svn-hook-pre-commit
Reviewers: jungejason, asukhachev, epriestley, aran
Reviewed By: epriestley
CC: aran, jungejason, epriestley, kunalb, asukhachev
Differential Revision: https://secure.phabricator.com/D1256
2011-12-20 20:26:05 -08:00
|
|
|
$this->data[$path] = $this->getEngine()->loadData($path);
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|
|
|
|
return $this->data[$path];
|
|
|
|
}
|
|
|
|
|
2012-11-21 14:52:50 -08:00
|
|
|
public function getCacheVersion() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
Split large path lists into blocks when linting
Summary:
Fixes T5097. When linting a large list of paths (e.g., with `--everything`), do this internally:
$chunks = array_chunk($paths, 32);
foreach ($chunks as $chunk) {
$this->lintChunk($chunk);
}
This keeps the advantages of parallelism and artifact sharing for future-based linters, without having memory usage grow in an unbounded way.
These callbacks changed:
- `willLintPath()`: Useless, no meaningful implementations. Internalized the required side effect and broke the hook.
- `didRunLinters()`: Now useless, with no meaningful implementations. Broke the hook.
- `didLintPaths()`: New hook which executes opposite `willLintPaths()`.
- `lintPath()`: Linters no longer need to implement this method.
XHPAST now has an explicit way to release shared futures.
These minor changes also happened:
- Formalized the "linter ID", which is a semi-durable identifier for the cache.
- Removed linter -> exception explicit mapping, which was unused. We now just collect exceptions.
- We do the `canRun()` checks first (and separately) now.
- Share more service call profiling code.
- Fix an issue where the test harness would use the path on disk, even if configuration set a different path.
Test Plan:
- Ran `arc lint --everything` in `arcanist/`.
- With no chunking, saw **unstable** memory usage with a peak at 941 MB.
- With chunk size 32, saw **stable** memory usage with a peak at 269 MB.
- With chunk size 8, saw **stable** memory usage with a peak at 180 MB.
- Ran with `--trace` and saw profiling information.
- Created this diff.
Reviewers: joshuaspence
Reviewed By: joshuaspence
Subscribers: epriestley
Maniphest Tasks: T5097
Differential Revision: https://secure.phabricator.com/D12501
2015-04-22 05:15:57 -07:00
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function getLintMessageFullCode($short_code) {
|
2011-01-09 15:22:25 -08:00
|
|
|
return $this->getLinterName().$short_code;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function getLintMessageSeverity($code) {
|
2011-01-09 15:22:25 -08:00
|
|
|
$map = $this->customSeverityMap;
|
|
|
|
if (isset($map[$code])) {
|
|
|
|
return $map[$code];
|
|
|
|
}
|
|
|
|
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
foreach ($this->customSeverityRules as $rule => $severity) {
|
|
|
|
if (preg_match($rule, $code)) {
|
|
|
|
return $severity;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-11 23:39:12 +10:00
|
|
|
$map = $this->getLintSeverityMap();
|
|
|
|
if (isset($map[$code])) {
|
|
|
|
return $map[$code];
|
|
|
|
}
|
|
|
|
|
2013-08-23 11:58:07 -07:00
|
|
|
return $this->getDefaultMessageSeverity($code);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getDefaultMessageSeverity($code) {
|
2011-01-09 15:22:25 -08:00
|
|
|
return ArcanistLintSeverity::SEVERITY_ERROR;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function isMessageEnabled($code) {
|
2011-11-16 21:53:47 -08:00
|
|
|
return ($this->getLintMessageSeverity($code) !==
|
|
|
|
ArcanistLintSeverity::SEVERITY_DISABLED);
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function getLintMessageName($code) {
|
2011-01-09 15:22:25 -08:00
|
|
|
$map = $this->getLintNameMap();
|
|
|
|
if (isset($map[$code])) {
|
|
|
|
return $map[$code];
|
|
|
|
}
|
2014-05-23 13:53:05 -07:00
|
|
|
return 'Unknown lint message!';
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final protected function addLintMessage(ArcanistLintMessage $message) {
|
2015-05-05 20:57:23 +10:00
|
|
|
$root = $this->getEngine()->getWorkingCopy()->getProjectRoot();
|
|
|
|
$path = Filesystem::resolvePath($message->getPath(), $root);
|
|
|
|
$message->setPath(Filesystem::readablePath($path, $root));
|
|
|
|
|
2011-01-09 15:22:25 -08:00
|
|
|
$this->messages[] = $message;
|
|
|
|
return $message;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final public function getLintMessages() {
|
2011-01-09 15:22:25 -08:00
|
|
|
return $this->messages;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final protected function raiseLintAtLine(
|
2011-01-09 15:22:25 -08:00
|
|
|
$line,
|
|
|
|
$char,
|
|
|
|
$code,
|
|
|
|
$desc,
|
|
|
|
$original = null,
|
|
|
|
$replacement = null) {
|
|
|
|
|
2013-02-08 14:30:25 -08:00
|
|
|
$message = id(new ArcanistLintMessage())
|
|
|
|
->setPath($this->getActivePath())
|
|
|
|
->setLine($line)
|
|
|
|
->setChar($char)
|
|
|
|
->setCode($this->getLintMessageFullCode($code))
|
|
|
|
->setSeverity($this->getLintMessageSeverity($code))
|
|
|
|
->setName($this->getLintMessageName($code))
|
|
|
|
->setDescription($desc)
|
2013-01-10 15:03:33 -08:00
|
|
|
->setOriginalText($original)
|
|
|
|
->setReplacementText($replacement);
|
2011-01-09 15:22:25 -08:00
|
|
|
|
2013-01-10 15:03:33 -08:00
|
|
|
return $this->addLintMessage($message);
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|
|
|
|
|
2014-06-20 18:26:44 +10:00
|
|
|
final protected function raiseLintAtPath($code, $desc) {
|
2011-01-09 15:22:25 -08:00
|
|
|
return $this->raiseLintAtLine(null, null, $code, $desc, null, null);
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final protected function raiseLintAtOffset(
|
2011-01-09 15:22:25 -08:00
|
|
|
$offset,
|
|
|
|
$code,
|
|
|
|
$desc,
|
|
|
|
$original = null,
|
|
|
|
$replacement = null) {
|
|
|
|
|
|
|
|
$path = $this->getActivePath();
|
|
|
|
$engine = $this->getEngine();
|
2011-01-09 20:40:13 -08:00
|
|
|
if ($offset === null) {
|
|
|
|
$line = null;
|
|
|
|
$char = null;
|
|
|
|
} else {
|
|
|
|
list($line, $char) = $engine->getLineAndCharFromOffset($path, $offset);
|
|
|
|
}
|
2011-01-09 15:22:25 -08:00
|
|
|
|
|
|
|
return $this->raiseLintAtLine(
|
|
|
|
$line + 1,
|
|
|
|
$char + 1,
|
|
|
|
$code,
|
|
|
|
$desc,
|
|
|
|
$original,
|
|
|
|
$replacement);
|
|
|
|
}
|
|
|
|
|
2012-01-17 16:30:59 -08:00
|
|
|
public function canRun() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-01-09 15:22:25 -08:00
|
|
|
abstract public function getLinterName();
|
2012-05-31 12:09:01 -07:00
|
|
|
|
2014-05-11 13:42:56 -07:00
|
|
|
public function getVersion() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-05-12 09:48:52 -07:00
|
|
|
final protected function isCodeEnabled($code) {
|
2013-02-14 15:18:39 -08:00
|
|
|
$severity = $this->getLintMessageSeverity($code);
|
|
|
|
return $this->getEngine()->isSeverityEnabled($severity);
|
|
|
|
}
|
|
|
|
|
2012-05-31 12:09:01 -07:00
|
|
|
public function getLintSeverityMap() {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getLintNameMap() {
|
|
|
|
return array();
|
|
|
|
}
|
2011-01-09 15:22:25 -08:00
|
|
|
|
2012-11-21 18:38:24 -08:00
|
|
|
public function getCacheGranularity() {
|
|
|
|
return self::GRANULARITY_FILE;
|
|
|
|
}
|
|
|
|
|
Lay groundwork for configuration-driven linters
Summary:
Ref T2039. That task has a bunch of discussion, but basically we do a poor job of serving the midrange of lint configuration right now.
If you have something simple, the default linters work.
If you have something complex, building your own engine lets you do whatever you want.
But many users want something in between, which isn't really well accommodated. The idea is to let you write a `.arclint` file, which looks something like this:
{
"linters" : {
"css" : {
"type" : "csslint",
"include" : "(\.css$)",
"exclude" : "(^externals/)",
"bin" : "/usr/local/bin/csslint"
},
"js" : {
"type" : "jshint",
"include" : "(\.js$)",
"exclude" : "(^externals/)",
"bin" : "support/bin/jshint",
"interpreter" : "/usr/local/bin/node"
}
}
}
...which will provide a bunch of common options around lint severity, interpreter and binary locaitons, included and excluded files, etc.
This implements some basics, and very rough support in the Filename linter.
Test Plan:
Generated a `.arclint` file and saw it apply filename lint correctly. Used `debug` mode and tried invalid regexps.
{
"debug" : true,
"linters" : {
"filename" : {
"type" : "filename",
"exclude" : ["@^externals/@"]
}
}
}
Next steps include:
- Provide an external linter archetype (T3186) and expose a common set of configuration here ("bin", "interpreter", "flags", "severity").
- Provide a `.arcunit` file which works similarly (it can probably be simpler).
Reviewers: btrahan, Firehed
Reviewed By: btrahan
CC: aran
Maniphest Tasks: T2039
Differential Revision: https://secure.phabricator.com/D6797
2013-08-22 16:02:16 -07:00
|
|
|
/**
|
|
|
|
* If this linter is selectable via `.arclint` configuration files, return
|
|
|
|
* a short, human-readable name to identify it. For example, `"jshint"` or
|
|
|
|
* `"pep8"`.
|
|
|
|
*
|
|
|
|
* If you do not implement this method, the linter will not be selectable
|
|
|
|
* through `.arclint` files.
|
|
|
|
*/
|
|
|
|
public function getLinterConfigurationName() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-08-23 11:52:44 -07:00
|
|
|
public function getLinterConfigurationOptions() {
|
2014-05-11 19:28:26 -07:00
|
|
|
if (!$this->canCustomizeLintSeverities()) {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
return array(
|
2014-05-11 20:23:07 -07:00
|
|
|
'severity' => array(
|
|
|
|
'type' => 'optional map<string|int, string>',
|
|
|
|
'help' => pht(
|
|
|
|
'Provide a map from lint codes to adjusted severity levels: error, '.
|
2014-09-27 10:21:23 +10:00
|
|
|
'warning, advice, autofix or disabled.'),
|
2014-05-11 20:23:07 -07:00
|
|
|
),
|
|
|
|
'severity.rules' => array(
|
|
|
|
'type' => 'optional map<string, string>',
|
|
|
|
'help' => pht(
|
|
|
|
'Provide a map of regular expressions to severity levels. All '.
|
|
|
|
'matching codes have their severity adjusted.'),
|
|
|
|
),
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
);
|
2013-08-23 11:52:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
public function setLinterConfigurationValue($key, $value) {
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
$sev_map = array(
|
|
|
|
'error' => ArcanistLintSeverity::SEVERITY_ERROR,
|
|
|
|
'warning' => ArcanistLintSeverity::SEVERITY_WARNING,
|
|
|
|
'autofix' => ArcanistLintSeverity::SEVERITY_AUTOFIX,
|
|
|
|
'advice' => ArcanistLintSeverity::SEVERITY_ADVICE,
|
|
|
|
'disabled' => ArcanistLintSeverity::SEVERITY_DISABLED,
|
|
|
|
);
|
|
|
|
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
switch ($key) {
|
|
|
|
case 'severity':
|
2014-05-11 19:28:26 -07:00
|
|
|
if (!$this->canCustomizeLintSeverities()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
$custom = array();
|
|
|
|
foreach ($value as $code => $severity) {
|
|
|
|
if (empty($sev_map[$severity])) {
|
|
|
|
$valid = implode(', ', array_keys($sev_map));
|
|
|
|
throw new Exception(
|
|
|
|
pht(
|
|
|
|
'Unknown lint severity "%s". Valid severities are: %s.',
|
|
|
|
$severity,
|
|
|
|
$valid));
|
|
|
|
}
|
|
|
|
$code = $this->getLintCodeFromLinterConfigurationKey($code);
|
|
|
|
$custom[$code] = $severity;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->setCustomSeverityMap($custom);
|
|
|
|
return;
|
2014-05-11 19:28:26 -07:00
|
|
|
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
case 'severity.rules':
|
2014-05-11 19:28:26 -07:00
|
|
|
if (!$this->canCustomizeLintSeverities()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
foreach ($value as $rule => $severity) {
|
|
|
|
if (@preg_match($rule, '') === false) {
|
|
|
|
throw new Exception(
|
|
|
|
pht(
|
|
|
|
'Severity rule "%s" is not a valid regular expression.',
|
|
|
|
$rule));
|
|
|
|
}
|
|
|
|
if (empty($sev_map[$severity])) {
|
|
|
|
$valid = implode(', ', array_keys($sev_map));
|
|
|
|
throw new Exception(
|
|
|
|
pht(
|
|
|
|
'Unknown lint severity "%s". Valid severities are: %s.',
|
|
|
|
$severity,
|
|
|
|
$valid));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$this->setCustomSeverityRules($value);
|
|
|
|
return;
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
}
|
|
|
|
|
2013-08-23 11:52:44 -07:00
|
|
|
throw new Exception("Incomplete implementation: {$key}!");
|
|
|
|
}
|
|
|
|
|
2014-05-11 19:28:26 -07:00
|
|
|
protected function canCustomizeLintSeverities() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
Ready more linters and linter functions for .arclint
Summary:
Ref T3186. Ref T2039. Continues work on readying linters for `.arclint`.
- **Ruby**: Make this an ExternalLinter.
- **Priority**: Currently, linters have an implicit "correct" order (notably, the "NoLint" linter needs to run before other linters). Make this explicit by introducing `getLinterPriority()`.
- **Binaries**: Currently, linters manually reject binary files. Instead, reject binary files by default (linters can override this if they do want to lint binary files).
- **Deleted Files**: Currently, linters manually reject deleted files (usually in engines). Instead, reject deleted files by default (linters can override this).
- **Severity**: Move this `.arclint` config option up to top level.
- **willLintPaths()**: This method is abstract, but almost all linters provide a trivial implementation. Provide a trivial implementation in the base class.
- **getLintSeverityMap()/getLintNameMap()**: A bunch of linters have empty implementations; these are redundant. Remove them.
- **Spelling**: clean up some dead / test-only / unconventional code.
- **`.arclint`**: Allow the filename, generated, nolint, text, spelling and ruby linters to be configured via `.arclint`.
Test Plan:
https://github.com/epriestley/arclint-examples/commit/458beca3d65b64d52ed612904ae66eb837118b94
Ran unit tests.
Reviewers: btrahan
Reviewed By: btrahan
CC: Firehed, aran
Maniphest Tasks: T2039, T3186
Differential Revision: https://secure.phabricator.com/D6805
2013-08-26 05:37:10 -07:00
|
|
|
protected function shouldLintBinaryFiles() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function shouldLintDeletedFiles() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
protected function shouldLintDirectories() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-06-10 18:20:59 -07:00
|
|
|
protected function shouldLintSymbolicLinks() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
Support PHPCS as a `.arclint` linter
Summary:
Ref T3186. Ref T2039. Ref T3771. A few effects here:
# Expose PHPCS as a `.arclint` linter.
# Turn PHPCS into an ArcanistExternalLinter linter.
# Add test coverage for PHPCS.
# Add a `severity.rules` option to `.arclint`. Some linters have very explicit builtin severities ("error", "warning") but their meanings are different from how arc interprets these terms. For example, PHPCS raises "wrong indentation level" as an "error". You can already use the "severity" map to adjust individual rules, but if you want to adjust an entire linter it's currently difficult. This rule map makes it easy. There's substantial precedent for this in other linters, notably all the Python linters.
For `severity.rules`, for example, this will turn all PHPCS "errors" into warnings, and all of its warnings into advice:
"severity.rules" : {
"(^PHPCS\\.E\\.)" : "warning",
"(^PHPCS\\.W\\.)" : "advice"
}
The user can use `severity` (or more rules) to get additional granularity adjustments if they desire.
Test Plan: https://github.com/epriestley/arclint-examples/commit/5bb919bc3a684d6654dcc28622de9a29ec08307d
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, ajtrichards
Maniphest Tasks: T2039, T3186, T3771
Differential Revision: https://secure.phabricator.com/D6830
2013-08-29 06:47:27 -07:00
|
|
|
/**
|
|
|
|
* Map a configuration lint code to an `arc` lint code. Primarily, this is
|
|
|
|
* intended for validation, but can also be used to normalize case or
|
|
|
|
* otherwise be more permissive in accepted inputs.
|
|
|
|
*
|
|
|
|
* If the code is not recognized, you should throw an exception.
|
|
|
|
*
|
|
|
|
* @param string Code specified in configuration.
|
|
|
|
* @return string Normalized code to use in severity map.
|
|
|
|
*/
|
|
|
|
protected function getLintCodeFromLinterConfigurationKey($code) {
|
|
|
|
return $code;
|
|
|
|
}
|
|
|
|
|
2014-05-11 18:39:28 -07:00
|
|
|
/**
|
|
|
|
* Retrieve an old lint configuration value from `.arcconfig` or a similar
|
|
|
|
* source.
|
|
|
|
*
|
|
|
|
* Modern linters should use @{method:getConfig} to read configuration from
|
|
|
|
* `.arclint`.
|
|
|
|
*
|
|
|
|
* @param string Configuration key to retrieve.
|
|
|
|
* @param wild Default value to return if key is not present in config.
|
|
|
|
* @return wild Configured value, or default if no configuration exists.
|
|
|
|
*/
|
2014-06-17 00:30:04 +10:00
|
|
|
final protected function getDeprecatedConfiguration($key, $default = null) {
|
2014-05-11 18:39:28 -07:00
|
|
|
// If we're being called in a context without an engine (probably from
|
|
|
|
// `arc linters`), just return the default value.
|
|
|
|
if (!$this->engine) {
|
|
|
|
return $default;
|
|
|
|
}
|
|
|
|
|
|
|
|
$config = $this->getEngine()->getConfigurationManager();
|
|
|
|
|
|
|
|
// Construct a sentinel object so we can tell if we're reading config
|
|
|
|
// or not.
|
|
|
|
$sentinel = (object)array();
|
|
|
|
$result = $config->getConfigFromAnySource($key, $sentinel);
|
|
|
|
|
|
|
|
// If we read config, warn the user that this mechanism is deprecated and
|
|
|
|
// discouraged.
|
|
|
|
if ($result !== $sentinel) {
|
|
|
|
$console = PhutilConsole::getConsole();
|
|
|
|
$console->writeErr(
|
|
|
|
"**%s**: %s\n",
|
|
|
|
pht('Deprecation Warning'),
|
|
|
|
pht(
|
|
|
|
'Configuration option "%s" is deprecated. Generally, linters should '.
|
|
|
|
'now be configured using an `.arclint` file. See "Arcanist User '.
|
|
|
|
'Guide: Lint" in the documentation for more information.',
|
|
|
|
$key));
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $default;
|
|
|
|
}
|
|
|
|
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|