2011-01-09 15:22:25 -08:00
|
|
|
<?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.
|
|
|
|
*/
|
|
|
|
|
2011-02-19 11:36:08 -08:00
|
|
|
/**
|
|
|
|
* Implements lint rules, like syntax checks for a specific language.
|
|
|
|
*
|
|
|
|
* @group linter
|
|
|
|
*/
|
2011-01-09 15:22:25 -08:00
|
|
|
abstract class ArcanistLinter {
|
|
|
|
|
|
|
|
protected $paths = array();
|
|
|
|
protected $data = array();
|
|
|
|
protected $engine;
|
|
|
|
protected $activePath;
|
|
|
|
protected $messages = array();
|
|
|
|
|
|
|
|
protected $stopAllLinters = false;
|
|
|
|
|
|
|
|
private $customSeverityMap = array();
|
|
|
|
|
|
|
|
public function setCustomSeverityMap(array $map) {
|
|
|
|
$this->customSeverityMap = $map;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getActivePath() {
|
|
|
|
return $this->activePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function stopAllLinters() {
|
|
|
|
$this->stopAllLinters = true;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function didStopAllLinters() {
|
|
|
|
return $this->stopAllLinters;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function addPath($path) {
|
|
|
|
$this->paths[$path] = $path;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getPaths() {
|
|
|
|
return array_values($this->paths);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function addData($path, $data) {
|
|
|
|
$this->data[$path] = $data;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getData($path) {
|
|
|
|
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];
|
|
|
|
}
|
|
|
|
|
2011-05-06 08:34:18 -07:00
|
|
|
public function setEngine(ArcanistLintEngine $engine) {
|
2011-01-09 15:22:25 -08:00
|
|
|
$this->engine = $engine;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getEngine() {
|
|
|
|
return $this->engine;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getLintMessageFullCode($short_code) {
|
|
|
|
return $this->getLinterName().$short_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getLintMessageSeverity($code) {
|
|
|
|
$map = $this->customSeverityMap;
|
|
|
|
if (isset($map[$code])) {
|
|
|
|
return $map[$code];
|
|
|
|
}
|
|
|
|
|
|
|
|
$map = $this->getLintSeverityMap();
|
|
|
|
if (isset($map[$code])) {
|
|
|
|
return $map[$code];
|
|
|
|
}
|
|
|
|
|
|
|
|
return ArcanistLintSeverity::SEVERITY_ERROR;
|
|
|
|
}
|
|
|
|
|
2011-11-16 21:53:47 -08:00
|
|
|
public function isMessageEnabled($code) {
|
|
|
|
return ($this->getLintMessageSeverity($code) !==
|
|
|
|
ArcanistLintSeverity::SEVERITY_DISABLED);
|
|
|
|
}
|
|
|
|
|
2011-01-09 15:22:25 -08:00
|
|
|
public function getLintMessageName($code) {
|
|
|
|
$map = $this->getLintNameMap();
|
|
|
|
if (isset($map[$code])) {
|
|
|
|
return $map[$code];
|
|
|
|
}
|
|
|
|
return "Unknown lint message!";
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function addLintMessage(ArcanistLintMessage $message) {
|
|
|
|
$this->messages[] = $message;
|
|
|
|
return $message;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getLintMessages() {
|
|
|
|
return $this->messages;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function raiseLintAtLine(
|
|
|
|
$line,
|
|
|
|
$char,
|
|
|
|
$code,
|
|
|
|
$desc,
|
|
|
|
$original = null,
|
|
|
|
$replacement = null) {
|
|
|
|
|
|
|
|
$dict = array(
|
|
|
|
'path' => $this->getActivePath(),
|
|
|
|
'line' => $line,
|
|
|
|
'char' => $char,
|
|
|
|
'code' => $this->getLintMessageFullCode($code),
|
|
|
|
'severity' => $this->getLintMessageSeverity($code),
|
|
|
|
'name' => $this->getLintMessageName($code),
|
|
|
|
'description' => $desc,
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($original !== null) {
|
|
|
|
$dict['original'] = $original;
|
|
|
|
}
|
|
|
|
if ($replacement !== null) {
|
|
|
|
$dict['replacement'] = $replacement;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->addLintMessage(ArcanistLintMessage::newFromDictionary($dict));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function raiseLintAtPath(
|
|
|
|
$code,
|
|
|
|
$desc) {
|
|
|
|
|
|
|
|
return $this->raiseLintAtLine(null, null, $code, $desc, null, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function raiseLintAtOffset(
|
|
|
|
$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);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function willLintPath($path) {
|
|
|
|
$this->stopAllLinters = false;
|
|
|
|
$this->activePath = $path;
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract public function willLintPaths(array $paths);
|
|
|
|
abstract public function lintPath($path);
|
|
|
|
abstract public function getLinterName();
|
|
|
|
abstract public function getLintSeverityMap();
|
|
|
|
abstract public function getLintNameMap();
|
|
|
|
|
|
|
|
}
|