mirror of
https://we.phorge.it/source/arcanist.git
synced 2024-12-01 19:22:41 +01:00
CppCheck linter
Summary: Lints cpp code using the cppcheck static linter. This linter needs to be downloaded and built from http://cppcheck.sourceforge.net/ Test Plan: Used it on a few files. Reviewers: epriestley Reviewed By: epriestley CC: aran, Korvin Differential Revision: https://secure.phabricator.com/D4353
This commit is contained in:
parent
854c1b67b1
commit
cf8d445c9f
2 changed files with 139 additions and 0 deletions
|
@ -36,6 +36,7 @@ phutil_register_library_map(array(
|
||||||
'ArcanistConduitLinter' => 'lint/linter/ArcanistConduitLinter.php',
|
'ArcanistConduitLinter' => 'lint/linter/ArcanistConduitLinter.php',
|
||||||
'ArcanistConfiguration' => 'configuration/ArcanistConfiguration.php',
|
'ArcanistConfiguration' => 'configuration/ArcanistConfiguration.php',
|
||||||
'ArcanistCoverWorkflow' => 'workflow/ArcanistCoverWorkflow.php',
|
'ArcanistCoverWorkflow' => 'workflow/ArcanistCoverWorkflow.php',
|
||||||
|
'ArcanistCppcheckLinter' => 'lint/linter/ArcanistCppcheckLinter.php',
|
||||||
'ArcanistCpplintLinter' => 'lint/linter/ArcanistCpplintLinter.php',
|
'ArcanistCpplintLinter' => 'lint/linter/ArcanistCpplintLinter.php',
|
||||||
'ArcanistCpplintLinterTestCase' => 'lint/linter/__tests__/ArcanistCpplintLinterTestCase.php',
|
'ArcanistCpplintLinterTestCase' => 'lint/linter/__tests__/ArcanistCpplintLinterTestCase.php',
|
||||||
'ArcanistDiffChange' => 'parser/diff/ArcanistDiffChange.php',
|
'ArcanistDiffChange' => 'parser/diff/ArcanistDiffChange.php',
|
||||||
|
@ -176,6 +177,7 @@ phutil_register_library_map(array(
|
||||||
'ArcanistCommitWorkflow' => 'ArcanistBaseWorkflow',
|
'ArcanistCommitWorkflow' => 'ArcanistBaseWorkflow',
|
||||||
'ArcanistConduitLinter' => 'ArcanistLinter',
|
'ArcanistConduitLinter' => 'ArcanistLinter',
|
||||||
'ArcanistCoverWorkflow' => 'ArcanistBaseWorkflow',
|
'ArcanistCoverWorkflow' => 'ArcanistBaseWorkflow',
|
||||||
|
'ArcanistCppcheckLinter' => 'ArcanistLinter',
|
||||||
'ArcanistCpplintLinter' => 'ArcanistLinter',
|
'ArcanistCpplintLinter' => 'ArcanistLinter',
|
||||||
'ArcanistCpplintLinterTestCase' => 'ArcanistArcanistLinterTestCase',
|
'ArcanistCpplintLinterTestCase' => 'ArcanistArcanistLinterTestCase',
|
||||||
'ArcanistDiffParserTestCase' => 'ArcanistTestCase',
|
'ArcanistDiffParserTestCase' => 'ArcanistTestCase',
|
||||||
|
|
137
src/lint/linter/ArcanistCppcheckLinter.php
Normal file
137
src/lint/linter/ArcanistCppcheckLinter.php
Normal file
|
@ -0,0 +1,137 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Uses cppcheck to do basic checks in a cpp file
|
||||||
|
*
|
||||||
|
* You can get it here:
|
||||||
|
* http://cppcheck.sourceforge.net/
|
||||||
|
*
|
||||||
|
* @group linter
|
||||||
|
*/
|
||||||
|
final class ArcanistCppcheckLinter extends ArcanistLinter {
|
||||||
|
|
||||||
|
public function willLintPaths(array $paths) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLinterName() {
|
||||||
|
return 'cppcheck';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLintOptions() {
|
||||||
|
$working_copy = $this->getEngine()->getWorkingCopy();
|
||||||
|
// You will for sure want some options. The below default tends to be ok
|
||||||
|
$options = $working_copy->getConfig(
|
||||||
|
'lint.cppcheck.options',
|
||||||
|
'-j2 --inconclusive --enable=performance,style,portability,information'
|
||||||
|
);
|
||||||
|
|
||||||
|
return $options;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLintPath() {
|
||||||
|
$working_copy = $this->getEngine()->getWorkingCopy();
|
||||||
|
$prefix = $working_copy->getConfig('lint.cppcheck.prefix');
|
||||||
|
$bin = $working_copy->getConfig('lint.cppcheck.bin');
|
||||||
|
|
||||||
|
if ($bin === null && $prefix === null) {
|
||||||
|
$bin = 'cppcheck';
|
||||||
|
} else {
|
||||||
|
if ($bin === null) {
|
||||||
|
$bin = 'cppcheck';
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($prefix !== null) {
|
||||||
|
if (!Filesystem::pathExists($prefix.'/'.$bin)) {
|
||||||
|
throw new ArcanistUsageException(
|
||||||
|
"Unable to find cppcheck binary in a specified directory. Make ".
|
||||||
|
"sure that 'lint.cppcheck.prefix' and 'lint.cppcheck.bin' keys are".
|
||||||
|
" set correctly. If you'd rather use a copy of cppcheck installed ".
|
||||||
|
"globally, you can just remove these keys from your .arcconfig");
|
||||||
|
}
|
||||||
|
|
||||||
|
$bin = csprintf("%s/%s", $prefix, $bin);
|
||||||
|
|
||||||
|
return $bin;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look for globally installed cppcheck
|
||||||
|
list($err) = exec_manual('which %s', $bin);
|
||||||
|
if ($err) {
|
||||||
|
throw new ArcanistUsageException(
|
||||||
|
"cppcheck does not appear to be installed on this system. Install".
|
||||||
|
"it (from http://cppcheck.sourceforge.net/) or configure".
|
||||||
|
"'lint.cppcheck.prefix' in your .arcconfig to point to the".
|
||||||
|
"directory where it resides."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $bin;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public function lintPath($path) {
|
||||||
|
$bin = $this->getLintPath();
|
||||||
|
$options = $this->getLintOptions();
|
||||||
|
|
||||||
|
list($rc, $stdout, $stderr) = exec_manual(
|
||||||
|
"%C %C --inline-suppr --xml-version=2 -q %s",
|
||||||
|
$bin,
|
||||||
|
$options,
|
||||||
|
$path
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($rc === 1) {
|
||||||
|
throw new Exception("cppcheck failed to run correctly:\n".$stderr);
|
||||||
|
}
|
||||||
|
|
||||||
|
$dom = new DOMDocument();
|
||||||
|
libxml_clear_errors();
|
||||||
|
if ($dom->loadXML($stderr) === false || libxml_get_errors()) {
|
||||||
|
throw new ArcanistUsageException('cppcheck Linter failed to load ' .
|
||||||
|
'output. Something happened when running cppcheck. ' .
|
||||||
|
"Output:\n$stderr" .
|
||||||
|
"\nTry running lint with --trace flag to get more details.");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
$errors = $dom->getElementsByTagName('error');
|
||||||
|
foreach ($errors as $error) {
|
||||||
|
$loc_node = $error->getElementsByTagName('location');
|
||||||
|
if (!$loc_node) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$location = $loc_node->item(0);
|
||||||
|
if (!$location) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$file = $location->getAttribute('file');
|
||||||
|
$line = $location->getAttribute('line');
|
||||||
|
|
||||||
|
$id = $error->getAttribute('id');
|
||||||
|
$severity = $error->getAttribute('severity');
|
||||||
|
$msg = $error->getAttribute('msg');
|
||||||
|
$inconclusive = $error->getAttribute('inconclusive');
|
||||||
|
$verbose_msg = $error->getAttribute('verbose');
|
||||||
|
|
||||||
|
$severity_code = ArcanistLintSeverity::SEVERITY_WARNING;
|
||||||
|
if ($inconclusive) {
|
||||||
|
$severity_code = ArcanistLintSeverity::SEVERITY_ADVICE;
|
||||||
|
} else if (stripos($severity, 'error') !== false) {
|
||||||
|
$severity_code = ArcanistLintSeverity::SEVERITY_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
$message = new ArcanistLintMessage();
|
||||||
|
$message->setPath($path);
|
||||||
|
$message->setLine($line);
|
||||||
|
$message->setCode($severity);
|
||||||
|
$message->setName($id);
|
||||||
|
$message->setDescription($msg);
|
||||||
|
$message->setSeverity($severity_code);
|
||||||
|
|
||||||
|
$this->addLintMessage($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in a new issue