mirror of
https://we.phorge.it/source/arcanist.git
synced 2024-11-26 08:42:40 +01:00
Modernize ArcanistCppcheckLinter
.
Summary: Ref T2039. Make the `ArcanistCppcheckLinter` compatible with `.arclint`. Test Plan: Wrote and executed unit tests. Reviewers: epriestley, #blessed_reviewers Reviewed By: epriestley, #blessed_reviewers Subscribers: epriestley, Korvin Maniphest Tasks: T2039 Differential Revision: https://secure.phabricator.com/D9068
This commit is contained in:
parent
54c377448d
commit
315314425e
7 changed files with 142 additions and 98 deletions
|
@ -45,6 +45,7 @@ phutil_register_library_map(array(
|
||||||
'ArcanistConfigurationManager' => 'configuration/ArcanistConfigurationManager.php',
|
'ArcanistConfigurationManager' => 'configuration/ArcanistConfigurationManager.php',
|
||||||
'ArcanistCoverWorkflow' => 'workflow/ArcanistCoverWorkflow.php',
|
'ArcanistCoverWorkflow' => 'workflow/ArcanistCoverWorkflow.php',
|
||||||
'ArcanistCppcheckLinter' => 'lint/linter/ArcanistCppcheckLinter.php',
|
'ArcanistCppcheckLinter' => 'lint/linter/ArcanistCppcheckLinter.php',
|
||||||
|
'ArcanistCppcheckLinterTestCase' => 'lint/linter/__tests__/ArcanistCppcheckLinterTestCase.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',
|
||||||
|
@ -225,7 +226,8 @@ phutil_register_library_map(array(
|
||||||
'ArcanistConduitLinter' => 'ArcanistLinter',
|
'ArcanistConduitLinter' => 'ArcanistLinter',
|
||||||
'ArcanistConfigurationDrivenLintEngine' => 'ArcanistLintEngine',
|
'ArcanistConfigurationDrivenLintEngine' => 'ArcanistLintEngine',
|
||||||
'ArcanistCoverWorkflow' => 'ArcanistBaseWorkflow',
|
'ArcanistCoverWorkflow' => 'ArcanistBaseWorkflow',
|
||||||
'ArcanistCppcheckLinter' => 'ArcanistLinter',
|
'ArcanistCppcheckLinter' => 'ArcanistExternalLinter',
|
||||||
|
'ArcanistCppcheckLinterTestCase' => 'ArcanistArcanistLinterTestCase',
|
||||||
'ArcanistCpplintLinter' => 'ArcanistExternalLinter',
|
'ArcanistCpplintLinter' => 'ArcanistExternalLinter',
|
||||||
'ArcanistCpplintLinterTestCase' => 'ArcanistArcanistLinterTestCase',
|
'ArcanistCpplintLinterTestCase' => 'ArcanistArcanistLinterTestCase',
|
||||||
'ArcanistDiffParserTestCase' => 'ArcanistTestCase',
|
'ArcanistDiffParserTestCase' => 'ArcanistTestCase',
|
||||||
|
|
|
@ -1,121 +1,110 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Uses cppcheck to do basic checks in a cpp file
|
* Uses Cppcheck to do basic checks in a C++ file.
|
||||||
*
|
|
||||||
* You can get it here:
|
|
||||||
* http://cppcheck.sourceforge.net/
|
|
||||||
*
|
|
||||||
* @group linter
|
|
||||||
*/
|
*/
|
||||||
final class ArcanistCppcheckLinter extends ArcanistLinter {
|
final class ArcanistCppcheckLinter extends ArcanistExternalLinter {
|
||||||
|
|
||||||
|
public function getInfoName() {
|
||||||
|
return 'Cppcheck';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getInfoURI() {
|
||||||
|
return 'http://cppcheck.sourceforge.net';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getInfoDescription() {
|
||||||
|
return pht('Use `cppcheck` to perform static analysis on C/C++ code.');
|
||||||
|
}
|
||||||
|
|
||||||
public function getLinterName() {
|
public function getLinterName() {
|
||||||
return 'cppcheck';
|
return 'cppcheck';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getLintOptions() {
|
public function getLinterConfigurationName() {
|
||||||
$config = $this->getEngine()->getConfigurationManager();
|
return 'cppcheck';
|
||||||
// You will for sure want some options. The below default tends to be ok
|
|
||||||
return $config->getConfigFromAnySource(
|
|
||||||
'lint.cppcheck.options',
|
|
||||||
'-j2 --inconclusive --enable=performance,style,portability,information');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getLintPath() {
|
public function getDefaultBinary() {
|
||||||
$config = $this->getEngine()->getConfigurationManager();
|
$prefix = $this->getDeprecatedConfiguration('lint.cppcheck.prefix');
|
||||||
$prefix = $config->getConfigFromAnySource('lint.cppcheck.prefix');
|
$bin = $this->getDeprecatedConfiguration('lint.cppcheck.bin', 'cppcheck');
|
||||||
$bin = $config->getConfigFromAnySource('lint.cppcheck.bin', 'cppcheck');
|
|
||||||
|
|
||||||
if ($prefix !== null) {
|
if ($prefix) {
|
||||||
if (!Filesystem::pathExists($prefix.'/'.$bin)) {
|
return $prefix.'/'.$bin;
|
||||||
throw new ArcanistUsageException(
|
} else {
|
||||||
"Unable to find cppcheck binary in a specified directory. Make ".
|
return $bin;
|
||||||
"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.");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return csprintf("%s/%s", $prefix, $bin);
|
public function getVersion() {
|
||||||
|
list($stdout) = execx('%C --version', $this->getExecutableCommand());
|
||||||
|
|
||||||
|
$matches = array();
|
||||||
|
$regex = '/^Cppcheck (?P<version>\d+\.\d+)$/';
|
||||||
|
if (preg_match($regex, $stdout, $matches)) {
|
||||||
|
return $matches['version'];
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Look for globally installed cppcheck
|
public function getInstallInstructions() {
|
||||||
list($err) = exec_manual('which %s', $bin);
|
return pht('Install Cpplint using `apt-get install cpplint` or similar.');
|
||||||
if ($err) {
|
}
|
||||||
throw new ArcanistUsageException(
|
|
||||||
"cppcheck does not appear to be installed on this system. Install ".
|
protected function getMandatoryFlags() {
|
||||||
"it (from http://cppcheck.sourceforge.net/) or configure ".
|
return array(
|
||||||
"'lint.cppcheck.prefix' in your .arcconfig to point to the ".
|
'--quiet',
|
||||||
"directory where it resides."
|
'--inline-suppr',
|
||||||
|
'--xml',
|
||||||
|
'--xml-version=2',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $bin;
|
protected function getDefaultFlags() {
|
||||||
}
|
return $this->getDeprecatedConfiguration(
|
||||||
|
'lint.cppcheck.options',
|
||||||
public function lintPath($path) {
|
array('-j2', '--enable=performance,style,portability,information'));
|
||||||
$bin = $this->getLintPath();
|
|
||||||
$options = $this->getLintOptions();
|
|
||||||
|
|
||||||
list($rc, $stdout, $stderr) = exec_manual(
|
|
||||||
"%C %C --inline-suppr --xml-version=2 -q %s",
|
|
||||||
$bin,
|
|
||||||
$options,
|
|
||||||
$this->getEngine()->getFilePathOnDisk($path));
|
|
||||||
|
|
||||||
if ($rc === 1) {
|
|
||||||
throw new Exception("cppcheck failed to run correctly:\n".$stderr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function parseLinterOutput($path, $err, $stdout, $stderr) {
|
||||||
$dom = new DOMDocument();
|
$dom = new DOMDocument();
|
||||||
libxml_clear_errors();
|
$ok = @$dom->loadXML($stderr);
|
||||||
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.");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if (!$ok) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
$errors = $dom->getElementsByTagName('error');
|
$errors = $dom->getElementsByTagName('error');
|
||||||
|
$messages = array();
|
||||||
foreach ($errors as $error) {
|
foreach ($errors as $error) {
|
||||||
$loc_node = $error->getElementsByTagName('location');
|
foreach ($error->getElementsByTagName('location') as $location) {
|
||||||
if (!$loc_node) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$location = $loc_node->item(0);
|
|
||||||
if (!$location) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$file = $location->getAttribute('file');
|
|
||||||
if ($file != Filesystem::resolvePath($path)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$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 = new ArcanistLintMessage();
|
||||||
$message->setPath($path);
|
$message->setPath($location->getAttribute('file'));
|
||||||
$message->setLine($line);
|
$message->setLine($location->getAttribute('line'));
|
||||||
$message->setCode($severity);
|
$message->setCode('Cppcheck');
|
||||||
$message->setName($id);
|
$message->setName($error->getAttribute('id'));
|
||||||
$message->setDescription($msg);
|
$message->setDescription($error->getAttribute('msg'));
|
||||||
$message->setSeverity($severity_code);
|
|
||||||
|
|
||||||
$this->addLintMessage($message);
|
switch ($error->getAttribute('severity')) {
|
||||||
|
case 'error':
|
||||||
|
$message->setSeverity(ArcanistLintSeverity::SEVERITY_ERROR);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
if ($error->getAttribute('inconclusive')) {
|
||||||
|
$message->setSeverity(ArcanistLintSeverity::SEVERITY_ADVICE);
|
||||||
|
} else {
|
||||||
|
$message->setSeverity(ArcanistLintSeverity::SEVERITY_WARNING);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
$messages[] = $message;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return $messages;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
12
src/lint/linter/__tests__/ArcanistCppcheckLinterTestCase.php
Normal file
12
src/lint/linter/__tests__/ArcanistCppcheckLinterTestCase.php
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class ArcanistCppcheckLinterTestCase
|
||||||
|
extends ArcanistArcanistLinterTestCase {
|
||||||
|
|
||||||
|
public function testCppcheckLint() {
|
||||||
|
return $this->executeTestsInDirectory(
|
||||||
|
dirname(__FILE__).'/cppcheck/',
|
||||||
|
new ArcanistCppcheckLinter());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
9
src/lint/linter/__tests__/cppcheck/file1.lint-test
Normal file
9
src/lint/linter/__tests__/cppcheck/file1.lint-test
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
char a[10];
|
||||||
|
a[10] = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
~~~~~~~~~~
|
||||||
|
error:4:
|
||||||
|
warning:4:
|
|
@ -0,0 +1,7 @@
|
||||||
|
void f() {
|
||||||
|
char arr[5];
|
||||||
|
// cppcheck-suppress arrayIndexOutOfBounds
|
||||||
|
// cppcheck-suppress unreadVariable
|
||||||
|
arr[10] = 0;
|
||||||
|
}
|
||||||
|
~~~~~~~~~~
|
5
src/lint/linter/__tests__/cppcheck/ok.lint-test
Normal file
5
src/lint/linter/__tests__/cppcheck/ok.lint-test
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
~~~~~~~~~~
|
20
src/lint/linter/__tests__/cppcheck/zblair.lint-test
Normal file
20
src/lint/linter/__tests__/cppcheck/zblair.lint-test
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
/**
|
||||||
|
* Taken from http://www.slideshare.net/zblair/cppcheck-10316379
|
||||||
|
*/
|
||||||
|
void foo(char* str) {
|
||||||
|
char* buf = new char[8];
|
||||||
|
strcpy(buf, str);
|
||||||
|
|
||||||
|
FILE* file = fopen("out.txt", "w");
|
||||||
|
if (!file)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (char* c = buf; *c; ++c)
|
||||||
|
fputc((int)*c, file);
|
||||||
|
|
||||||
|
delete buf;
|
||||||
|
}
|
||||||
|
~~~~~~~~~~
|
||||||
|
error:10:
|
||||||
|
error:15:
|
||||||
|
error:16:
|
Loading…
Reference in a new issue