2011-02-13 20:57:14 +01:00
|
|
|
<?php
|
|
|
|
|
2011-02-19 20:36:08 +01:00
|
|
|
/**
|
|
|
|
* Facilitiates implementation of test cases for @{class:ArcanistLinter}s.
|
|
|
|
*
|
|
|
|
* @group testcase
|
|
|
|
*/
|
2012-11-03 02:09:38 +01:00
|
|
|
abstract class ArcanistLinterTestCase extends ArcanistPhutilTestCase {
|
2011-02-13 20:57:14 +01:00
|
|
|
|
2011-02-16 20:53:34 +01:00
|
|
|
public function executeTestsInDirectory($root, $linter, $working_copy) {
|
2011-02-13 20:57:14 +01:00
|
|
|
foreach (Filesystem::listDirectory($root, $hidden = false) as $file) {
|
2011-02-16 20:53:34 +01:00
|
|
|
$this->lintFile($root.$file, $linter, $working_copy);
|
2011-02-13 20:57:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-16 20:53:34 +01:00
|
|
|
private function lintFile($file, $linter, $working_copy) {
|
2011-02-13 20:57:14 +01:00
|
|
|
$linter = clone $linter;
|
|
|
|
|
|
|
|
$contents = Filesystem::readFile($file);
|
|
|
|
$contents = explode("~~~~~~~~~~\n", $contents);
|
|
|
|
if (count($contents) < 2) {
|
|
|
|
throw new Exception(
|
|
|
|
"Expected '~~~~~~~~~~' separating test case and results.");
|
|
|
|
}
|
|
|
|
|
|
|
|
list ($data, $expect, $xform, $config) = array_merge(
|
|
|
|
$contents,
|
|
|
|
array(null, null));
|
|
|
|
|
|
|
|
$basename = basename($file);
|
|
|
|
|
|
|
|
if ($config) {
|
|
|
|
$config = json_decode($config, true);
|
|
|
|
if (!is_array($config)) {
|
|
|
|
throw new Exception(
|
|
|
|
"Invalid configuration in test '{$basename}', not valid JSON.");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$config = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: ?
|
|
|
|
validate_parameter_list(
|
|
|
|
$config,
|
|
|
|
array(
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'project' => true,
|
|
|
|
'path' => true,
|
|
|
|
'hook' => true,
|
|
|
|
));
|
|
|
|
*/
|
|
|
|
|
|
|
|
$exception = null;
|
|
|
|
$after_lint = null;
|
|
|
|
$messages = null;
|
|
|
|
$exception_message = false;
|
|
|
|
$caught_exception = false;
|
|
|
|
try {
|
|
|
|
|
|
|
|
$path = idx($config, 'path', 'lint/'.$basename.'.php');
|
|
|
|
|
|
|
|
$engine = new UnitTestableArcanistLintEngine();
|
|
|
|
$engine->setWorkingCopy($working_copy);
|
|
|
|
$engine->setPaths(array($path));
|
|
|
|
|
2011-02-16 20:53:34 +01:00
|
|
|
$engine->setCommitHookMode(idx($config, 'hook', false));
|
2011-02-13 20:57:14 +01:00
|
|
|
|
|
|
|
$linter->addPath($path);
|
|
|
|
$linter->addData($path, $data);
|
2012-11-06 00:58:24 +01:00
|
|
|
$linter->setConfig(idx($config, 'config', array()));
|
2011-02-13 20:57:14 +01:00
|
|
|
|
|
|
|
$engine->addLinter($linter);
|
|
|
|
$engine->addFileData($path, $data);
|
|
|
|
|
|
|
|
$results = $engine->run();
|
|
|
|
$this->assertEqual(
|
|
|
|
1,
|
|
|
|
count($results),
|
|
|
|
'Expect one result returned by linter.');
|
|
|
|
|
|
|
|
$result = reset($results);
|
|
|
|
$patcher = ArcanistLintPatcher::newFromArcanistLintResult($result);
|
|
|
|
$after_lint = $patcher->getModifiedFileContent();
|
|
|
|
|
|
|
|
} catch (ArcanistPhutilTestTerminatedException $ex) {
|
|
|
|
throw $ex;
|
|
|
|
} catch (Exception $exception) {
|
|
|
|
$caught_exception = true;
|
2012-10-20 15:16:19 +02:00
|
|
|
if ($exception instanceof PhutilAggregateException) {
|
|
|
|
$caught_exception = false;
|
|
|
|
foreach ($exception->getExceptions() as $ex) {
|
|
|
|
if ($ex instanceof ArcanistUsageException) {
|
|
|
|
$this->assertSkipped($ex->getMessage());
|
|
|
|
} else {
|
|
|
|
$caught_exception = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-02-13 20:57:14 +01:00
|
|
|
$exception_message = $exception->getMessage()."\n\n".
|
|
|
|
$exception->getTraceAsString();
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ($basename) {
|
|
|
|
default:
|
|
|
|
$this->assertEqual(false, $caught_exception, $exception_message);
|
|
|
|
$this->compareLint($basename, $expect, $result);
|
|
|
|
$this->compareTransform($xform, $after_lint);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function compareLint($file, $expect, $result) {
|
|
|
|
$seen = array();
|
|
|
|
$raised = array();
|
2012-01-28 20:17:45 +01:00
|
|
|
$message_map = array();
|
2011-02-13 20:57:14 +01:00
|
|
|
foreach ($result->getMessages() as $message) {
|
|
|
|
$sev = $message->getSeverity();
|
|
|
|
$line = $message->getLine();
|
|
|
|
$char = $message->getChar();
|
|
|
|
$code = $message->getCode();
|
|
|
|
$name = $message->getName();
|
2012-01-28 20:17:45 +01:00
|
|
|
$message_key = $sev.":".$line.":".$char;
|
|
|
|
$message_map[$message_key] = $message;
|
|
|
|
$seen[] = $message_key;
|
2011-02-13 20:57:14 +01:00
|
|
|
$raised[] = " {$sev} at line {$line}, char {$char}: {$code} {$name}";
|
|
|
|
}
|
|
|
|
$expect = trim($expect);
|
|
|
|
if ($expect) {
|
|
|
|
$expect = explode("\n", $expect);
|
|
|
|
} else {
|
|
|
|
$expect = array();
|
|
|
|
}
|
|
|
|
foreach ($expect as $key => $expected) {
|
Add a lint check for clobbering locals with iterators
Summary:
See D2049, D2050. Identify reuses of locals as iterator variables. Before raising an error, we require:
- Variable is declared before the loop.
- Variable is used after the loop, ignoring uses as an iterator variable.
I think this identifies all problems with a very low false positive rate (the false positives are suspicious/unconventional code, but not necessarily errors).
Also fix an issue identified by the linter.
Test Plan:
- Verified this identified the bugs in D2049 and D2050.
- Ran linter against libphutil/, arcanist/ and phabricator/ (see D2051, this, and next diff).
- Ran unit tests.
Reviewers: vrana, btrahan
Reviewed By: vrana
CC: aran, epriestley, jungejason
Differential Revision: https://secure.phabricator.com/D2052
2012-03-29 22:21:18 +02:00
|
|
|
$expect[$key] = head(explode(' ', $expected));
|
2011-02-13 20:57:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$expect = array_fill_keys($expect, true);
|
|
|
|
$seen = array_fill_keys($seen, true);
|
|
|
|
|
|
|
|
if (!$raised) {
|
|
|
|
$raised = array("No messages.");
|
|
|
|
}
|
|
|
|
$raised = "Actually raised:\n".implode("\n", $raised);
|
|
|
|
|
|
|
|
foreach (array_diff_key($expect, $seen) as $missing => $ignored) {
|
|
|
|
list($sev, $line, $char) = explode(':', $missing);
|
|
|
|
$this->assertFailure(
|
|
|
|
"In '{$file}', ".
|
|
|
|
"expected lint to raise {$sev} on line {$line} at char {$char}, ".
|
|
|
|
"but no {$sev} was raised. {$raised}");
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (array_diff_key($seen, $expect) as $surprising => $ignored) {
|
2012-01-28 20:17:45 +01:00
|
|
|
|
|
|
|
$message = $message_map[$surprising];
|
|
|
|
$message_info = $message->getDescription();
|
|
|
|
|
2011-02-13 20:57:14 +01:00
|
|
|
list($sev, $line, $char) = explode(':', $surprising);
|
|
|
|
$this->assertFailure(
|
|
|
|
"In '{$file}', ".
|
|
|
|
"lint raised {$sev} on line {$line} at char {$char}, ".
|
2012-01-28 20:17:45 +01:00
|
|
|
"but nothing was expected:\n\n{$message_info}\n\n{$raised}");
|
2011-02-13 20:57:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-05 19:25:43 +01:00
|
|
|
protected function compareTransform($expected, $actual) {
|
2011-02-13 20:57:14 +01:00
|
|
|
if (!strlen($expected)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$this->assertEqual(
|
|
|
|
$expected,
|
|
|
|
$actual,
|
|
|
|
"File as patched by lint did not match the expected patched file.");
|
|
|
|
}
|
|
|
|
}
|