2011-01-10 00:22:25 +01:00
|
|
|
<?php
|
|
|
|
|
2011-02-19 20:36:08 +01:00
|
|
|
/**
|
2012-01-31 21:07:19 +01:00
|
|
|
* Uses XHPAST to apply lint rules to PHP.
|
2011-02-19 20:36:08 +01:00
|
|
|
*/
|
2013-01-29 07:03:03 +01:00
|
|
|
final class ArcanistXHPASTLinter extends ArcanistBaseXHPASTLinter {
|
2011-01-10 00:22:25 +01:00
|
|
|
|
2012-01-26 04:05:47 +01:00
|
|
|
const LINT_PHP_SYNTAX_ERROR = 1;
|
|
|
|
const LINT_UNABLE_TO_PARSE = 2;
|
|
|
|
const LINT_VARIABLE_VARIABLE = 3;
|
|
|
|
const LINT_EXTRACT_USE = 4;
|
|
|
|
const LINT_UNDECLARED_VARIABLE = 5;
|
|
|
|
const LINT_PHP_SHORT_TAG = 6;
|
|
|
|
const LINT_PHP_ECHO_TAG = 7;
|
|
|
|
const LINT_PHP_CLOSE_TAG = 8;
|
|
|
|
const LINT_NAMING_CONVENTIONS = 9;
|
|
|
|
const LINT_IMPLICIT_CONSTRUCTOR = 10;
|
|
|
|
const LINT_DYNAMIC_DEFINE = 12;
|
|
|
|
const LINT_STATIC_THIS = 13;
|
|
|
|
const LINT_PREG_QUOTE_MISUSE = 14;
|
|
|
|
const LINT_PHP_OPEN_TAG = 15;
|
|
|
|
const LINT_TODO_COMMENT = 16;
|
|
|
|
const LINT_EXIT_EXPRESSION = 17;
|
|
|
|
const LINT_COMMENT_STYLE = 18;
|
|
|
|
const LINT_CLASS_FILENAME_MISMATCH = 19;
|
|
|
|
const LINT_TAUTOLOGICAL_EXPRESSION = 20;
|
|
|
|
const LINT_PLUS_OPERATOR_ON_STRINGS = 21;
|
|
|
|
const LINT_DUPLICATE_KEYS_IN_ARRAY = 22;
|
|
|
|
const LINT_REUSED_ITERATORS = 23;
|
|
|
|
const LINT_BRACE_FORMATTING = 24;
|
|
|
|
const LINT_PARENTHESES_SPACING = 25;
|
|
|
|
const LINT_CONTROL_STATEMENT_SPACING = 26;
|
|
|
|
const LINT_BINARY_EXPRESSION_SPACING = 27;
|
2012-01-28 20:29:30 +01:00
|
|
|
const LINT_ARRAY_INDEX_SPACING = 28;
|
2012-03-08 21:20:19 +01:00
|
|
|
const LINT_IMPLICIT_FALLTHROUGH = 30;
|
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
|
|
|
const LINT_REUSED_AS_ITERATOR = 32;
|
2012-06-22 01:21:04 +02:00
|
|
|
const LINT_COMMENT_SPACING = 34;
|
2012-08-15 21:56:36 +02:00
|
|
|
const LINT_SLOWNESS = 36;
|
2013-02-19 22:50:13 +01:00
|
|
|
const LINT_CLOSING_CALL_PAREN = 37;
|
|
|
|
const LINT_CLOSING_DECL_PAREN = 38;
|
2013-03-05 16:45:02 +01:00
|
|
|
const LINT_REUSED_ITERATOR_REFERENCE = 39;
|
2013-04-15 19:22:56 +02:00
|
|
|
const LINT_KEYWORD_CASING = 40;
|
2014-05-17 04:25:37 +02:00
|
|
|
const LINT_DOUBLE_QUOTE = 41;
|
2014-05-19 15:33:22 +02:00
|
|
|
const LINT_ELSEIF_USAGE = 42;
|
2014-05-19 16:48:58 +02:00
|
|
|
const LINT_SEMICOLON_SPACING = 43;
|
2014-05-28 01:16:39 +02:00
|
|
|
const LINT_CONCATENATION_OPERATOR = 44;
|
2014-06-17 17:30:15 +02:00
|
|
|
const LINT_PHP_COMPATIBILITY = 45;
|
2014-08-05 14:35:34 +02:00
|
|
|
const LINT_LANGUAGE_CONSTRUCT_PAREN = 46;
|
2014-09-09 14:48:45 +02:00
|
|
|
const LINT_EMPTY_STATEMENT = 47;
|
2011-01-10 00:22:25 +01:00
|
|
|
|
2014-05-12 13:46:56 +02:00
|
|
|
private $naminghook;
|
|
|
|
private $switchhook;
|
2014-06-17 17:30:15 +02:00
|
|
|
private $version;
|
|
|
|
private $windowsVersion;
|
2014-05-12 13:46:56 +02:00
|
|
|
|
2014-05-12 14:06:04 +02:00
|
|
|
public function getInfoName() {
|
|
|
|
return 'XHPAST Lint';
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getInfoDescription() {
|
2014-07-11 17:25:51 +02:00
|
|
|
return pht('Use XHPAST to enforce coding conventions on PHP source files.');
|
2014-05-12 14:06:04 +02:00
|
|
|
}
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
public function getLintNameMap() {
|
|
|
|
return array(
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_PHP_SYNTAX_ERROR => 'PHP Syntax Error!',
|
|
|
|
self::LINT_UNABLE_TO_PARSE => 'Unable to Parse',
|
|
|
|
self::LINT_VARIABLE_VARIABLE => 'Use of Variable Variable',
|
|
|
|
self::LINT_EXTRACT_USE => 'Use of extract()',
|
|
|
|
self::LINT_UNDECLARED_VARIABLE => 'Use of Undeclared Variable',
|
|
|
|
self::LINT_PHP_SHORT_TAG => 'Use of Short Tag "<?"',
|
|
|
|
self::LINT_PHP_ECHO_TAG => 'Use of Echo Tag "<?="',
|
|
|
|
self::LINT_PHP_CLOSE_TAG => 'Use of Close Tag "?>"',
|
|
|
|
self::LINT_NAMING_CONVENTIONS => 'Naming Conventions',
|
|
|
|
self::LINT_IMPLICIT_CONSTRUCTOR => 'Implicit Constructor',
|
|
|
|
self::LINT_DYNAMIC_DEFINE => 'Dynamic define()',
|
|
|
|
self::LINT_STATIC_THIS => 'Use of $this in Static Context',
|
|
|
|
self::LINT_PREG_QUOTE_MISUSE => 'Misuse of preg_quote()',
|
|
|
|
self::LINT_PHP_OPEN_TAG => 'Expected Open Tag',
|
|
|
|
self::LINT_TODO_COMMENT => 'TODO Comment',
|
|
|
|
self::LINT_EXIT_EXPRESSION => 'Exit Used as Expression',
|
|
|
|
self::LINT_COMMENT_STYLE => 'Comment Style',
|
|
|
|
self::LINT_CLASS_FILENAME_MISMATCH => 'Class-Filename Mismatch',
|
|
|
|
self::LINT_TAUTOLOGICAL_EXPRESSION => 'Tautological Expression',
|
|
|
|
self::LINT_PLUS_OPERATOR_ON_STRINGS => 'Not String Concatenation',
|
|
|
|
self::LINT_DUPLICATE_KEYS_IN_ARRAY => 'Duplicate Keys in Array',
|
|
|
|
self::LINT_REUSED_ITERATORS => 'Reuse of Iterator Variable',
|
|
|
|
self::LINT_BRACE_FORMATTING => 'Brace placement',
|
|
|
|
self::LINT_PARENTHESES_SPACING => 'Spaces Inside Parentheses',
|
|
|
|
self::LINT_CONTROL_STATEMENT_SPACING => 'Space After Control Statement',
|
|
|
|
self::LINT_BINARY_EXPRESSION_SPACING => 'Space Around Binary Operator',
|
|
|
|
self::LINT_ARRAY_INDEX_SPACING => 'Spacing Before Array Index',
|
2012-03-08 21:20:19 +01:00
|
|
|
self::LINT_IMPLICIT_FALLTHROUGH => 'Implicit Fallthrough',
|
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
|
|
|
self::LINT_REUSED_AS_ITERATOR => 'Variable Reused As Iterator',
|
2012-06-22 01:21:04 +02:00
|
|
|
self::LINT_COMMENT_SPACING => 'Comment Spaces',
|
2012-08-15 21:56:36 +02:00
|
|
|
self::LINT_SLOWNESS => 'Slow Construct',
|
2013-02-19 22:50:13 +01:00
|
|
|
self::LINT_CLOSING_CALL_PAREN => 'Call Formatting',
|
|
|
|
self::LINT_CLOSING_DECL_PAREN => 'Declaration Formatting',
|
2013-03-05 16:45:02 +01:00
|
|
|
self::LINT_REUSED_ITERATOR_REFERENCE => 'Reuse of Iterator References',
|
2013-04-15 19:22:56 +02:00
|
|
|
self::LINT_KEYWORD_CASING => 'Keyword Conventions',
|
2014-05-17 04:25:37 +02:00
|
|
|
self::LINT_DOUBLE_QUOTE => 'Unnecessary Double Quotes',
|
2014-05-19 15:33:22 +02:00
|
|
|
self::LINT_ELSEIF_USAGE => 'ElseIf Usage',
|
2014-05-19 16:48:58 +02:00
|
|
|
self::LINT_SEMICOLON_SPACING => 'Semicolon Spacing',
|
2014-05-28 01:16:39 +02:00
|
|
|
self::LINT_CONCATENATION_OPERATOR => 'Concatenation Spacing',
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY => 'PHP Compatibility',
|
2014-08-05 14:35:34 +02:00
|
|
|
self::LINT_LANGUAGE_CONSTRUCT_PAREN => 'Language Construct Parentheses',
|
2014-09-09 14:48:45 +02:00
|
|
|
self::LINT_EMPTY_STATEMENT => 'Empty Block Statement',
|
2011-01-10 00:22:25 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getLinterName() {
|
|
|
|
return 'XHP';
|
|
|
|
}
|
|
|
|
|
2014-05-12 04:33:40 +02:00
|
|
|
public function getLinterConfigurationName() {
|
|
|
|
return 'xhpast';
|
|
|
|
}
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
public function getLintSeverityMap() {
|
2013-01-11 21:38:08 +01:00
|
|
|
$disabled = ArcanistLintSeverity::SEVERITY_DISABLED;
|
|
|
|
$advice = ArcanistLintSeverity::SEVERITY_ADVICE;
|
|
|
|
$warning = ArcanistLintSeverity::SEVERITY_WARNING;
|
2012-01-26 04:05:47 +01:00
|
|
|
|
2013-01-11 21:38:08 +01:00
|
|
|
return array(
|
|
|
|
self::LINT_TODO_COMMENT => $disabled,
|
|
|
|
self::LINT_UNABLE_TO_PARSE => $warning,
|
|
|
|
self::LINT_NAMING_CONVENTIONS => $warning,
|
|
|
|
self::LINT_PREG_QUOTE_MISUSE => $advice,
|
|
|
|
self::LINT_BRACE_FORMATTING => $warning,
|
|
|
|
self::LINT_PARENTHESES_SPACING => $warning,
|
|
|
|
self::LINT_CONTROL_STATEMENT_SPACING => $warning,
|
|
|
|
self::LINT_BINARY_EXPRESSION_SPACING => $warning,
|
|
|
|
self::LINT_ARRAY_INDEX_SPACING => $warning,
|
|
|
|
self::LINT_IMPLICIT_FALLTHROUGH => $warning,
|
|
|
|
self::LINT_SLOWNESS => $warning,
|
|
|
|
self::LINT_COMMENT_SPACING => $advice,
|
2013-02-19 22:50:13 +01:00
|
|
|
self::LINT_CLOSING_CALL_PAREN => $warning,
|
|
|
|
self::LINT_CLOSING_DECL_PAREN => $warning,
|
2013-03-05 16:45:02 +01:00
|
|
|
self::LINT_REUSED_ITERATOR_REFERENCE => $warning,
|
2013-04-15 19:22:56 +02:00
|
|
|
self::LINT_KEYWORD_CASING => $warning,
|
2014-05-17 04:25:37 +02:00
|
|
|
self::LINT_DOUBLE_QUOTE => $advice,
|
2014-05-19 15:33:22 +02:00
|
|
|
self::LINT_ELSEIF_USAGE => $advice,
|
2014-05-19 16:48:58 +02:00
|
|
|
self::LINT_SEMICOLON_SPACING => $advice,
|
2014-05-28 01:16:39 +02:00
|
|
|
self::LINT_CONCATENATION_OPERATOR => $warning,
|
2014-08-05 14:35:34 +02:00
|
|
|
self::LINT_LANGUAGE_CONSTRUCT_PAREN => $warning,
|
2014-09-09 14:48:45 +02:00
|
|
|
self::LINT_EMPTY_STATEMENT => $advice,
|
2011-01-10 00:22:25 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2014-05-12 13:46:56 +02:00
|
|
|
public function getLinterConfigurationOptions() {
|
|
|
|
return parent::getLinterConfigurationOptions() + array(
|
|
|
|
'xhpast.naminghook' => array(
|
2014-05-12 14:06:14 +02:00
|
|
|
'type' => 'optional string',
|
|
|
|
'help' => pht(
|
|
|
|
'Name of a concrete subclass of ArcanistXHPASTLintNamingHook which '.
|
|
|
|
'enforces more granular naming convention rules for symbols.'),
|
2014-05-12 13:46:56 +02:00
|
|
|
),
|
|
|
|
'xhpast.switchhook' => array(
|
2014-05-12 14:06:14 +02:00
|
|
|
'type' => 'optional string',
|
|
|
|
'help' => pht(
|
|
|
|
'Name of a concrete subclass of ArcanistXHPASTLintSwitchHook which '.
|
|
|
|
'tunes the analysis of switch() statements for this linter.'),
|
2014-05-12 13:46:56 +02:00
|
|
|
),
|
2014-06-17 17:30:15 +02:00
|
|
|
'xhpast.php-version' => array(
|
|
|
|
'type' => 'optional string',
|
|
|
|
'help' => pht('PHP version to target.'),
|
|
|
|
),
|
|
|
|
'xhpast.php-version.windows' => array(
|
|
|
|
'type' => 'optional string',
|
2014-06-20 10:26:44 +02:00
|
|
|
'help' => pht('PHP version to target on Windows.'),
|
2014-06-17 17:30:15 +02:00
|
|
|
),
|
2014-05-12 13:46:56 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setLinterConfigurationValue($key, $value) {
|
|
|
|
switch ($key) {
|
|
|
|
case 'xhpast.naminghook':
|
|
|
|
$this->naminghook = $value;
|
|
|
|
return;
|
|
|
|
case 'xhpast.switchhook':
|
|
|
|
$this->switchhook = $value;
|
|
|
|
return;
|
2014-06-17 17:30:15 +02:00
|
|
|
case 'xhpast.php-version':
|
|
|
|
$this->version = $value;
|
|
|
|
return;
|
|
|
|
case 'xhpast.php-version.windows':
|
|
|
|
$this->windowsVersion = $value;
|
|
|
|
return;
|
2014-05-12 13:46:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return parent::setLinterConfigurationValue($key, $value);
|
|
|
|
}
|
|
|
|
|
2014-05-28 15:33:58 +02:00
|
|
|
public function getVersion() {
|
|
|
|
// The version number should be incremented whenever a new rule is added.
|
2014-08-05 14:35:34 +02:00
|
|
|
return '8';
|
2013-01-19 18:19:21 +01:00
|
|
|
}
|
|
|
|
|
2013-02-19 22:42:09 +01:00
|
|
|
protected function resolveFuture($path, Future $future) {
|
|
|
|
$tree = $this->getXHPASTTreeForPath($path);
|
|
|
|
if (!$tree) {
|
2014-05-12 04:28:46 +02:00
|
|
|
$ex = $this->getXHPASTExceptionForPath($path);
|
|
|
|
if ($ex instanceof XHPASTSyntaxErrorException) {
|
|
|
|
$this->raiseLintAtLine(
|
|
|
|
$ex->getErrorLine(),
|
|
|
|
1,
|
|
|
|
self::LINT_PHP_SYNTAX_ERROR,
|
|
|
|
'This file contains a syntax error: '.$ex->getMessage());
|
|
|
|
} else if ($ex instanceof Exception) {
|
|
|
|
$this->raiseLintAtPath(self::LINT_UNABLE_TO_PARSE, $ex->getMessage());
|
|
|
|
}
|
2011-01-10 00:22:25 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-02-19 22:42:09 +01:00
|
|
|
$root = $tree->getRootNode();
|
2011-01-10 00:22:25 +01:00
|
|
|
|
2013-02-15 00:18:39 +01:00
|
|
|
$method_codes = array(
|
|
|
|
'lintStrstrUsedForCheck' => self::LINT_SLOWNESS,
|
|
|
|
'lintStrposUsedForStart' => self::LINT_SLOWNESS,
|
|
|
|
'lintImplicitFallthrough' => self::LINT_IMPLICIT_FALLTHROUGH,
|
|
|
|
'lintBraceFormatting' => self::LINT_BRACE_FORMATTING,
|
|
|
|
'lintTautologicalExpressions' => self::LINT_TAUTOLOGICAL_EXPRESSION,
|
|
|
|
'lintCommentSpaces' => self::LINT_COMMENT_SPACING,
|
|
|
|
'lintHashComments' => self::LINT_COMMENT_STYLE,
|
|
|
|
'lintReusedIterators' => self::LINT_REUSED_ITERATORS,
|
2013-03-05 16:45:02 +01:00
|
|
|
'lintReusedIteratorReferences' => self::LINT_REUSED_ITERATOR_REFERENCE,
|
2013-02-15 00:18:39 +01:00
|
|
|
'lintVariableVariables' => self::LINT_VARIABLE_VARIABLE,
|
|
|
|
'lintUndeclaredVariables' => array(
|
|
|
|
self::LINT_EXTRACT_USE,
|
|
|
|
self::LINT_REUSED_AS_ITERATOR,
|
|
|
|
self::LINT_UNDECLARED_VARIABLE,
|
|
|
|
),
|
|
|
|
'lintPHPTagUse' => array(
|
|
|
|
self::LINT_PHP_SHORT_TAG,
|
|
|
|
self::LINT_PHP_ECHO_TAG,
|
|
|
|
self::LINT_PHP_OPEN_TAG,
|
|
|
|
self::LINT_PHP_CLOSE_TAG,
|
|
|
|
),
|
|
|
|
'lintNamingConventions' => self::LINT_NAMING_CONVENTIONS,
|
|
|
|
'lintSurpriseConstructors' => self::LINT_IMPLICIT_CONSTRUCTOR,
|
|
|
|
'lintParenthesesShouldHugExpressions' => self::LINT_PARENTHESES_SPACING,
|
|
|
|
'lintSpaceAfterControlStatementKeywords' =>
|
|
|
|
self::LINT_CONTROL_STATEMENT_SPACING,
|
|
|
|
'lintSpaceAroundBinaryOperators' => self::LINT_BINARY_EXPRESSION_SPACING,
|
|
|
|
'lintDynamicDefines' => self::LINT_DYNAMIC_DEFINE,
|
|
|
|
'lintUseOfThisInStaticMethods' => self::LINT_STATIC_THIS,
|
|
|
|
'lintPregQuote' => self::LINT_PREG_QUOTE_MISUSE,
|
|
|
|
'lintExitExpressions' => self::LINT_EXIT_EXPRESSION,
|
|
|
|
'lintArrayIndexWhitespace' => self::LINT_ARRAY_INDEX_SPACING,
|
|
|
|
'lintTODOComments' => self::LINT_TODO_COMMENT,
|
|
|
|
'lintPrimaryDeclarationFilenameMatch' =>
|
|
|
|
self::LINT_CLASS_FILENAME_MISMATCH,
|
|
|
|
'lintPlusOperatorOnStrings' => self::LINT_PLUS_OPERATOR_ON_STRINGS,
|
|
|
|
'lintDuplicateKeysInArray' => self::LINT_DUPLICATE_KEYS_IN_ARRAY,
|
2013-02-19 22:50:13 +01:00
|
|
|
'lintClosingCallParen' => self::LINT_CLOSING_CALL_PAREN,
|
|
|
|
'lintClosingDeclarationParen' => self::LINT_CLOSING_DECL_PAREN,
|
2013-04-15 19:22:56 +02:00
|
|
|
'lintKeywordCasing' => self::LINT_KEYWORD_CASING,
|
2014-05-17 04:25:37 +02:00
|
|
|
'lintStrings' => self::LINT_DOUBLE_QUOTE,
|
2014-05-19 15:33:22 +02:00
|
|
|
'lintElseIfStatements' => self::LINT_ELSEIF_USAGE,
|
2014-05-19 16:48:58 +02:00
|
|
|
'lintSemicolons' => self::LINT_SEMICOLON_SPACING,
|
2014-05-28 01:16:39 +02:00
|
|
|
'lintSpaceAroundConcatenationOperators' =>
|
|
|
|
self::LINT_CONCATENATION_OPERATOR,
|
2014-06-17 17:30:15 +02:00
|
|
|
'lintPHPCompatibility' => self::LINT_PHP_COMPATIBILITY,
|
2014-08-05 14:35:34 +02:00
|
|
|
'lintLanguageConstructParentheses' => self::LINT_LANGUAGE_CONSTRUCT_PAREN,
|
2014-09-09 14:48:45 +02:00
|
|
|
'lintEmptyBlockStatements' => self::LINT_EMPTY_STATEMENT,
|
2013-02-15 00:18:39 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
foreach ($method_codes as $method => $codes) {
|
|
|
|
foreach ((array)$codes as $code) {
|
|
|
|
if ($this->isCodeEnabled($code)) {
|
|
|
|
call_user_func(array($this, $method), $root);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-15 22:30:40 +02:00
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintStrstrUsedForCheck(XHPASTNode $root) {
|
2012-08-15 22:30:40 +02:00
|
|
|
$expressions = $root->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
foreach ($expressions as $expression) {
|
|
|
|
$operator = $expression->getChildOfType(1, 'n_OPERATOR');
|
|
|
|
$operator = $operator->getConcreteString();
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($operator !== '===' && $operator !== '!==') {
|
2012-08-15 22:30:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$false = $expression->getChildByIndex(0);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($false->getTypeName() === 'n_SYMBOL_NAME' &&
|
|
|
|
$false->getConcreteString() === 'false') {
|
2012-08-15 22:30:40 +02:00
|
|
|
$strstr = $expression->getChildByIndex(2);
|
|
|
|
} else {
|
|
|
|
$strstr = $false;
|
|
|
|
$false = $expression->getChildByIndex(2);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($false->getTypeName() !== 'n_SYMBOL_NAME' ||
|
|
|
|
$false->getConcreteString() !== 'false') {
|
2012-08-15 22:30:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($strstr->getTypeName() !== 'n_FUNCTION_CALL') {
|
2012-08-15 22:30:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$name = strtolower($strstr->getChildByIndex(0)->getConcreteString());
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name === 'strstr' || $name === 'strchr') {
|
2012-08-15 22:30:40 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$strstr,
|
|
|
|
self::LINT_SLOWNESS,
|
2014-05-23 22:53:05 +02:00
|
|
|
'Use strpos() for checking if the string contains something.');
|
2014-07-11 17:21:21 +02:00
|
|
|
} else if ($name === 'stristr') {
|
2012-08-15 22:30:40 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$strstr,
|
|
|
|
self::LINT_SLOWNESS,
|
2014-05-23 22:53:05 +02:00
|
|
|
'Use stripos() for checking if the string contains something.');
|
2012-08-15 22:30:40 +02:00
|
|
|
}
|
|
|
|
}
|
2012-08-15 21:56:36 +02:00
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintStrposUsedForStart(XHPASTNode $root) {
|
2012-08-15 21:56:36 +02:00
|
|
|
$expressions = $root->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
foreach ($expressions as $expression) {
|
|
|
|
$operator = $expression->getChildOfType(1, 'n_OPERATOR');
|
|
|
|
$operator = $operator->getConcreteString();
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($operator !== '===' && $operator !== '!==') {
|
2012-08-15 21:56:36 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$zero = $expression->getChildByIndex(0);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($zero->getTypeName() === 'n_NUMERIC_SCALAR' &&
|
|
|
|
$zero->getConcreteString() === '0') {
|
2012-08-15 21:56:36 +02:00
|
|
|
$strpos = $expression->getChildByIndex(2);
|
|
|
|
} else {
|
|
|
|
$strpos = $zero;
|
|
|
|
$zero = $expression->getChildByIndex(2);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($zero->getTypeName() !== 'n_NUMERIC_SCALAR' ||
|
|
|
|
$zero->getConcreteString() !== '0') {
|
2012-08-15 21:56:36 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($strpos->getTypeName() !== 'n_FUNCTION_CALL') {
|
2012-08-15 21:56:36 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$name = strtolower($strpos->getChildByIndex(0)->getConcreteString());
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name === 'strpos') {
|
2012-08-15 21:56:36 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$strpos,
|
|
|
|
self::LINT_SLOWNESS,
|
2014-05-23 22:53:05 +02:00
|
|
|
'Use strncmp() for checking if the string starts with something.');
|
2014-07-11 17:21:21 +02:00
|
|
|
} else if ($name === 'stripos') {
|
2012-08-15 21:56:36 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$strpos,
|
|
|
|
self::LINT_SLOWNESS,
|
2014-05-23 22:53:05 +02:00
|
|
|
'Use strncasecmp() for checking if the string starts with '.
|
|
|
|
'something.');
|
2012-08-15 21:56:36 +02:00
|
|
|
}
|
|
|
|
}
|
2012-06-22 02:41:26 +02:00
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintPHPCompatibility(XHPASTNode $root) {
|
|
|
|
if (!$this->version) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-07-05 09:01:36 +02:00
|
|
|
$target = phutil_get_library_root('phutil').
|
2014-06-17 17:30:15 +02:00
|
|
|
'/../resources/php_compat_info.json';
|
2014-06-22 17:56:44 +02:00
|
|
|
$compat_info = phutil_json_decode(Filesystem::readFile($target));
|
2014-06-17 17:30:15 +02:00
|
|
|
|
|
|
|
$calls = $root->selectDescendantsOfType('n_FUNCTION_CALL');
|
|
|
|
foreach ($calls as $call) {
|
|
|
|
$node = $call->getChildByIndex(0);
|
|
|
|
$name = $node->getConcreteString();
|
|
|
|
$version = idx($compat_info['functions'], $name);
|
|
|
|
|
|
|
|
if ($version && version_compare($version['min'], $this->version, '>')) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$node,
|
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but `{$name}()` was ".
|
|
|
|
"not introduced until PHP {$version['min']}.");
|
|
|
|
} else if (array_key_exists($name, $compat_info['params'])) {
|
|
|
|
$params = $call->getChildOfType(1, 'n_CALL_PARAMETER_LIST');
|
|
|
|
foreach (array_values($params->getChildren()) as $i => $param) {
|
|
|
|
$version = idx($compat_info['params'][$name], $i);
|
|
|
|
if ($version && version_compare($version, $this->version, '>')) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$param,
|
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but parameter ".
|
|
|
|
($i + 1)." of `{$name}()` was not introduced until PHP ".
|
|
|
|
"{$version}.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->windowsVersion) {
|
|
|
|
$windows = idx($compat_info['functions_windows'], $name);
|
|
|
|
|
|
|
|
if ($windows === false) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$node,
|
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->windowsVersion} on Windows, ".
|
|
|
|
"but `{$name}()` is not available there.");
|
|
|
|
} else if (version_compare($windows, $this->windowsVersion, '>')) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$node,
|
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->windowsVersion} on Windows, ".
|
|
|
|
"but `{$name}()` is not available there until PHP ".
|
|
|
|
"{$this->windowsVersion}.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$classes = $root->selectDescendantsOfType('n_CLASS_NAME');
|
|
|
|
foreach ($classes as $node) {
|
|
|
|
$name = $node->getConcreteString();
|
|
|
|
$version = idx($compat_info['interfaces'], $name);
|
|
|
|
$version = idx($compat_info['classes'], $name, $version);
|
|
|
|
if ($version && version_compare($version['min'], $this->version, '>')) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$node,
|
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but `{$name}` was not ".
|
|
|
|
"introduced until PHP {$version['min']}.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Technically, this will include function names. This is unlikely to
|
|
|
|
// cause any issues (unless, of course, there existed a function that had
|
|
|
|
// the same name as some constant).
|
|
|
|
$constants = $root->selectDescendantsOfType('n_SYMBOL_NAME');
|
|
|
|
foreach ($constants as $node) {
|
|
|
|
$name = $node->getConcreteString();
|
|
|
|
$version = idx($compat_info['constants'], $name);
|
|
|
|
if ($version && version_compare($version['min'], $this->version, '>')) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$node,
|
2014-06-22 17:56:57 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
2014-06-17 17:30:15 +02:00
|
|
|
"This codebase targets PHP {$this->version}, but `{$name}` was not ".
|
|
|
|
"introduced until PHP {$version['min']}.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (version_compare($this->version, '5.3.0') < 0) {
|
|
|
|
$this->lintPHP53Features($root);
|
|
|
|
}
|
2012-03-09 22:51:02 +01:00
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
if (version_compare($this->version, '5.4.0') < 0) {
|
|
|
|
$this->lintPHP54Features($root);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function lintPHP53Features(XHPASTNode $root) {
|
2012-03-09 22:51:02 +01:00
|
|
|
$functions = $root->selectTokensOfType('T_FUNCTION');
|
|
|
|
foreach ($functions as $function) {
|
|
|
|
$next = $function->getNextToken();
|
|
|
|
while ($next) {
|
|
|
|
if ($next->isSemantic()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
$next = $next->getNextToken();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($next) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($next->getTypeName() === '(') {
|
2012-03-09 22:51:02 +01:00
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$function,
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but anonymous ".
|
|
|
|
"functions were not introduced until PHP 5.3.");
|
2012-03-09 22:51:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$namespaces = $root->selectTokensOfType('T_NAMESPACE');
|
|
|
|
foreach ($namespaces as $namespace) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$namespace,
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but namespaces were not ".
|
|
|
|
"introduced until PHP 5.3.");
|
2012-03-09 22:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: This is only "use x;", in anonymous functions the node type is
|
|
|
|
// n_LEXICAL_VARIABLE_LIST even though both tokens are T_USE.
|
|
|
|
|
|
|
|
// TODO: We parse n_USE in a slightly crazy way right now; that would be
|
|
|
|
// a better selector once it's fixed.
|
|
|
|
|
|
|
|
$uses = $root->selectDescendantsOfType('n_USE_LIST');
|
|
|
|
foreach ($uses as $use) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$use,
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but namespaces were not ".
|
|
|
|
"introduced until PHP 5.3.");
|
2012-03-09 22:51:02 +01:00
|
|
|
}
|
2012-06-17 18:01:47 +02:00
|
|
|
|
|
|
|
$statics = $root->selectDescendantsOfType('n_CLASS_STATIC_ACCESS');
|
|
|
|
foreach ($statics as $static) {
|
2013-02-01 20:17:46 +01:00
|
|
|
$name = $static->getChildByIndex(0);
|
|
|
|
if ($name->getTypeName() != 'n_CLASS_NAME') {
|
|
|
|
continue;
|
|
|
|
}
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name->getConcreteString() === 'static') {
|
2012-06-17 18:01:47 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$name,
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but `static::` was not ".
|
|
|
|
"introduced until PHP 5.3.");
|
2012-06-17 18:01:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-13 20:19:25 +02:00
|
|
|
$ternaries = $root->selectDescendantsOfType('n_TERNARY_EXPRESSION');
|
|
|
|
foreach ($ternaries as $ternary) {
|
|
|
|
$yes = $ternary->getChildByIndex(1);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($yes->getTypeName() === 'n_EMPTY') {
|
2012-09-13 20:19:25 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$ternary,
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but short ternary was ".
|
|
|
|
"not introduced until PHP 5.3.");
|
2012-09-13 20:19:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-19 18:19:21 +01:00
|
|
|
$heredocs = $root->selectDescendantsOfType('n_HEREDOC');
|
|
|
|
foreach ($heredocs as $heredoc) {
|
|
|
|
if (preg_match('/^<<<[\'"]/', $heredoc->getConcreteString())) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$heredoc,
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
|
|
|
"This codebase targets PHP {$this->version}, but nowdoc was not ".
|
|
|
|
"introduced until PHP 5.3.");
|
2014-06-16 17:00:23 +02:00
|
|
|
}
|
|
|
|
}
|
2012-03-08 21:20:19 +01:00
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintPHP54Features(XHPASTNode $root) {
|
2012-08-15 13:36:50 +02:00
|
|
|
$indexes = $root->selectDescendantsOfType('n_INDEX_ACCESS');
|
|
|
|
foreach ($indexes as $index) {
|
|
|
|
$left = $index->getChildByIndex(0);
|
|
|
|
switch ($left->getTypeName()) {
|
|
|
|
case 'n_FUNCTION_CALL':
|
2013-02-12 16:07:35 +01:00
|
|
|
case 'n_METHOD_CALL':
|
2012-08-15 13:36:50 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$index->getChildByIndex(1),
|
2014-06-17 17:30:15 +02:00
|
|
|
self::LINT_PHP_COMPATIBILITY,
|
2014-05-23 22:53:05 +02:00
|
|
|
'The f()[...] syntax was not introduced until PHP 5.4, but this '.
|
|
|
|
'codebase targets an earlier version of PHP. You can rewrite '.
|
|
|
|
'this expression using idx().');
|
2012-08-15 13:36:50 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintImplicitFallthrough(XHPASTNode $root) {
|
2013-02-05 20:00:07 +01:00
|
|
|
$hook_obj = null;
|
|
|
|
$working_copy = $this->getEngine()->getWorkingCopy();
|
|
|
|
if ($working_copy) {
|
2014-05-12 13:46:56 +02:00
|
|
|
$hook_class = $this->switchhook
|
|
|
|
? $this->switchhook
|
|
|
|
: $this->getDeprecatedConfiguration('lint.xhpast.switchhook');
|
2013-02-05 20:00:07 +01:00
|
|
|
if ($hook_class) {
|
|
|
|
$hook_obj = newv($hook_class, array());
|
|
|
|
assert_instances_of(array($hook_obj), 'ArcanistXHPASTLintSwitchHook');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-08 21:20:19 +01:00
|
|
|
$switches = $root->selectDescendantsOfType('n_SWITCH');
|
|
|
|
foreach ($switches as $switch) {
|
|
|
|
$blocks = array();
|
|
|
|
|
|
|
|
$cases = $switch->selectDescendantsOfType('n_CASE');
|
|
|
|
foreach ($cases as $case) {
|
|
|
|
$blocks[] = $case;
|
|
|
|
}
|
|
|
|
|
|
|
|
$defaults = $switch->selectDescendantsOfType('n_DEFAULT');
|
|
|
|
foreach ($defaults as $default) {
|
|
|
|
$blocks[] = $default;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
foreach ($blocks as $key => $block) {
|
2012-09-08 02:46:35 +02:00
|
|
|
// Collect all the tokens in this block which aren't at top level.
|
|
|
|
// We want to ignore "break", and "continue" in these blocks.
|
|
|
|
$lower_level = $block->selectDescendantsOfType('n_WHILE');
|
|
|
|
$lower_level->add($block->selectDescendantsOfType('n_DO_WHILE'));
|
|
|
|
$lower_level->add($block->selectDescendantsOfType('n_FOR'));
|
|
|
|
$lower_level->add($block->selectDescendantsOfType('n_FOREACH'));
|
|
|
|
$lower_level->add($block->selectDescendantsOfType('n_SWITCH'));
|
|
|
|
$lower_level_tokens = array();
|
|
|
|
foreach ($lower_level as $lower_level_block) {
|
|
|
|
$lower_level_tokens += $lower_level_block->getTokens();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Collect all the tokens in this block which aren't in this scope
|
|
|
|
// (because they're inside class, function or interface declarations).
|
|
|
|
// We want to ignore all of these tokens.
|
|
|
|
$decls = $block->selectDescendantsOfType('n_FUNCTION_DECLARATION');
|
|
|
|
$decls->add($block->selectDescendantsOfType('n_CLASS_DECLARATION'));
|
|
|
|
// For completeness; these can't actually have anything.
|
|
|
|
$decls->add($block->selectDescendantsOfType('n_INTERFACE_DECLARATION'));
|
|
|
|
$different_scope_tokens = array();
|
|
|
|
foreach ($decls as $decl) {
|
|
|
|
$different_scope_tokens += $decl->getTokens();
|
|
|
|
}
|
|
|
|
|
|
|
|
$lower_level_tokens += $different_scope_tokens;
|
2012-03-08 21:20:19 +01:00
|
|
|
|
|
|
|
// Get all the trailing nonsemantic tokens, since we need to look for
|
|
|
|
// "fallthrough" comments past the end of the semantic block.
|
|
|
|
|
2012-09-08 02:46:35 +02:00
|
|
|
$tokens = $block->getTokens();
|
2012-03-08 21:20:19 +01:00
|
|
|
$last = end($tokens);
|
|
|
|
while ($last && $last = $last->getNextToken()) {
|
2013-02-22 22:49:16 +01:00
|
|
|
if ($last->isSemantic()) {
|
|
|
|
break;
|
2012-03-08 21:20:19 +01:00
|
|
|
}
|
2013-02-22 22:49:16 +01:00
|
|
|
$tokens[$last->getTokenID()] = $last;
|
2012-03-08 21:20:19 +01:00
|
|
|
}
|
|
|
|
|
2012-09-08 02:46:35 +02:00
|
|
|
$blocks[$key] = array(
|
|
|
|
$tokens,
|
|
|
|
$lower_level_tokens,
|
|
|
|
$different_scope_tokens,
|
|
|
|
);
|
2012-03-08 21:20:19 +01:00
|
|
|
}
|
|
|
|
|
2012-09-08 02:46:35 +02:00
|
|
|
foreach ($blocks as $token_lists) {
|
|
|
|
list(
|
|
|
|
$tokens,
|
|
|
|
$lower_level_tokens,
|
|
|
|
$different_scope_tokens) = $token_lists;
|
2012-03-08 21:20:19 +01:00
|
|
|
|
|
|
|
// Test each block (case or default statement) to see if it's OK. It's
|
|
|
|
// OK if:
|
|
|
|
//
|
|
|
|
// - it is empty; or
|
2012-09-08 02:46:35 +02:00
|
|
|
// - it ends in break, return, throw, continue or exit at top level; or
|
2012-03-08 21:20:19 +01:00
|
|
|
// - it has a comment with "fallthrough" in its text.
|
|
|
|
|
|
|
|
// Empty blocks are OK, so we start this at `true` and only set it to
|
|
|
|
// false if we find a statement.
|
|
|
|
$block_ok = true;
|
|
|
|
|
|
|
|
// Keeps track of whether the current statement is one that validates
|
|
|
|
// the block (break, return, throw, continue) or something else.
|
|
|
|
$statement_ok = false;
|
|
|
|
|
2012-09-08 02:46:35 +02:00
|
|
|
foreach ($tokens as $token_id => $token) {
|
2012-03-08 21:20:19 +01:00
|
|
|
if (!$token->isSemantic()) {
|
|
|
|
// Liberally match "fall" in the comment text so that comments like
|
|
|
|
// "fallthru", "fall through", "fallthrough", etc., are accepted.
|
|
|
|
if (preg_match('/fall/i', $token->getValue())) {
|
|
|
|
$block_ok = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$tok_type = $token->getTypeName();
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($tok_type === 'T_FUNCTION' ||
|
|
|
|
$tok_type === 'T_CLASS' ||
|
|
|
|
$tok_type === 'T_INTERFACE') {
|
2012-09-08 02:46:35 +02:00
|
|
|
// These aren't statements, but mark the block as nonempty anyway.
|
|
|
|
$block_ok = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($tok_type === ';') {
|
2012-03-08 21:20:19 +01:00
|
|
|
if ($statement_ok) {
|
|
|
|
$statment_ok = false;
|
|
|
|
} else {
|
|
|
|
$block_ok = false;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($tok_type === 'T_BREAK' ||
|
|
|
|
$tok_type === 'T_CONTINUE') {
|
2012-09-08 02:46:35 +02:00
|
|
|
if (empty($lower_level_tokens[$token_id])) {
|
|
|
|
$statement_ok = true;
|
|
|
|
$block_ok = true;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($tok_type === 'T_RETURN' ||
|
|
|
|
$tok_type === 'T_THROW' ||
|
|
|
|
$tok_type === 'T_EXIT' ||
|
2013-02-05 20:00:07 +01:00
|
|
|
($hook_obj && $hook_obj->checkSwitchToken($token))) {
|
2012-09-08 02:46:35 +02:00
|
|
|
if (empty($different_scope_tokens[$token_id])) {
|
|
|
|
$statement_ok = true;
|
|
|
|
$block_ok = true;
|
|
|
|
}
|
|
|
|
continue;
|
2012-03-08 21:20:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$block_ok) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
head($tokens),
|
|
|
|
self::LINT_IMPLICIT_FALLTHROUGH,
|
|
|
|
"This 'case' or 'default' has a nonempty block which does not ".
|
|
|
|
"end with 'break', 'continue', 'return', 'throw' or 'exit'. Did ".
|
|
|
|
"you forget to add one of those? If you intend to fall through, ".
|
|
|
|
"add a '// fallthrough' comment to silence this warning.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-08-01 22:18:53 +02:00
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintBraceFormatting(XHPASTNode $root) {
|
2011-08-01 22:18:53 +02:00
|
|
|
foreach ($root->selectDescendantsOfType('n_STATEMENT_LIST') as $list) {
|
|
|
|
$tokens = $list->getTokens();
|
|
|
|
if (!$tokens || head($tokens)->getValue() != '{') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
list($before, $after) = $list->getSurroundingNonsemanticTokens();
|
2012-01-04 20:15:33 +01:00
|
|
|
if (!$before) {
|
|
|
|
$first = head($tokens);
|
|
|
|
|
|
|
|
// Only insert the space if we're after a closing parenthesis. If
|
|
|
|
// we're in a construct like "else{}", other rules will insert space
|
|
|
|
// after the 'else' correctly.
|
|
|
|
$prev = $first->getPrevToken();
|
2014-07-11 17:21:21 +02:00
|
|
|
if (!$prev || $prev->getValue() !== ')') {
|
2012-01-04 20:15:33 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$first,
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_BRACE_FORMATTING,
|
2012-01-04 20:15:33 +01:00
|
|
|
'Put opening braces on the same line as control statements and '.
|
|
|
|
'declarations, with a single space before them.',
|
|
|
|
' '.$first->getValue());
|
2014-07-11 17:21:21 +02:00
|
|
|
} else if (count($before) === 1) {
|
2011-08-01 22:18:53 +02:00
|
|
|
$before = reset($before);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($before->getValue() !== ' ') {
|
2011-08-01 22:18:53 +02:00
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$before,
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_BRACE_FORMATTING,
|
2011-08-01 22:18:53 +02:00
|
|
|
'Put opening braces on the same line as control statements and '.
|
|
|
|
'declarations, with a single space before them.',
|
|
|
|
' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-02-06 22:04:01 +01:00
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintTautologicalExpressions(XHPASTNode $root) {
|
2011-02-06 22:04:01 +01:00
|
|
|
$expressions = $root->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
|
|
|
|
static $operators = array(
|
|
|
|
'-' => true,
|
|
|
|
'/' => true,
|
|
|
|
'-=' => true,
|
|
|
|
'/=' => true,
|
|
|
|
'<=' => true,
|
|
|
|
'<' => true,
|
|
|
|
'==' => true,
|
|
|
|
'===' => true,
|
2011-02-14 01:43:32 +01:00
|
|
|
'!=' => true,
|
|
|
|
'!==' => true,
|
2011-02-06 22:04:01 +01:00
|
|
|
'>=' => true,
|
|
|
|
'>' => true,
|
|
|
|
);
|
|
|
|
|
2011-06-13 17:58:01 +02:00
|
|
|
static $logical = array(
|
|
|
|
'||' => true,
|
|
|
|
'&&' => true,
|
|
|
|
);
|
|
|
|
|
2011-02-06 22:04:01 +01:00
|
|
|
foreach ($expressions as $expr) {
|
|
|
|
$operator = $expr->getChildByIndex(1)->getConcreteString();
|
2011-06-13 17:58:01 +02:00
|
|
|
if (!empty($operators[$operator])) {
|
|
|
|
$left = $expr->getChildByIndex(0)->getSemanticString();
|
|
|
|
$right = $expr->getChildByIndex(2)->getSemanticString();
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($left === $right) {
|
2011-06-13 17:58:01 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$expr,
|
|
|
|
self::LINT_TAUTOLOGICAL_EXPRESSION,
|
|
|
|
'Both sides of this expression are identical, so it always '.
|
|
|
|
'evaluates to a constant.');
|
|
|
|
}
|
2011-02-06 22:04:01 +01:00
|
|
|
}
|
|
|
|
|
2011-06-13 17:58:01 +02:00
|
|
|
if (!empty($logical[$operator])) {
|
|
|
|
$left = $expr->getChildByIndex(0)->getSemanticString();
|
|
|
|
$right = $expr->getChildByIndex(2)->getSemanticString();
|
|
|
|
|
|
|
|
// NOTE: These will be null to indicate "could not evaluate".
|
|
|
|
$left = $this->evaluateStaticBoolean($left);
|
|
|
|
$right = $this->evaluateStaticBoolean($right);
|
2011-02-06 22:04:01 +01:00
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if (($operator === '||' && ($left === true || $right === true)) ||
|
|
|
|
($operator === '&&' && ($left === false || $right === false))) {
|
2011-06-13 17:58:01 +02:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$expr,
|
|
|
|
self::LINT_TAUTOLOGICAL_EXPRESSION,
|
|
|
|
'The logical value of this expression is static. Did you forget '.
|
|
|
|
'to remove some debugging code?');
|
|
|
|
}
|
2011-02-06 22:04:01 +01:00
|
|
|
}
|
|
|
|
}
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
2011-06-13 17:58:01 +02:00
|
|
|
/**
|
|
|
|
* Statically evaluate a boolean value from an XHP tree.
|
|
|
|
*
|
|
|
|
* TODO: Improve this and move it to XHPAST proper?
|
|
|
|
*
|
|
|
|
* @param string The "semantic string" of a single value.
|
|
|
|
* @return mixed ##true## or ##false## if the value could be evaluated
|
|
|
|
* statically; ##null## if static evaluation was not possible.
|
|
|
|
*/
|
|
|
|
private function evaluateStaticBoolean($string) {
|
|
|
|
switch (strtolower($string)) {
|
|
|
|
case '0':
|
|
|
|
case 'null':
|
|
|
|
case 'false':
|
|
|
|
return false;
|
|
|
|
case '1':
|
|
|
|
case 'true':
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
protected function lintCommentSpaces(XHPASTNode $root) {
|
2012-06-22 01:21:04 +02:00
|
|
|
foreach ($root->selectTokensOfType('T_COMMENT') as $comment) {
|
|
|
|
$value = $comment->getValue();
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($value[0] !== '#') {
|
2012-06-22 01:21:04 +02:00
|
|
|
$match = null;
|
|
|
|
if (preg_match('@^(/[/*]+)[^/*\s]@', $value, $match)) {
|
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
$comment->getOffset(),
|
|
|
|
self::LINT_COMMENT_SPACING,
|
|
|
|
'Put space after comment start.',
|
|
|
|
$match[1],
|
|
|
|
$match[1].' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
protected function lintHashComments(XHPASTNode $root) {
|
2012-03-08 21:20:46 +01:00
|
|
|
foreach ($root->selectTokensOfType('T_COMMENT') as $comment) {
|
|
|
|
$value = $comment->getValue();
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($value[0] !== '#') {
|
2012-03-08 21:20:46 +01:00
|
|
|
continue;
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
2012-03-08 21:20:46 +01:00
|
|
|
|
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
$comment->getOffset(),
|
|
|
|
self::LINT_COMMENT_STYLE,
|
|
|
|
'Use "//" single-line comments, not "#".',
|
|
|
|
'#',
|
2012-06-22 01:21:04 +02:00
|
|
|
(preg_match('/^#\S/', $value) ? '// ' : '//'));
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-21 02:59:20 +02:00
|
|
|
/**
|
|
|
|
* Find cases where loops get nested inside each other but use the same
|
|
|
|
* iterator variable. For example:
|
|
|
|
*
|
|
|
|
* COUNTEREXAMPLE
|
|
|
|
* foreach ($list as $thing) {
|
|
|
|
* foreach ($stuff as $thing) { // <-- Raises an error for reuse of $thing
|
|
|
|
* // ...
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
*/
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintReusedIterators(XHPASTNode $root) {
|
2011-06-21 02:59:20 +02:00
|
|
|
$used_vars = array();
|
|
|
|
|
|
|
|
$for_loops = $root->selectDescendantsOfType('n_FOR');
|
|
|
|
foreach ($for_loops as $for_loop) {
|
|
|
|
$var_map = array();
|
|
|
|
|
|
|
|
// Find all the variables that are assigned to in the for() expression.
|
|
|
|
$for_expr = $for_loop->getChildOfType(0, 'n_FOR_EXPRESSION');
|
|
|
|
$bin_exprs = $for_expr->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
foreach ($bin_exprs as $bin_expr) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($bin_expr->getChildByIndex(1)->getConcreteString() === '=') {
|
2013-02-09 00:04:32 +01:00
|
|
|
$var = $bin_expr->getChildByIndex(0);
|
|
|
|
$var_map[$var->getConcreteString()] = $var;
|
2011-06-21 02:59:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$used_vars[$for_loop->getID()] = $var_map;
|
|
|
|
}
|
|
|
|
|
|
|
|
$foreach_loops = $root->selectDescendantsOfType('n_FOREACH');
|
|
|
|
foreach ($foreach_loops as $foreach_loop) {
|
|
|
|
$var_map = array();
|
|
|
|
|
|
|
|
$foreach_expr = $foreach_loop->getChildOftype(0, 'n_FOREACH_EXPRESSION');
|
|
|
|
|
|
|
|
// We might use one or two vars, i.e. "foreach ($x as $y => $z)" or
|
|
|
|
// "foreach ($x as $y)".
|
|
|
|
$possible_used_vars = array(
|
|
|
|
$foreach_expr->getChildByIndex(1),
|
|
|
|
$foreach_expr->getChildByIndex(2),
|
|
|
|
);
|
|
|
|
foreach ($possible_used_vars as $var) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($var->getTypeName() === 'n_EMPTY') {
|
2011-06-21 02:59:20 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$name = $var->getConcreteString();
|
|
|
|
$name = trim($name, '&'); // Get rid of ref silliness.
|
2013-02-09 00:04:32 +01:00
|
|
|
$var_map[$name] = $var;
|
2011-06-21 02:59:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$used_vars[$foreach_loop->getID()] = $var_map;
|
|
|
|
}
|
|
|
|
|
|
|
|
$all_loops = $for_loops->add($foreach_loops);
|
|
|
|
foreach ($all_loops as $loop) {
|
|
|
|
$child_for_loops = $loop->selectDescendantsOfType('n_FOR');
|
|
|
|
$child_foreach_loops = $loop->selectDescendantsOfType('n_FOREACH');
|
|
|
|
$child_loops = $child_for_loops->add($child_foreach_loops);
|
|
|
|
|
|
|
|
$outer_vars = $used_vars[$loop->getID()];
|
|
|
|
foreach ($child_loops as $inner_loop) {
|
|
|
|
$inner_vars = $used_vars[$inner_loop->getID()];
|
|
|
|
$shared = array_intersect_key($outer_vars, $inner_vars);
|
|
|
|
if ($shared) {
|
|
|
|
$shared_desc = implode(', ', array_keys($shared));
|
2013-02-09 00:04:32 +01:00
|
|
|
$message = $this->raiseLintAtNode(
|
2011-06-21 02:59:20 +02:00
|
|
|
$inner_loop->getChildByIndex(0),
|
|
|
|
self::LINT_REUSED_ITERATORS,
|
|
|
|
"This loop reuses iterator variables ({$shared_desc}) from an ".
|
|
|
|
"outer loop. You might be clobbering the outer iterator. Change ".
|
|
|
|
"the inner loop to use a different iterator name.");
|
2013-02-09 00:04:32 +01:00
|
|
|
|
|
|
|
$locations = array();
|
|
|
|
foreach ($shared as $var) {
|
|
|
|
$locations[] = $this->getOtherLocation($var->getOffset());
|
|
|
|
}
|
|
|
|
$message->setOtherLocations($locations);
|
2011-06-21 02:59:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 16:45:02 +01:00
|
|
|
/**
|
|
|
|
* Find cases where a foreach loop is being iterated using a variable
|
|
|
|
* reference and the same variable is used outside of the loop without
|
|
|
|
* calling unset() or reassigning the variable to another variable
|
|
|
|
* reference.
|
|
|
|
*
|
|
|
|
* COUNTEREXAMPLE
|
|
|
|
* foreach ($ar as &$a) {
|
|
|
|
* // ...
|
|
|
|
* }
|
|
|
|
* $a = 1; // <-- Raises an error for using $a
|
|
|
|
*
|
|
|
|
*/
|
2014-03-04 20:02:18 +01:00
|
|
|
protected function lintReusedIteratorReferences(XHPASTNode $root) {
|
2013-03-05 16:45:02 +01:00
|
|
|
|
|
|
|
$fdefs = $root->selectDescendantsOfType('n_FUNCTION_DECLARATION');
|
|
|
|
$mdefs = $root->selectDescendantsOfType('n_METHOD_DECLARATION');
|
|
|
|
$defs = $fdefs->add($mdefs);
|
|
|
|
|
|
|
|
foreach ($defs as $def) {
|
|
|
|
|
|
|
|
$body = $def->getChildByIndex(5);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($body->getTypeName() === 'n_EMPTY') {
|
2013-03-05 16:45:02 +01:00
|
|
|
// Abstract method declaration.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$exclude = array();
|
|
|
|
|
|
|
|
// Exclude uses of variables, unsets, and foreach loops
|
|
|
|
// within closures - they are checked on their own
|
|
|
|
$func_defs = $body->selectDescendantsOfType('n_FUNCTION_DECLARATION');
|
|
|
|
foreach ($func_defs as $func_def) {
|
|
|
|
$vars = $func_def->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($vars as $var) {
|
|
|
|
$exclude[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$unset_lists = $func_def->selectDescendantsOfType('n_UNSET_LIST');
|
|
|
|
foreach ($unset_lists as $unset_list) {
|
|
|
|
$exclude[$unset_list->getID()] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$foreaches = $func_def->selectDescendantsOfType('n_FOREACH');
|
|
|
|
foreach ($foreaches as $foreach) {
|
|
|
|
$exclude[$foreach->getID()] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find all variables that are unset within the scope
|
|
|
|
$unset_vars = array();
|
|
|
|
$unset_lists = $body->selectDescendantsOfType('n_UNSET_LIST');
|
|
|
|
foreach ($unset_lists as $unset_list) {
|
|
|
|
if (isset($exclude[$unset_list->getID()])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$unset_list_vars = $unset_list->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($unset_list_vars as $var) {
|
|
|
|
$concrete = $this->getConcreteVariableString($var);
|
|
|
|
$unset_vars[$concrete][] = $var->getOffset();
|
|
|
|
$exclude[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find all reference variables in foreach expressions
|
|
|
|
$reference_vars = array();
|
|
|
|
$foreaches = $body->selectDescendantsOfType('n_FOREACH');
|
|
|
|
foreach ($foreaches as $foreach) {
|
|
|
|
if (isset($exclude[$foreach->getID()])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$foreach_expr = $foreach->getChildOfType(0, 'n_FOREACH_EXPRESSION');
|
|
|
|
$var = $foreach_expr->getChildByIndex(2);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($var->getTypeName() !== 'n_VARIABLE_REFERENCE') {
|
2013-03-05 16:45:02 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$reference = $var->getChildByIndex(0);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($reference->getTypeName() !== 'n_VARIABLE') {
|
2013-03-05 16:45:02 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$reference_name = $this->getConcreteVariableString($reference);
|
|
|
|
$reference_vars[$reference_name][] = $reference->getOffset();
|
|
|
|
$exclude[$reference->getID()] = true;
|
|
|
|
|
|
|
|
// Exclude uses of the reference variable within the foreach loop
|
|
|
|
$foreach_vars = $foreach->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($foreach_vars as $var) {
|
|
|
|
$name = $this->getConcreteVariableString($var);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name === $reference_name) {
|
2013-03-05 16:45:02 +01:00
|
|
|
$exclude[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allow usage if the reference variable is assigned to another
|
|
|
|
// reference variable
|
|
|
|
$binary = $body->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
foreach ($binary as $expr) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($expr->getChildByIndex(1)->getConcreteString() !== '=') {
|
2013-03-05 16:45:02 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$lval = $expr->getChildByIndex(0);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($lval->getTypeName() !== 'n_VARIABLE') {
|
2013-03-05 16:45:02 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$rval = $expr->getChildByIndex(2);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($rval->getTypeName() !== 'n_VARIABLE_REFERENCE') {
|
2013-03-05 16:45:02 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Counts as unsetting a variable
|
|
|
|
$concrete = $this->getConcreteVariableString($lval);
|
|
|
|
$unset_vars[$concrete][] = $lval->getOffset();
|
|
|
|
$exclude[$lval->getID()] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$all_vars = array();
|
|
|
|
$all = $body->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($all as $var) {
|
|
|
|
if (isset($exclude[$var->getID()])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$name = $this->getConcreteVariableString($var);
|
|
|
|
|
|
|
|
if (!isset($reference_vars[$name])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the closest reference offset to this variable
|
|
|
|
$reference_offset = null;
|
|
|
|
foreach ($reference_vars[$name] as $offset) {
|
|
|
|
if ($offset < $var->getOffset()) {
|
|
|
|
$reference_offset = $offset;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!$reference_offset) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if an unset exists between reference and usage of this
|
|
|
|
// variable
|
|
|
|
$warn = true;
|
|
|
|
if (isset($unset_vars[$name])) {
|
|
|
|
foreach ($unset_vars[$name] as $unset_offset) {
|
|
|
|
if ($unset_offset > $reference_offset &&
|
|
|
|
$unset_offset < $var->getOffset()) {
|
|
|
|
$warn = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($warn) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$var,
|
|
|
|
self::LINT_REUSED_ITERATOR_REFERENCE,
|
2014-06-10 20:02:42 +02:00
|
|
|
'This variable was used already as a by-reference iterator '.
|
|
|
|
'variable. Such variables survive outside the foreach loop, '.
|
2013-03-05 16:45:02 +01:00
|
|
|
'do not reuse.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
protected function lintVariableVariables(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$vvars = $root->selectDescendantsOfType('n_VARIABLE_VARIABLE');
|
|
|
|
foreach ($vvars as $vvar) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$vvar,
|
|
|
|
self::LINT_VARIABLE_VARIABLE,
|
|
|
|
'Rewrite this code to use an array. Variable variables are unclear '.
|
|
|
|
'and hinder static analysis.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintUndeclaredVariables(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
// These things declare variables in a function:
|
|
|
|
// Explicit parameters
|
|
|
|
// Assignment
|
|
|
|
// Assignment via list()
|
|
|
|
// Static
|
|
|
|
// Global
|
|
|
|
// Lexical vars
|
|
|
|
// Builtins ($this)
|
|
|
|
// foreach()
|
|
|
|
// catch
|
|
|
|
//
|
|
|
|
// These things make lexical scope unknowable:
|
|
|
|
// Use of extract()
|
|
|
|
// Assignment to variable variables ($$x)
|
|
|
|
// Global with variable variables
|
|
|
|
//
|
|
|
|
// These things don't count as "using" a variable:
|
|
|
|
// isset()
|
|
|
|
// empty()
|
|
|
|
// Static class variables
|
|
|
|
//
|
|
|
|
// The general approach here is to find each function/method declaration,
|
|
|
|
// then:
|
|
|
|
//
|
|
|
|
// 1. Identify all the variable declarations, and where they first occur
|
|
|
|
// in the function/method declaration.
|
|
|
|
// 2. Identify all the uses that don't really count (as above).
|
|
|
|
// 3. Everything else must be a use of a variable.
|
|
|
|
// 4. For each variable, check if any uses occur before the declaration
|
|
|
|
// and warn about them.
|
|
|
|
//
|
|
|
|
// We also keep track of where lexical scope becomes unknowable (e.g.,
|
|
|
|
// because the function calls extract() or uses dynamic variables,
|
|
|
|
// preventing us from keeping track of which variables are defined) so we
|
|
|
|
// can stop issuing warnings after that.
|
2013-07-10 17:26:11 +02:00
|
|
|
//
|
|
|
|
// TODO: Support functions defined inside other functions which is commonly
|
|
|
|
// used with anonymous functions.
|
2011-01-10 00:22:25 +01:00
|
|
|
|
|
|
|
$fdefs = $root->selectDescendantsOfType('n_FUNCTION_DECLARATION');
|
|
|
|
$mdefs = $root->selectDescendantsOfType('n_METHOD_DECLARATION');
|
|
|
|
$defs = $fdefs->add($mdefs);
|
|
|
|
|
|
|
|
foreach ($defs as $def) {
|
|
|
|
|
|
|
|
// We keep track of the first offset where scope becomes unknowable, and
|
|
|
|
// silence any warnings after that. Default it to INT_MAX so we can min()
|
|
|
|
// it later to keep track of the first problem we encounter.
|
|
|
|
$scope_destroyed_at = PHP_INT_MAX;
|
|
|
|
|
|
|
|
$declarations = array(
|
|
|
|
'$this' => 0,
|
2012-01-28 20:17:45 +01:00
|
|
|
) + array_fill_keys($this->getSuperGlobalNames(), 0);
|
2011-01-10 00:22:25 +01:00
|
|
|
$declaration_tokens = array();
|
|
|
|
$exclude_tokens = array();
|
|
|
|
$vars = array();
|
|
|
|
|
|
|
|
// First up, find all the different kinds of declarations, as explained
|
|
|
|
// above. Put the tokens into the $vars array.
|
|
|
|
|
|
|
|
$param_list = $def->getChildOfType(3, 'n_DECLARATION_PARAMETER_LIST');
|
|
|
|
$param_vars = $param_list->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($param_vars as $var) {
|
|
|
|
$vars[] = $var;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is PHP5.3 closure syntax: function () use ($x) {};
|
|
|
|
$lexical_vars = $def
|
|
|
|
->getChildByIndex(4)
|
|
|
|
->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($lexical_vars as $var) {
|
|
|
|
$vars[] = $var;
|
|
|
|
}
|
|
|
|
|
|
|
|
$body = $def->getChildByIndex(5);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($body->getTypeName() === 'n_EMPTY') {
|
2011-01-10 00:22:25 +01:00
|
|
|
// Abstract method declaration.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$static_vars = $body
|
|
|
|
->selectDescendantsOfType('n_STATIC_DECLARATION')
|
|
|
|
->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($static_vars as $var) {
|
|
|
|
$vars[] = $var;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
$global_vars = $body
|
|
|
|
->selectDescendantsOfType('n_GLOBAL_DECLARATION_LIST');
|
|
|
|
foreach ($global_vars as $var_list) {
|
|
|
|
foreach ($var_list->getChildren() as $var) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($var->getTypeName() === 'n_VARIABLE') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$vars[] = $var;
|
|
|
|
} else {
|
|
|
|
// Dynamic global variable, i.e. "global $$x;".
|
|
|
|
$scope_destroyed_at = min($scope_destroyed_at, $var->getOffset());
|
|
|
|
// An error is raised elsewhere, no need to raise here.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-21 21:23:51 +02:00
|
|
|
// Include "catch (Exception $ex)", but not variables in the body of the
|
|
|
|
// catch block.
|
|
|
|
$catches = $body->selectDescendantsOfType('n_CATCH');
|
|
|
|
foreach ($catches as $catch) {
|
|
|
|
$vars[] = $catch->getChildOfType(1, 'n_VARIABLE');
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$binary = $body->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
foreach ($binary as $expr) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($expr->getChildByIndex(1)->getConcreteString() !== '=') {
|
2011-01-10 00:22:25 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$lval = $expr->getChildByIndex(0);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($lval->getTypeName() === 'n_VARIABLE') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$vars[] = $lval;
|
2014-07-11 17:21:21 +02:00
|
|
|
} else if ($lval->getTypeName() === 'n_LIST') {
|
2011-01-10 00:22:25 +01:00
|
|
|
// Recursivey grab everything out of list(), since the grammar
|
|
|
|
// permits list() to be nested. Also note that list() is ONLY valid
|
|
|
|
// as an lval assignments, so we could safely lift this out of the
|
|
|
|
// n_BINARY_EXPRESSION branch.
|
|
|
|
$assign_vars = $lval->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($assign_vars as $var) {
|
|
|
|
$vars[] = $var;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($lval->getTypeName() === 'n_VARIABLE_VARIABLE') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$scope_destroyed_at = min($scope_destroyed_at, $lval->getOffset());
|
|
|
|
// No need to raise here since we raise an error elsewhere.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$calls = $body->selectDescendantsOfType('n_FUNCTION_CALL');
|
|
|
|
foreach ($calls as $call) {
|
|
|
|
$name = strtolower($call->getChildByIndex(0)->getConcreteString());
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name === 'empty' || $name === 'isset') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$params = $call
|
|
|
|
->getChildOfType(1, 'n_CALL_PARAMETER_LIST')
|
|
|
|
->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($params as $var) {
|
|
|
|
$exclude_tokens[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name !== 'extract') {
|
2011-01-10 00:22:25 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$scope_destroyed_at = min($scope_destroyed_at, $call->getOffset());
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$call,
|
|
|
|
self::LINT_EXTRACT_USE,
|
|
|
|
'Avoid extract(). It is confusing and hinders static analysis.');
|
|
|
|
}
|
|
|
|
|
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
|
|
|
// Now we have every declaration except foreach(), handled below. Build
|
|
|
|
// two maps, one which just keeps track of which tokens are part of
|
|
|
|
// declarations ($declaration_tokens) and one which has the first offset
|
|
|
|
// where a variable is declared ($declarations).
|
2011-01-10 00:22:25 +01:00
|
|
|
|
|
|
|
foreach ($vars as $var) {
|
2011-05-12 03:51:25 +02:00
|
|
|
$concrete = $this->getConcreteVariableString($var);
|
2011-01-10 00:22:25 +01:00
|
|
|
$declarations[$concrete] = min(
|
|
|
|
idx($declarations, $concrete, PHP_INT_MAX),
|
|
|
|
$var->getOffset());
|
|
|
|
$declaration_tokens[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
|
2013-07-10 17:26:11 +02:00
|
|
|
// Excluded tokens are ones we don't "count" as being used, described
|
2011-01-10 00:22:25 +01:00
|
|
|
// above. Put them into $exclude_tokens.
|
|
|
|
|
|
|
|
$class_statics = $body
|
|
|
|
->selectDescendantsOfType('n_CLASS_STATIC_ACCESS');
|
|
|
|
$class_static_vars = $class_statics
|
|
|
|
->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($class_static_vars as $var) {
|
|
|
|
$exclude_tokens[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
|
|
// Find all the variables in scope, and figure out where they are used.
|
|
|
|
// We want to find foreach() iterators which are both declared before and
|
|
|
|
// used after the foreach() loop.
|
|
|
|
|
|
|
|
$uses = array();
|
2011-01-10 00:22:25 +01:00
|
|
|
|
|
|
|
$all_vars = $body->selectDescendantsOfType('n_VARIABLE');
|
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
|
|
|
$all = array();
|
|
|
|
|
|
|
|
// NOTE: $all_vars is not a real array so we can't unset() it.
|
2011-01-10 00:22:25 +01:00
|
|
|
foreach ($all_vars as $var) {
|
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
|
|
|
|
|
|
|
// Be strict since it's easier; we don't let you reuse an iterator you
|
|
|
|
// declared before a loop after the loop, even if you're just assigning
|
|
|
|
// to it.
|
|
|
|
|
2013-07-10 17:26:11 +02:00
|
|
|
$concrete = $this->getConcreteVariableString($var);
|
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
|
|
|
$uses[$concrete][$var->getID()] = $var->getOffset();
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
if (isset($declaration_tokens[$var->getID()])) {
|
|
|
|
// We know this is part of a declaration, so it's fine.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (isset($exclude_tokens[$var->getID()])) {
|
|
|
|
// We know this is part of isset() or similar, so it's fine.
|
|
|
|
continue;
|
|
|
|
}
|
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
|
|
|
|
2013-07-10 17:26:11 +02:00
|
|
|
$all[$var->getOffset()] = $concrete;
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Do foreach() last, we want to handle implicit redeclaration of a
|
|
|
|
// variable already in scope since this probably means we're ovewriting a
|
|
|
|
// local.
|
|
|
|
|
|
|
|
// NOTE: Processing foreach expressions in order allows programs which
|
|
|
|
// reuse iterator variables in other foreach() loops -- this is fine. We
|
|
|
|
// have a separate warning to prevent nested loops from reusing the same
|
|
|
|
// iterators.
|
|
|
|
|
|
|
|
$foreaches = $body->selectDescendantsOfType('n_FOREACH');
|
|
|
|
$all_foreach_vars = array();
|
|
|
|
foreach ($foreaches as $foreach) {
|
|
|
|
$foreach_expr = $foreach->getChildOfType(0, 'n_FOREACH_EXPRESSION');
|
|
|
|
|
|
|
|
$foreach_vars = array();
|
|
|
|
|
|
|
|
// Determine the end of the foreach() loop.
|
|
|
|
$foreach_tokens = $foreach->getTokens();
|
|
|
|
$last_token = end($foreach_tokens);
|
|
|
|
$foreach_end = $last_token->getOffset();
|
|
|
|
|
|
|
|
$key_var = $foreach_expr->getChildByIndex(1);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($key_var->getTypeName() === 'n_VARIABLE') {
|
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
|
|
|
$foreach_vars[] = $key_var;
|
|
|
|
}
|
|
|
|
|
|
|
|
$value_var = $foreach_expr->getChildByIndex(2);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($value_var->getTypeName() === 'n_VARIABLE') {
|
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
|
|
|
$foreach_vars[] = $value_var;
|
|
|
|
} else {
|
|
|
|
// The root-level token may be a reference, as in:
|
|
|
|
// foreach ($a as $b => &$c) { ... }
|
|
|
|
// Reach into the n_VARIABLE_REFERENCE node to grab the n_VARIABLE
|
|
|
|
// node.
|
2013-03-05 16:45:02 +01:00
|
|
|
$var = $value_var->getChildByIndex(0);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($var->getTypeName() === 'n_VARIABLE_VARIABLE') {
|
2013-03-05 16:45:02 +01:00
|
|
|
$var = $var->getChildByIndex(0);
|
|
|
|
}
|
|
|
|
$foreach_vars[] = $var;
|
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
|
|
|
}
|
|
|
|
|
|
|
|
// Remove all uses of the iterators inside of the foreach() loop from
|
|
|
|
// the $uses map.
|
|
|
|
|
|
|
|
foreach ($foreach_vars as $var) {
|
|
|
|
$concrete = $this->getConcreteVariableString($var);
|
|
|
|
$offset = $var->getOffset();
|
|
|
|
|
|
|
|
foreach ($uses[$concrete] as $id => $use_offset) {
|
|
|
|
if (($use_offset >= $offset) && ($use_offset < $foreach_end)) {
|
|
|
|
unset($uses[$concrete][$id]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$all_foreach_vars[] = $var;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($all_foreach_vars as $var) {
|
|
|
|
$concrete = $this->getConcreteVariableString($var);
|
|
|
|
$offset = $var->getOffset();
|
|
|
|
|
|
|
|
// If a variable was declared before a foreach() and is used after
|
|
|
|
// it, raise a message.
|
|
|
|
|
|
|
|
if (isset($declarations[$concrete])) {
|
|
|
|
if ($declarations[$concrete] < $offset) {
|
|
|
|
if (!empty($uses[$concrete]) &&
|
|
|
|
max($uses[$concrete]) > $offset) {
|
2013-02-09 00:04:32 +01:00
|
|
|
$message = $this->raiseLintAtNode(
|
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
|
|
|
$var,
|
|
|
|
self::LINT_REUSED_AS_ITERATOR,
|
|
|
|
'This iterator variable is a previously declared local '.
|
|
|
|
'variable. To avoid overwriting locals, do not reuse them '.
|
|
|
|
'as iterator variables.');
|
2013-02-09 00:04:32 +01:00
|
|
|
$message->setOtherLocations(array(
|
|
|
|
$this->getOtherLocation($declarations[$concrete]),
|
|
|
|
$this->getOtherLocation(max($uses[$concrete])),
|
|
|
|
));
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-12 11:11:22 +02:00
|
|
|
// This is a declaration, exclude it from the "declare variables prior
|
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
|
|
|
// to use" check below.
|
2013-07-10 17:26:11 +02:00
|
|
|
unset($all[$var->getOffset()]);
|
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
|
|
|
|
|
|
|
$vars[] = $var;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now rebuild declarations to include foreach().
|
|
|
|
|
|
|
|
foreach ($vars as $var) {
|
|
|
|
$concrete = $this->getConcreteVariableString($var);
|
|
|
|
$declarations[$concrete] = min(
|
|
|
|
idx($declarations, $concrete, PHP_INT_MAX),
|
|
|
|
$var->getOffset());
|
|
|
|
$declaration_tokens[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
|
2013-07-10 17:26:11 +02:00
|
|
|
foreach (array('n_STRING_SCALAR', 'n_HEREDOC') as $type) {
|
|
|
|
foreach ($body->selectDescendantsOfType($type) as $string) {
|
|
|
|
foreach ($string->getStringVariables() as $offset => $var) {
|
|
|
|
$all[$string->getOffset() + $offset - 1] = '$'.$var;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
// Issue a warning for every variable token, unless it appears in a
|
|
|
|
// declaration, we know about a prior declaration, we have explicitly
|
|
|
|
// exlcuded it, or scope has been made unknowable before it appears.
|
|
|
|
|
|
|
|
$issued_warnings = array();
|
2013-07-10 17:26:11 +02:00
|
|
|
foreach ($all as $offset => $concrete) {
|
|
|
|
if ($offset >= $scope_destroyed_at) {
|
2011-01-10 00:22:25 +01:00
|
|
|
// This appears after an extract() or $$var so we have no idea
|
|
|
|
// whether it's legitimate or not. We raised a harshly-worded warning
|
|
|
|
// when scope was made unknowable, so just ignore anything we can't
|
|
|
|
// figure out.
|
|
|
|
continue;
|
|
|
|
}
|
2013-07-10 17:26:11 +02:00
|
|
|
if ($offset >= idx($declarations, $concrete, PHP_INT_MAX)) {
|
2011-01-10 00:22:25 +01:00
|
|
|
// The use appears after the variable is declared, so it's fine.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!empty($issued_warnings[$concrete])) {
|
|
|
|
// We've already issued a warning for this variable so we don't need
|
|
|
|
// to issue another one.
|
|
|
|
continue;
|
|
|
|
}
|
2013-07-10 17:26:11 +02:00
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
$offset,
|
2011-01-10 00:22:25 +01:00
|
|
|
self::LINT_UNDECLARED_VARIABLE,
|
|
|
|
'Declare variables prior to use (even if you are passing them '.
|
|
|
|
'as reference parameters). You may have misspelled this '.
|
2013-07-10 17:26:11 +02:00
|
|
|
'variable name.',
|
|
|
|
$concrete);
|
2011-01-10 00:22:25 +01:00
|
|
|
$issued_warnings[$concrete] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function getConcreteVariableString(XHPASTNode $var) {
|
2011-05-12 03:51:25 +02:00
|
|
|
$concrete = $var->getConcreteString();
|
|
|
|
// Strip off curly braces as in $obj->{$property}.
|
|
|
|
$concrete = trim($concrete, '{}');
|
|
|
|
return $concrete;
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintPHPTagUse(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$tokens = $root->getTokens();
|
|
|
|
foreach ($tokens as $token) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($token->getTypeName() === 'T_OPEN_TAG') {
|
|
|
|
if (trim($token->getValue()) === '<?') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
|
|
|
self::LINT_PHP_SHORT_TAG,
|
|
|
|
'Use the full form of the PHP open tag, "<?php".',
|
|
|
|
"<?php\n");
|
|
|
|
}
|
|
|
|
break;
|
2014-07-11 17:21:21 +02:00
|
|
|
} else if ($token->getTypeName() === 'T_OPEN_TAG_WITH_ECHO') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
|
|
|
self::LINT_PHP_ECHO_TAG,
|
|
|
|
'Avoid the PHP echo short form, "<?=".');
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
if (!preg_match('/^#!/', $token->getValue())) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
|
|
|
self::LINT_PHP_OPEN_TAG,
|
|
|
|
'PHP files should start with "<?php", which may be preceded by '.
|
|
|
|
'a "#!" line for scripts.');
|
|
|
|
}
|
2012-03-08 02:52:46 +01:00
|
|
|
break;
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
2012-03-08 21:20:46 +01:00
|
|
|
|
|
|
|
foreach ($root->selectTokensOfType('T_CLOSE_TAG') as $token) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
|
|
|
self::LINT_PHP_CLOSE_TAG,
|
|
|
|
'Do not use the PHP closing tag, "?>".');
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintNamingConventions(XHPASTNode $root) {
|
2011-08-30 22:07:01 +02:00
|
|
|
// We're going to build up a list of <type, name, token, error> tuples
|
|
|
|
// and then try to instantiate a hook class which has the opportunity to
|
|
|
|
// override us.
|
|
|
|
$names = array();
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
$classes = $root->selectDescendantsOfType('n_CLASS_DECLARATION');
|
|
|
|
foreach ($classes as $class) {
|
|
|
|
$name_token = $class->getChildByIndex(1);
|
|
|
|
$name_string = $name_token->getConcreteString();
|
2012-01-28 20:17:45 +01:00
|
|
|
|
|
|
|
$names[] = array(
|
|
|
|
'class',
|
|
|
|
$name_string,
|
|
|
|
$name_token,
|
|
|
|
ArcanistXHPASTLintNamingHook::isUpperCamelCase($name_string)
|
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: classes should be named using '.
|
|
|
|
'UpperCamelCase.',
|
|
|
|
);
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$ifaces = $root->selectDescendantsOfType('n_INTERFACE_DECLARATION');
|
|
|
|
foreach ($ifaces as $iface) {
|
|
|
|
$name_token = $iface->getChildByIndex(1);
|
|
|
|
$name_string = $name_token->getConcreteString();
|
2011-08-30 22:07:01 +02:00
|
|
|
$names[] = array(
|
|
|
|
'interface',
|
|
|
|
$name_string,
|
|
|
|
$name_token,
|
2012-01-28 20:17:45 +01:00
|
|
|
ArcanistXHPASTLintNamingHook::isUpperCamelCase($name_string)
|
2011-08-30 22:07:01 +02:00
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: interfaces should be named using '.
|
|
|
|
'UpperCamelCase.',
|
|
|
|
);
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
$functions = $root->selectDescendantsOfType('n_FUNCTION_DECLARATION');
|
|
|
|
foreach ($functions as $function) {
|
|
|
|
$name_token = $function->getChildByIndex(2);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name_token->getTypeName() === 'n_EMPTY') {
|
2011-01-10 00:22:25 +01:00
|
|
|
// Unnamed closure.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$name_string = $name_token->getConcreteString();
|
2011-08-30 22:07:01 +02:00
|
|
|
$names[] = array(
|
|
|
|
'function',
|
|
|
|
$name_string,
|
|
|
|
$name_token,
|
2012-01-28 20:17:45 +01:00
|
|
|
ArcanistXHPASTLintNamingHook::isLowercaseWithUnderscores(
|
|
|
|
ArcanistXHPASTLintNamingHook::stripPHPFunction($name_string))
|
2011-08-30 22:07:01 +02:00
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: functions should be named using '.
|
|
|
|
'lowercase_with_underscores.',
|
|
|
|
);
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
$methods = $root->selectDescendantsOfType('n_METHOD_DECLARATION');
|
|
|
|
foreach ($methods as $method) {
|
|
|
|
$name_token = $method->getChildByIndex(2);
|
|
|
|
$name_string = $name_token->getConcreteString();
|
2011-08-30 22:07:01 +02:00
|
|
|
$names[] = array(
|
|
|
|
'method',
|
|
|
|
$name_string,
|
|
|
|
$name_token,
|
2012-01-28 20:17:45 +01:00
|
|
|
ArcanistXHPASTLintNamingHook::isLowerCamelCase(
|
|
|
|
ArcanistXHPASTLintNamingHook::stripPHPFunction($name_string))
|
2011-08-30 22:07:01 +02:00
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: methods should be named using '.
|
|
|
|
'lowerCamelCase.',
|
|
|
|
);
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
2012-01-28 20:17:45 +01:00
|
|
|
$param_tokens = array();
|
2011-01-10 00:22:25 +01:00
|
|
|
|
|
|
|
$params = $root->selectDescendantsOfType('n_DECLARATION_PARAMETER_LIST');
|
|
|
|
foreach ($params as $param_list) {
|
|
|
|
foreach ($param_list->getChildren() as $param) {
|
|
|
|
$name_token = $param->getChildByIndex(1);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($name_token->getTypeName() === 'n_VARIABLE_REFERENCE') {
|
2012-01-28 20:17:45 +01:00
|
|
|
$name_token = $name_token->getChildOfType(0, 'n_VARIABLE');
|
|
|
|
}
|
|
|
|
$param_tokens[$name_token->getID()] = true;
|
2011-01-10 00:22:25 +01:00
|
|
|
$name_string = $name_token->getConcreteString();
|
2012-01-28 20:17:45 +01:00
|
|
|
|
2011-08-30 22:07:01 +02:00
|
|
|
$names[] = array(
|
|
|
|
'parameter',
|
|
|
|
$name_string,
|
|
|
|
$name_token,
|
2012-01-28 20:17:45 +01:00
|
|
|
ArcanistXHPASTLintNamingHook::isLowercaseWithUnderscores(
|
|
|
|
ArcanistXHPASTLintNamingHook::stripPHPVariable($name_string))
|
2011-08-30 22:07:01 +02:00
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: parameters should be named using '.
|
|
|
|
'lowercase_with_underscores.',
|
|
|
|
);
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
$constants = $root->selectDescendantsOfType(
|
|
|
|
'n_CLASS_CONSTANT_DECLARATION_LIST');
|
|
|
|
foreach ($constants as $constant_list) {
|
|
|
|
foreach ($constant_list->getChildren() as $constant) {
|
|
|
|
$name_token = $constant->getChildByIndex(0);
|
|
|
|
$name_string = $name_token->getConcreteString();
|
2011-08-30 22:07:01 +02:00
|
|
|
$names[] = array(
|
|
|
|
'constant',
|
|
|
|
$name_string,
|
|
|
|
$name_token,
|
2012-01-28 20:17:45 +01:00
|
|
|
ArcanistXHPASTLintNamingHook::isUppercaseWithUnderscores($name_string)
|
2011-08-30 22:07:01 +02:00
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: class constants should be named '.
|
|
|
|
'using UPPERCASE_WITH_UNDERSCORES.',
|
|
|
|
);
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-28 20:17:45 +01:00
|
|
|
$member_tokens = array();
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
$props = $root->selectDescendantsOfType('n_CLASS_MEMBER_DECLARATION_LIST');
|
|
|
|
foreach ($props as $prop_list) {
|
2012-01-28 20:17:45 +01:00
|
|
|
foreach ($prop_list->getChildren() as $token_id => $prop) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($prop->getTypeName() === 'n_CLASS_MEMBER_MODIFIER_LIST') {
|
2011-01-10 00:22:25 +01:00
|
|
|
continue;
|
|
|
|
}
|
2012-01-28 20:17:45 +01:00
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
$name_token = $prop->getChildByIndex(0);
|
2012-01-28 20:17:45 +01:00
|
|
|
$member_tokens[$name_token->getID()] = true;
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
$name_string = $name_token->getConcreteString();
|
2011-08-30 22:07:01 +02:00
|
|
|
$names[] = array(
|
|
|
|
'member',
|
|
|
|
$name_string,
|
|
|
|
$name_token,
|
2012-01-28 20:17:45 +01:00
|
|
|
ArcanistXHPASTLintNamingHook::isLowerCamelCase(
|
|
|
|
ArcanistXHPASTLintNamingHook::stripPHPVariable($name_string))
|
2011-08-30 22:07:01 +02:00
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: class properties should be named '.
|
|
|
|
'using lowerCamelCase.',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-28 20:17:45 +01:00
|
|
|
$superglobal_map = array_fill_keys(
|
|
|
|
$this->getSuperGlobalNames(),
|
|
|
|
true);
|
|
|
|
|
|
|
|
|
|
|
|
$fdefs = $root->selectDescendantsOfType('n_FUNCTION_DECLARATION');
|
|
|
|
$mdefs = $root->selectDescendantsOfType('n_METHOD_DECLARATION');
|
|
|
|
$defs = $fdefs->add($mdefs);
|
|
|
|
|
|
|
|
foreach ($defs as $def) {
|
|
|
|
$globals = $def->selectDescendantsOfType('n_GLOBAL_DECLARATION_LIST');
|
|
|
|
$globals = $globals->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
|
|
|
|
$globals_map = array();
|
|
|
|
foreach ($globals as $global) {
|
|
|
|
$global_string = $global->getConcreteString();
|
|
|
|
$globals_map[$global_string] = true;
|
|
|
|
$names[] = array(
|
2013-10-19 01:10:06 +02:00
|
|
|
'user',
|
2012-01-28 20:17:45 +01:00
|
|
|
$global_string,
|
|
|
|
$global,
|
|
|
|
|
|
|
|
// No advice for globals, but hooks have an option to provide some.
|
|
|
|
null);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exclude access of static properties, since lint will be raised at
|
|
|
|
// their declaration if they're invalid and they may not conform to
|
|
|
|
// variable rules. This is slightly overbroad (includes the entire
|
|
|
|
// rhs of a "Class::..." token) to cover cases like "Class:$x[0]". These
|
2013-10-20 16:41:48 +02:00
|
|
|
// variables are simply made exempt from naming conventions.
|
2012-01-28 20:17:45 +01:00
|
|
|
$exclude_tokens = array();
|
|
|
|
$statics = $def->selectDescendantsOfType('n_CLASS_STATIC_ACCESS');
|
|
|
|
foreach ($statics as $static) {
|
|
|
|
$rhs = $static->getChildByIndex(1);
|
|
|
|
$rhs_vars = $def->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($rhs_vars as $var) {
|
|
|
|
$exclude_tokens[$var->getID()] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$vars = $def->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($vars as $token_id => $var) {
|
|
|
|
if (isset($member_tokens[$token_id])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (isset($param_tokens[$token_id])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (isset($exclude_tokens[$token_id])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$var_string = $var->getConcreteString();
|
|
|
|
|
|
|
|
// Awkward artifact of "$o->{$x}".
|
|
|
|
$var_string = trim($var_string, '{}');
|
|
|
|
|
|
|
|
if (isset($superglobal_map[$var_string])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (isset($globals_map[$var_string])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$names[] = array(
|
|
|
|
'variable',
|
|
|
|
$var_string,
|
|
|
|
$var,
|
|
|
|
ArcanistXHPASTLintNamingHook::isLowercaseWithUnderscores(
|
|
|
|
ArcanistXHPASTLintNamingHook::stripPHPVariable($var_string))
|
|
|
|
? null
|
|
|
|
: 'Follow naming conventions: variables should be named using '.
|
|
|
|
'lowercase_with_underscores.',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-30 22:07:01 +02:00
|
|
|
$engine = $this->getEngine();
|
|
|
|
$working_copy = $engine->getWorkingCopy();
|
|
|
|
|
|
|
|
if ($working_copy) {
|
|
|
|
// If a naming hook is configured, give it a chance to override the
|
|
|
|
// default results for all the symbol names.
|
2014-05-12 13:46:56 +02:00
|
|
|
$hook_class = $this->naminghook
|
|
|
|
? $this->naminghook
|
|
|
|
: $working_copy->getProjectConfig('lint.xhpast.naminghook');
|
2011-08-30 22:07:01 +02:00
|
|
|
if ($hook_class) {
|
|
|
|
$hook_obj = newv($hook_class, array());
|
|
|
|
foreach ($names as $k => $name_attrs) {
|
|
|
|
list($type, $name, $token, $default) = $name_attrs;
|
|
|
|
$result = $hook_obj->lintSymbolName($type, $name, $default);
|
|
|
|
$names[$k][3] = $result;
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-08-30 22:07:01 +02:00
|
|
|
|
|
|
|
// Raise anything we're left with.
|
|
|
|
foreach ($names as $k => $name_attrs) {
|
|
|
|
list($type, $name, $token, $result) = $name_attrs;
|
|
|
|
if ($result) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$token,
|
|
|
|
self::LINT_NAMING_CONVENTIONS,
|
|
|
|
$result);
|
|
|
|
}
|
|
|
|
}
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintSurpriseConstructors(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$classes = $root->selectDescendantsOfType('n_CLASS_DECLARATION');
|
|
|
|
foreach ($classes as $class) {
|
|
|
|
$class_name = $class->getChildByIndex(1)->getConcreteString();
|
|
|
|
$methods = $class->selectDescendantsOfType('n_METHOD_DECLARATION');
|
|
|
|
foreach ($methods as $method) {
|
|
|
|
$method_name_token = $method->getChildByIndex(2);
|
|
|
|
$method_name = $method_name_token->getConcreteString();
|
2014-07-11 17:21:21 +02:00
|
|
|
if (strtolower($class_name) === strtolower($method_name)) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$method_name_token,
|
|
|
|
self::LINT_IMPLICIT_CONSTRUCTOR,
|
|
|
|
'Name constructors __construct() explicitly. This method is a '.
|
|
|
|
'constructor because it has the same name as the class it is '.
|
|
|
|
'defined in.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintParenthesesShouldHugExpressions(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$calls = $root->selectDescendantsOfType('n_CALL_PARAMETER_LIST');
|
|
|
|
$controls = $root->selectDescendantsOfType('n_CONTROL_CONDITION');
|
|
|
|
$fors = $root->selectDescendantsOfType('n_FOR_EXPRESSION');
|
|
|
|
$foreach = $root->selectDescendantsOfType('n_FOREACH_EXPRESSION');
|
|
|
|
$decl = $root->selectDescendantsOfType('n_DECLARATION_PARAMETER_LIST');
|
|
|
|
|
|
|
|
$all_paren_groups = $calls
|
|
|
|
->add($controls)
|
|
|
|
->add($fors)
|
|
|
|
->add($foreach)
|
|
|
|
->add($decl);
|
|
|
|
foreach ($all_paren_groups as $group) {
|
|
|
|
$tokens = $group->getTokens();
|
|
|
|
|
|
|
|
$token_o = array_shift($tokens);
|
|
|
|
$token_c = array_pop($tokens);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($token_o->getTypeName() !== '(') {
|
2011-01-10 00:22:25 +01:00
|
|
|
throw new Exception('Expected open paren!');
|
|
|
|
}
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($token_c->getTypeName() !== ')') {
|
2011-01-10 00:22:25 +01:00
|
|
|
throw new Exception('Expected close paren!');
|
|
|
|
}
|
|
|
|
|
|
|
|
$nonsem_o = $token_o->getNonsemanticTokensAfter();
|
|
|
|
$nonsem_c = $token_c->getNonsemanticTokensBefore();
|
|
|
|
|
|
|
|
if (!$nonsem_o) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$raise = array();
|
|
|
|
|
|
|
|
$string_o = implode('', mpull($nonsem_o, 'getValue'));
|
|
|
|
if (preg_match('/^[ ]+$/', $string_o)) {
|
|
|
|
$raise[] = array($nonsem_o, $string_o);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($nonsem_o !== $nonsem_c) {
|
|
|
|
$string_c = implode('', mpull($nonsem_c, 'getValue'));
|
|
|
|
if (preg_match('/^[ ]+$/', $string_c)) {
|
|
|
|
$raise[] = array($nonsem_c, $string_c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($raise as $warning) {
|
|
|
|
list($tokens, $string) = $warning;
|
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
reset($tokens)->getOffset(),
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_PARENTHESES_SPACING,
|
2011-01-10 00:22:25 +01:00
|
|
|
'Parentheses should hug their contents.',
|
|
|
|
$string,
|
|
|
|
'');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintSpaceAfterControlStatementKeywords(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
foreach ($root->getTokens() as $id => $token) {
|
|
|
|
switch ($token->getTypeName()) {
|
|
|
|
case 'T_IF':
|
|
|
|
case 'T_ELSE':
|
|
|
|
case 'T_FOR':
|
|
|
|
case 'T_FOREACH':
|
|
|
|
case 'T_WHILE':
|
|
|
|
case 'T_DO':
|
|
|
|
case 'T_SWITCH':
|
|
|
|
$after = $token->getNonsemanticTokensAfter();
|
|
|
|
if (empty($after)) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_CONTROL_STATEMENT_SPACING,
|
2011-01-10 00:22:25 +01:00
|
|
|
'Convention: put a space after control statements.',
|
|
|
|
$token->getValue().' ');
|
2014-07-11 17:21:21 +02:00
|
|
|
} else if (count($after) === 1) {
|
2012-01-04 19:59:00 +01:00
|
|
|
$space = head($after);
|
2012-01-12 03:03:56 +01:00
|
|
|
|
|
|
|
// If we have an else clause with braces, $space may not be
|
|
|
|
// a single white space. e.g.,
|
|
|
|
//
|
|
|
|
// if ($x)
|
|
|
|
// echo 'foo'
|
|
|
|
// else // <- $space is not " " but "\n ".
|
|
|
|
// echo 'bar'
|
|
|
|
//
|
|
|
|
// We just require it starts with either a whitespace or a newline.
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($token->getTypeName() === 'T_ELSE' ||
|
|
|
|
$token->getTypeName() === 'T_DO') {
|
2012-01-12 03:03:56 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($space->isAnyWhitespace() && $space->getValue() !== ' ') {
|
2012-01-04 19:59:00 +01:00
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$space,
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_CONTROL_STATEMENT_SPACING,
|
2012-01-04 19:59:00 +01:00
|
|
|
'Convention: put a single space after control statements.',
|
|
|
|
' ');
|
|
|
|
}
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintSpaceAroundBinaryOperators(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$expressions = $root->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
foreach ($expressions as $expression) {
|
|
|
|
$operator = $expression->getChildByIndex(1);
|
|
|
|
$operator_value = $operator->getConcreteString();
|
2012-04-09 01:09:11 +02:00
|
|
|
list($before, $after) = $operator->getSurroundingNonsemanticTokens();
|
|
|
|
|
|
|
|
$replace = null;
|
|
|
|
if (empty($before) && empty($after)) {
|
|
|
|
$replace = " {$operator_value} ";
|
|
|
|
} else if (empty($before)) {
|
|
|
|
$replace = " {$operator_value}";
|
|
|
|
} else if (empty($after)) {
|
|
|
|
$replace = "{$operator_value} ";
|
|
|
|
}
|
2011-01-10 00:22:25 +01:00
|
|
|
|
2012-04-09 01:09:11 +02:00
|
|
|
if ($replace !== null) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$operator,
|
|
|
|
self::LINT_BINARY_EXPRESSION_SPACING,
|
|
|
|
'Convention: logical and arithmetic operators should be '.
|
|
|
|
'surrounded by whitespace.',
|
|
|
|
$replace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$tokens = $root->selectTokensOfType(',');
|
|
|
|
foreach ($tokens as $token) {
|
|
|
|
$next = $token->getNextToken();
|
|
|
|
switch ($next->getTypeName()) {
|
|
|
|
case ')':
|
|
|
|
case 'T_WHITESPACE':
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_BINARY_EXPRESSION_SPACING,
|
2012-04-09 01:09:11 +02:00
|
|
|
'Convention: comma should be followed by space.',
|
|
|
|
', ');
|
|
|
|
break;
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
2012-04-09 01:09:11 +02:00
|
|
|
|
2014-05-28 01:16:39 +02:00
|
|
|
$tokens = $root->selectTokensOfType('T_DOUBLE_ARROW');
|
|
|
|
foreach ($tokens as $token) {
|
|
|
|
$prev = $token->getPrevToken();
|
|
|
|
$next = $token->getNextToken();
|
|
|
|
|
|
|
|
$prev_type = $prev->getTypeName();
|
|
|
|
$next_type = $next->getTypeName();
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
$prev_space = ($prev_type === 'T_WHITESPACE');
|
|
|
|
$next_space = ($next_type === 'T_WHITESPACE');
|
2014-05-28 01:16:39 +02:00
|
|
|
|
|
|
|
$replace = null;
|
|
|
|
if (!$prev_space && !$next_space) {
|
|
|
|
$replace = ' => ';
|
|
|
|
} else if ($prev_space && !$next_space) {
|
|
|
|
$replace = '=> ';
|
|
|
|
} else if (!$prev_space && $next_space) {
|
|
|
|
$replace = ' =>';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($replace !== null) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
|
|
|
self::LINT_BINARY_EXPRESSION_SPACING,
|
|
|
|
'Convention: double arrow should be surrounded by whitespace.',
|
|
|
|
$replace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-09 01:09:11 +02:00
|
|
|
// TODO: Spacing around default parameter assignment in function/method
|
|
|
|
// declarations (which is not n_BINARY_EXPRESSION).
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintSpaceAroundConcatenationOperators(XHPASTNode $root) {
|
2014-05-28 01:16:39 +02:00
|
|
|
$tokens = $root->selectTokensOfType('.');
|
|
|
|
foreach ($tokens as $token) {
|
|
|
|
$prev = $token->getPrevToken();
|
|
|
|
$next = $token->getNextToken();
|
|
|
|
|
|
|
|
foreach (array('prev' => $prev, 'next' => $next) as $wtoken) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($wtoken->getTypeName() !== 'T_WHITESPACE') {
|
2014-05-28 01:16:39 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$value = $wtoken->getValue();
|
|
|
|
if (strpos($value, "\n") !== false) {
|
|
|
|
// If the whitespace has a newline, it's conventional.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-06-09 21:25:14 +02:00
|
|
|
$next = $wtoken->getNextToken();
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($next && $next->getTypeName() === 'T_COMMENT') {
|
2014-06-09 21:25:14 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-05-28 01:16:39 +02:00
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$wtoken,
|
|
|
|
self::LINT_BINARY_EXPRESSION_SPACING,
|
|
|
|
'Convention: no spaces around "." (string concatenation) operator.',
|
|
|
|
'');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintDynamicDefines(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$calls = $root->selectDescendantsOfType('n_FUNCTION_CALL');
|
|
|
|
foreach ($calls as $call) {
|
|
|
|
$name = $call->getChildByIndex(0)->getConcreteString();
|
2014-07-11 17:21:21 +02:00
|
|
|
if (strtolower($name) === 'define') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$parameter_list = $call->getChildOfType(1, 'n_CALL_PARAMETER_LIST');
|
|
|
|
$defined = $parameter_list->getChildByIndex(0);
|
|
|
|
if (!$defined->isStaticScalar()) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$defined,
|
|
|
|
self::LINT_DYNAMIC_DEFINE,
|
|
|
|
'First argument to define() must be a string literal.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintUseOfThisInStaticMethods(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$classes = $root->selectDescendantsOfType('n_CLASS_DECLARATION');
|
|
|
|
foreach ($classes as $class) {
|
|
|
|
$methods = $class->selectDescendantsOfType('n_METHOD_DECLARATION');
|
|
|
|
foreach ($methods as $method) {
|
|
|
|
|
|
|
|
$attributes = $method
|
|
|
|
->getChildByIndex(0, 'n_METHOD_MODIFIER_LIST')
|
|
|
|
->selectDescendantsOfType('n_STRING');
|
|
|
|
|
|
|
|
$method_is_static = false;
|
|
|
|
$method_is_abstract = false;
|
|
|
|
foreach ($attributes as $attribute) {
|
2014-07-11 17:21:21 +02:00
|
|
|
if (strtolower($attribute->getConcreteString()) === 'static') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$method_is_static = true;
|
|
|
|
}
|
2014-07-11 17:21:21 +02:00
|
|
|
if (strtolower($attribute->getConcreteString()) === 'abstract') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$method_is_abstract = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($method_is_abstract) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$method_is_static) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$body = $method->getChildOfType(5, 'n_STATEMENT_LIST');
|
|
|
|
|
|
|
|
$variables = $body->selectDescendantsOfType('n_VARIABLE');
|
|
|
|
foreach ($variables as $variable) {
|
|
|
|
if ($method_is_static &&
|
2014-07-11 17:21:21 +02:00
|
|
|
strtolower($variable->getConcreteString()) === '$this') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$variable,
|
|
|
|
self::LINT_STATIC_THIS,
|
|
|
|
'You can not reference "$this" inside a static method.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* preg_quote() takes two arguments, but the second one is optional because
|
2014-07-09 01:12:13 +02:00
|
|
|
* it is possible to use (), [] or {} as regular expression delimiters. If
|
2011-12-03 01:55:02 +01:00
|
|
|
* you don't pass a second argument, you're probably going to get something
|
|
|
|
* wrong.
|
2011-01-10 00:22:25 +01:00
|
|
|
*/
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintPregQuote(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$function_calls = $root->selectDescendantsOfType('n_FUNCTION_CALL');
|
|
|
|
foreach ($function_calls as $call) {
|
|
|
|
$name = $call->getChildByIndex(0)->getConcreteString();
|
|
|
|
if (strtolower($name) === 'preg_quote') {
|
|
|
|
$parameter_list = $call->getChildOfType(1, 'n_CALL_PARAMETER_LIST');
|
|
|
|
if (count($parameter_list->getChildren()) !== 2) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$call,
|
|
|
|
self::LINT_PREG_QUOTE_MISUSE,
|
2012-10-06 01:32:06 +02:00
|
|
|
'If you use pattern delimiters that require escaping (such as //, '.
|
|
|
|
'but not ()) then you should pass two arguments to preg_quote(), '.
|
|
|
|
'so that preg_quote() knows which delimiter to escape.');
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Exit is parsed as an expression, but using it as such is almost always
|
|
|
|
* wrong. That is, this is valid:
|
|
|
|
*
|
|
|
|
* strtoupper(33 * exit - 6);
|
|
|
|
*
|
|
|
|
* When exit is used as an expression, it causes the program to terminate with
|
|
|
|
* exit code 0. This is likely not what is intended; these statements have
|
|
|
|
* different effects:
|
|
|
|
*
|
|
|
|
* exit(-1);
|
|
|
|
* exit -1;
|
|
|
|
*
|
|
|
|
* The former exits with a failure code, the latter with a success code!
|
|
|
|
*/
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintExitExpressions(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$unaries = $root->selectDescendantsOfType('n_UNARY_PREFIX_EXPRESSION');
|
|
|
|
foreach ($unaries as $unary) {
|
|
|
|
$operator = $unary->getChildByIndex(0)->getConcreteString();
|
2014-07-11 17:21:21 +02:00
|
|
|
if (strtolower($operator) === 'exit') {
|
|
|
|
if ($unary->getParentNode()->getTypeName() !== 'n_STATEMENT') {
|
2011-01-10 00:22:25 +01:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$unary,
|
|
|
|
self::LINT_EXIT_EXPRESSION,
|
2014-05-23 22:53:05 +02:00
|
|
|
'Use exit as a statement, not an expression.');
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintArrayIndexWhitespace(XHPASTNode $root) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$indexes = $root->selectDescendantsOfType('n_INDEX_ACCESS');
|
|
|
|
foreach ($indexes as $index) {
|
|
|
|
$tokens = $index->getChildByIndex(0)->getTokens();
|
|
|
|
$last = array_pop($tokens);
|
|
|
|
$trailing = $last->getNonsemanticTokensAfter();
|
|
|
|
$trailing_text = implode('', mpull($trailing, 'getValue'));
|
|
|
|
if (preg_match('/^ +$/', $trailing_text)) {
|
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
$last->getOffset() + strlen($last->getValue()),
|
2012-01-26 04:05:47 +01:00
|
|
|
self::LINT_ARRAY_INDEX_SPACING,
|
2011-01-10 00:22:25 +01:00
|
|
|
'Convention: no spaces before index access.',
|
|
|
|
$trailing_text,
|
|
|
|
'');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 17:30:15 +02:00
|
|
|
private function lintTODOComments(XHPASTNode $root) {
|
2012-03-08 21:20:46 +01:00
|
|
|
$comments = $root->selectTokensOfType('T_COMMENT') +
|
|
|
|
$root->selectTokensOfType('T_DOC_COMMENT');
|
2011-01-10 00:22:25 +01:00
|
|
|
|
2012-03-08 21:20:46 +01:00
|
|
|
foreach ($comments as $token) {
|
2011-01-10 00:22:25 +01:00
|
|
|
$value = $token->getValue();
|
2014-03-04 20:03:09 +01:00
|
|
|
if ($token->getTypeName() === 'T_DOC_COMMENT') {
|
|
|
|
$regex = '/(TODO|@todo)/';
|
|
|
|
} else {
|
|
|
|
$regex = '/TODO/';
|
|
|
|
}
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
$matches = null;
|
|
|
|
$preg = preg_match_all(
|
2014-03-04 20:03:09 +01:00
|
|
|
$regex,
|
2011-01-10 00:22:25 +01:00
|
|
|
$value,
|
|
|
|
$matches,
|
|
|
|
PREG_OFFSET_CAPTURE);
|
|
|
|
|
|
|
|
foreach ($matches[0] as $match) {
|
|
|
|
list($string, $offset) = $match;
|
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
$token->getOffset() + $offset,
|
|
|
|
self::LINT_TODO_COMMENT,
|
|
|
|
'This comment has a TODO.',
|
|
|
|
$string);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-14 23:28:01 +01:00
|
|
|
/**
|
|
|
|
* Lint that if the file declares exactly one interface or class,
|
|
|
|
* the name of the file matches the name of the class,
|
|
|
|
* unless the classname is funky like an XHP element.
|
|
|
|
*/
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintPrimaryDeclarationFilenameMatch(XHPASTNode $root) {
|
2011-01-14 23:28:01 +01:00
|
|
|
$classes = $root->selectDescendantsOfType('n_CLASS_DECLARATION');
|
|
|
|
$interfaces = $root->selectDescendantsOfType('n_INTERFACE_DECLARATION');
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if (count($classes) + count($interfaces) !== 1) {
|
2011-01-14 23:28:01 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$declarations = count($classes) ? $classes : $interfaces;
|
|
|
|
$declarations->rewind();
|
|
|
|
$declaration = $declarations->current();
|
|
|
|
|
|
|
|
$decl_name = $declaration->getChildByIndex(1);
|
|
|
|
$decl_string = $decl_name->getConcreteString();
|
|
|
|
|
2011-01-26 16:03:52 +01:00
|
|
|
// Exclude strangely named classes, e.g. XHP tags.
|
|
|
|
if (!preg_match('/^\w+$/', $decl_string)) {
|
2011-01-14 23:28:01 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$rename = $decl_string.'.php';
|
|
|
|
|
|
|
|
$path = $this->getActivePath();
|
|
|
|
$filename = basename($path);
|
|
|
|
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($rename === $filename) {
|
2011-01-14 23:28:01 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$decl_name,
|
|
|
|
self::LINT_CLASS_FILENAME_MISMATCH,
|
|
|
|
"The name of this file differs from the name of the class or interface ".
|
2013-02-19 23:09:20 +01:00
|
|
|
"it declares. Rename the file to '{$rename}'.");
|
2011-01-14 23:28:01 +01:00
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintPlusOperatorOnStrings(XHPASTNode $root) {
|
2011-03-11 00:11:41 +01:00
|
|
|
$binops = $root->selectDescendantsOfType('n_BINARY_EXPRESSION');
|
|
|
|
foreach ($binops as $binop) {
|
|
|
|
$op = $binop->getChildByIndex(1);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($op->getConcreteString() !== '+') {
|
2011-03-11 00:11:41 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$left = $binop->getChildByIndex(0);
|
|
|
|
$right = $binop->getChildByIndex(2);
|
2014-07-11 17:21:21 +02:00
|
|
|
if (($left->getTypeName() === 'n_STRING_SCALAR') ||
|
|
|
|
($right->getTypeName() === 'n_STRING_SCALAR')) {
|
2011-03-11 00:11:41 +01:00
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$binop,
|
|
|
|
self::LINT_PLUS_OPERATOR_ON_STRINGS,
|
|
|
|
"In PHP, '.' is the string concatenation operator, not '+'. This ".
|
|
|
|
"expression uses '+' with a string literal as an operand.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-25 21:24:25 +02:00
|
|
|
/**
|
|
|
|
* Finds duplicate keys in array initializers, as in
|
2014-07-09 01:12:13 +02:00
|
|
|
* array(1 => 'anything', 1 => 'foo'). Since the first entry is ignored,
|
2011-05-25 21:24:25 +02:00
|
|
|
* this is almost certainly an error.
|
|
|
|
*/
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintDuplicateKeysInArray(XHPASTNode $root) {
|
2011-05-25 21:24:25 +02:00
|
|
|
$array_literals = $root->selectDescendantsOfType('n_ARRAY_LITERAL');
|
|
|
|
foreach ($array_literals as $array_literal) {
|
|
|
|
$nodes_by_key = array();
|
|
|
|
$keys_warn = array();
|
|
|
|
$list_node = $array_literal->getChildByIndex(0);
|
|
|
|
foreach ($list_node->getChildren() as $array_entry) {
|
|
|
|
$key_node = $array_entry->getChildByIndex(0);
|
|
|
|
|
|
|
|
switch ($key_node->getTypeName()) {
|
|
|
|
case 'n_STRING_SCALAR':
|
|
|
|
case 'n_NUMERIC_SCALAR':
|
|
|
|
// Scalars: array(1 => 'v1', '1' => 'v2');
|
|
|
|
$key = 'scalar:'.(string)$key_node->evalStatic();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'n_SYMBOL_NAME':
|
|
|
|
case 'n_VARIABLE':
|
|
|
|
case 'n_CLASS_STATIC_ACCESS':
|
|
|
|
// Constants: array(CONST => 'v1', CONST => 'v2');
|
|
|
|
// Variables: array($a => 'v1', $a => 'v2');
|
|
|
|
// Class constants and vars: array(C::A => 'v1', C::A => 'v2');
|
|
|
|
$key = $key_node->getTypeName().':'.$key_node->getConcreteString();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
$key = null;
|
2012-03-09 17:57:03 +01:00
|
|
|
break;
|
2011-05-25 21:24:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($key !== null) {
|
|
|
|
if (isset($nodes_by_key[$key])) {
|
|
|
|
$keys_warn[$key] = true;
|
|
|
|
}
|
|
|
|
$nodes_by_key[$key][] = $key_node;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($keys_warn as $key => $_) {
|
2013-02-09 00:04:32 +01:00
|
|
|
$node = array_pop($nodes_by_key[$key]);
|
|
|
|
$message = $this->raiseLintAtNode(
|
|
|
|
$node,
|
|
|
|
self::LINT_DUPLICATE_KEYS_IN_ARRAY,
|
2014-05-23 22:53:05 +02:00
|
|
|
'Duplicate key in array initializer. PHP will ignore all '.
|
|
|
|
'but the last entry.');
|
2013-02-09 00:04:32 +01:00
|
|
|
|
|
|
|
$locations = array();
|
|
|
|
foreach ($nodes_by_key[$key] as $node) {
|
|
|
|
$locations[] = $this->getOtherLocation($node->getOffset());
|
2011-05-25 21:24:25 +02:00
|
|
|
}
|
2013-02-09 00:04:32 +01:00
|
|
|
$message->setOtherLocations($locations);
|
2011-05-25 21:24:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintClosingCallParen(XHPASTNode $root) {
|
2013-02-19 22:50:13 +01:00
|
|
|
$calls = $root->selectDescendantsOfType('n_FUNCTION_CALL');
|
|
|
|
$calls = $calls->add($root->selectDescendantsOfType('n_METHOD_CALL'));
|
|
|
|
|
|
|
|
foreach ($calls as $call) {
|
|
|
|
// If the last parameter of a call is a HEREDOC, don't apply this rule.
|
|
|
|
$params = $call
|
|
|
|
->getChildOfType(1, 'n_CALL_PARAMETER_LIST')
|
|
|
|
->getChildren();
|
|
|
|
|
|
|
|
if ($params) {
|
|
|
|
$last_param = last($params);
|
2014-07-11 17:21:21 +02:00
|
|
|
if ($last_param->getTypeName() === 'n_HEREDOC') {
|
2013-02-19 22:50:13 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$tokens = $call->getTokens();
|
|
|
|
$last = array_pop($tokens);
|
|
|
|
|
|
|
|
$trailing = $last->getNonsemanticTokensBefore();
|
|
|
|
$trailing_text = implode('', mpull($trailing, 'getValue'));
|
|
|
|
if (preg_match('/^\s+$/', $trailing_text)) {
|
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
$last->getOffset() - strlen($trailing_text),
|
|
|
|
self::LINT_CLOSING_CALL_PAREN,
|
|
|
|
'Convention: no spaces before closing parenthesis in calls.',
|
|
|
|
$trailing_text,
|
|
|
|
'');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintClosingDeclarationParen(XHPASTNode $root) {
|
2013-02-19 22:50:13 +01:00
|
|
|
$decs = $root->selectDescendantsOfType('n_FUNCTION_DECLARATION');
|
|
|
|
$decs = $decs->add($root->selectDescendantsOfType('n_METHOD_DECLARATION'));
|
|
|
|
|
|
|
|
foreach ($decs as $dec) {
|
|
|
|
$params = $dec->getChildOfType(3, 'n_DECLARATION_PARAMETER_LIST');
|
|
|
|
$tokens = $params->getTokens();
|
|
|
|
$last = array_pop($tokens);
|
|
|
|
|
|
|
|
$trailing = $last->getNonsemanticTokensBefore();
|
|
|
|
$trailing_text = implode('', mpull($trailing, 'getValue'));
|
|
|
|
if (preg_match('/^\s+$/', $trailing_text)) {
|
|
|
|
$this->raiseLintAtOffset(
|
|
|
|
$last->getOffset() - strlen($trailing_text),
|
|
|
|
self::LINT_CLOSING_DECL_PAREN,
|
|
|
|
'Convention: no spaces before closing parenthesis in function and '.
|
|
|
|
'method declarations.',
|
|
|
|
$trailing_text,
|
|
|
|
'');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:02:18 +01:00
|
|
|
private function lintKeywordCasing(XHPASTNode $root) {
|
2013-04-15 19:22:56 +02:00
|
|
|
$keywords = array();
|
|
|
|
|
|
|
|
$symbols = $root->selectDescendantsOfType('n_SYMBOL_NAME');
|
|
|
|
foreach ($symbols as $symbol) {
|
|
|
|
$keywords[] = head($symbol->getTokens());
|
|
|
|
}
|
|
|
|
|
|
|
|
$arrays = $root->selectDescendantsOfType('n_ARRAY_LITERAL');
|
|
|
|
foreach ($arrays as $array) {
|
|
|
|
$keywords[] = head($array->getTokens());
|
|
|
|
}
|
|
|
|
|
|
|
|
$typehints = $root->selectDescendantsOfType('n_TYPE_NAME');
|
|
|
|
foreach ($typehints as $typehint) {
|
|
|
|
$keywords[] = head($typehint->getTokens());
|
|
|
|
}
|
|
|
|
|
2013-09-12 22:01:25 +02:00
|
|
|
$new_invocations = $root->selectDescendantsOfType('n_NEW');
|
|
|
|
foreach ($new_invocations as $invocation) {
|
|
|
|
$keywords[] = head($invocation->getTokens());
|
|
|
|
}
|
|
|
|
|
2013-04-15 19:22:56 +02:00
|
|
|
// NOTE: Although PHP generally allows arbitrary casing for all language
|
|
|
|
// keywords, it's exceedingly rare for anyone to type, e.g., "CLASS" or
|
|
|
|
// "cLaSs" in the wild. This list just attempts to cover unconventional
|
|
|
|
// spellings which see some level of use, not all keywords exhaustively.
|
|
|
|
// There is no token or node type which spans all keywords, so this is
|
|
|
|
// significantly simpler.
|
|
|
|
|
|
|
|
static $keyword_map = array(
|
|
|
|
'true' => 'true',
|
|
|
|
'false' => 'false',
|
|
|
|
'null' => 'null',
|
|
|
|
'array' => 'array',
|
2013-09-12 22:01:25 +02:00
|
|
|
'new' => 'new',
|
2013-04-15 19:22:56 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
foreach ($keywords as $keyword) {
|
|
|
|
$value = $keyword->getValue();
|
|
|
|
$value_key = strtolower($value);
|
|
|
|
if (!isset($keyword_map[$value_key])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$expected_spelling = $keyword_map[$value_key];
|
|
|
|
if ($value !== $expected_spelling) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$keyword,
|
|
|
|
self::LINT_KEYWORD_CASING,
|
|
|
|
"Convention: spell keyword '{$value}' as '{$expected_spelling}'.",
|
|
|
|
$expected_spelling);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-02-19 22:50:13 +01:00
|
|
|
|
2014-05-17 04:25:37 +02:00
|
|
|
private function lintStrings(XHPASTNode $root) {
|
|
|
|
$nodes = $root->selectDescendantsOfTypes(array(
|
|
|
|
'n_CONCATENATION_LIST',
|
2014-06-17 17:30:15 +02:00
|
|
|
'n_STRING_SCALAR',
|
|
|
|
));
|
2014-05-17 04:25:37 +02:00
|
|
|
|
|
|
|
foreach ($nodes as $node) {
|
|
|
|
$strings = array();
|
|
|
|
|
|
|
|
if ($node->getTypeName() === 'n_CONCATENATION_LIST') {
|
|
|
|
$strings = $node->selectDescendantsOfType('n_STRING_SCALAR');
|
|
|
|
} else if ($node->getTypeName() === 'n_STRING_SCALAR') {
|
|
|
|
$strings = array($node);
|
|
|
|
|
|
|
|
if ($node->getParentNode()->getTypeName() === 'n_CONCATENATION_LIST') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$valid = false;
|
|
|
|
$invalid_nodes = array();
|
|
|
|
$fixes = array();
|
|
|
|
|
|
|
|
foreach ($strings as $string) {
|
|
|
|
$concrete_string = $string->getConcreteString();
|
|
|
|
$single_quoted = ($concrete_string[0] === "'");
|
|
|
|
$contents = substr($concrete_string, 1, -1);
|
|
|
|
|
|
|
|
// Double quoted strings are allowed when the string contains the
|
|
|
|
// following characters.
|
|
|
|
static $allowed_chars = array(
|
|
|
|
'\n',
|
|
|
|
'\r',
|
|
|
|
'\t',
|
|
|
|
'\v',
|
2014-05-18 15:47:37 +02:00
|
|
|
'\e',
|
|
|
|
'\f',
|
2014-05-17 04:25:37 +02:00
|
|
|
'\'',
|
2014-05-18 15:47:37 +02:00
|
|
|
'\0',
|
|
|
|
'\1',
|
|
|
|
'\2',
|
|
|
|
'\3',
|
|
|
|
'\4',
|
|
|
|
'\5',
|
|
|
|
'\6',
|
|
|
|
'\7',
|
|
|
|
'\x',
|
2014-05-17 04:25:37 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
$contains_special_chars = false;
|
|
|
|
foreach ($allowed_chars as $allowed_char) {
|
|
|
|
if (strpos($contents, $allowed_char) !== false) {
|
|
|
|
$contains_special_chars = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$string->isConstantString()) {
|
|
|
|
$valid = true;
|
|
|
|
} else if ($contains_special_chars && !$single_quoted) {
|
|
|
|
$valid = true;
|
|
|
|
} else if (!$contains_special_chars && !$single_quoted) {
|
|
|
|
$invalid_nodes[] = $string;
|
2014-05-19 16:49:51 +02:00
|
|
|
$fixes[$string->getID()] = "'".str_replace('\"', '"', $contents)."'";
|
2014-05-17 04:25:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$valid) {
|
|
|
|
foreach ($invalid_nodes as $invalid_node) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$invalid_node,
|
|
|
|
self::LINT_DOUBLE_QUOTE,
|
|
|
|
pht(
|
|
|
|
'String does not require double quotes. For consistency, '.
|
|
|
|
'prefer single quotes.'),
|
|
|
|
$fixes[$invalid_node->getID()]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-19 15:33:22 +02:00
|
|
|
protected function lintElseIfStatements(XHPASTNode $root) {
|
|
|
|
$tokens = $root->selectTokensOfType('T_ELSEIF');
|
|
|
|
|
|
|
|
foreach ($tokens as $token) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$token,
|
|
|
|
self::LINT_ELSEIF_USAGE,
|
|
|
|
pht('Usage of `else if` is preferred over `elseif`.'),
|
|
|
|
'else if');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-19 16:48:58 +02:00
|
|
|
protected function lintSemicolons(XHPASTNode $root) {
|
|
|
|
$tokens = $root->selectTokensOfType(';');
|
|
|
|
|
|
|
|
foreach ($tokens as $token) {
|
|
|
|
$prev = $token->getPrevToken();
|
|
|
|
|
|
|
|
if ($prev->isAnyWhitespace()) {
|
|
|
|
$this->raiseLintAtToken(
|
|
|
|
$prev,
|
|
|
|
self::LINT_SEMICOLON_SPACING,
|
|
|
|
pht('Space found before semicolon.'),
|
|
|
|
'');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-05 14:35:34 +02:00
|
|
|
protected function lintLanguageConstructParentheses(XHPASTNode $root) {
|
|
|
|
$nodes = $root->selectDescendantsOfTypes(array(
|
|
|
|
'n_INCLUDE_FILE',
|
|
|
|
'n_ECHO_LIST',
|
|
|
|
));
|
|
|
|
|
|
|
|
foreach ($nodes as $node) {
|
|
|
|
$child = head($node->getChildren());
|
|
|
|
|
|
|
|
if ($child->getTypeName() === 'n_PARENTHETICAL_EXPRESSION') {
|
|
|
|
list($before, $after) = $child->getSurroundingNonsemanticTokens();
|
|
|
|
|
|
|
|
$replace = preg_replace(
|
|
|
|
'/^\((.*)\)$/',
|
|
|
|
'$1',
|
|
|
|
$child->getConcreteString());
|
|
|
|
|
|
|
|
if (!$before) {
|
|
|
|
$replace = ' '.$replace;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$child,
|
|
|
|
self::LINT_LANGUAGE_CONSTRUCT_PAREN,
|
|
|
|
pht('Language constructs do not require parentheses.'),
|
|
|
|
$replace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 14:48:45 +02:00
|
|
|
protected function lintEmptyBlockStatements(XHPASTNode $root) {
|
|
|
|
$nodes = $root->selectDescendantsOfType('n_STATEMENT_LIST');
|
|
|
|
|
|
|
|
foreach ($nodes as $node) {
|
|
|
|
$tokens = $node->getTokens();
|
|
|
|
$token = head($tokens);
|
|
|
|
|
|
|
|
if (count($tokens) <= 2) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Safety check... if the first token isn't an opening brace then
|
|
|
|
// there's nothing to do here.
|
|
|
|
if ($token->getTypeName() != '{') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$only_whitespace = true;
|
|
|
|
for ($token = $token->getNextToken();
|
|
|
|
$token && $token->getTypeName() != '}';
|
|
|
|
$token = $token->getNextToken()) {
|
|
|
|
$only_whitespace = $only_whitespace && $token->isAnyWhitespace();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($tokens) > 2 && $only_whitespace) {
|
|
|
|
$this->raiseLintAtNode(
|
|
|
|
$node,
|
|
|
|
self::LINT_EMPTY_STATEMENT,
|
|
|
|
pht(
|
|
|
|
"Braces for an empty block statement shouldn't ".
|
|
|
|
"contain only whitespace."),
|
|
|
|
'{}');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-28 20:17:45 +01:00
|
|
|
public function getSuperGlobalNames() {
|
|
|
|
return array(
|
|
|
|
'$GLOBALS',
|
|
|
|
'$_SERVER',
|
|
|
|
'$_GET',
|
|
|
|
'$_POST',
|
|
|
|
'$_FILES',
|
|
|
|
'$_COOKIE',
|
|
|
|
'$_SESSION',
|
|
|
|
'$_REQUEST',
|
|
|
|
'$_ENV',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2011-01-10 00:22:25 +01:00
|
|
|
}
|