1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-05 20:48:45 +01:00
psalm/src/Psalm/Checker/Statements/ExpressionChecker.php

1834 lines
69 KiB
PHP
Raw Normal View History

2016-10-22 19:23:18 +02:00
<?php
namespace Psalm\Checker\Statements;
use PhpParser;
use Psalm\Checker\ClassChecker;
use Psalm\Checker\ClassLikeChecker;
use Psalm\Checker\ClosureChecker;
use Psalm\Checker\CommentChecker;
use Psalm\Checker\MethodChecker;
2017-01-07 20:35:07 +01:00
use Psalm\Checker\Statements\Expression\AssertionFinder;
2016-11-01 16:37:58 +01:00
use Psalm\Checker\Statements\Expression\AssignmentChecker;
2016-11-01 19:14:35 +01:00
use Psalm\Checker\Statements\Expression\CallChecker;
use Psalm\Checker\Statements\Expression\FetchChecker;
2016-11-02 07:29:00 +01:00
use Psalm\Checker\StatementsChecker;
2016-10-22 19:23:18 +02:00
use Psalm\Checker\TypeChecker;
use Psalm\CodeLocation;
2016-10-22 19:23:18 +02:00
use Psalm\Config;
use Psalm\Context;
use Psalm\Issue\ForbiddenCode;
use Psalm\Issue\InvalidClone;
use Psalm\Issue\InvalidOperand;
use Psalm\Issue\InvalidScope;
2016-10-22 19:23:18 +02:00
use Psalm\Issue\InvalidStaticVariable;
use Psalm\Issue\MixedOperand;
use Psalm\Issue\NullOperand;
2017-02-11 23:55:08 +01:00
use Psalm\Issue\PossiblyNullOperand;
2016-10-22 19:23:18 +02:00
use Psalm\Issue\PossiblyUndefinedVariable;
use Psalm\Issue\UndefinedVariable;
2016-11-06 01:53:39 +01:00
use Psalm\Issue\UnrecognizedExpression;
2016-11-02 07:29:00 +01:00
use Psalm\IssueBuffer;
2017-01-07 20:35:07 +01:00
use Psalm\StatementsSource;
2016-10-22 19:23:18 +02:00
use Psalm\Type;
use Psalm\Type\Atomic\Generic;
use Psalm\Type\Atomic\ObjectLike;
use Psalm\Type\Atomic\Scalar;
use Psalm\Type\Atomic\TNumeric;
use Psalm\Type\Atomic\TInt;
use Psalm\Type\Atomic\TVoid;
use Psalm\Type\Atomic\TFloat;
use Psalm\Type\Atomic\TString;
use Psalm\Type\Atomic\TBool;
use Psalm\Type\Atomic\TFalse;
use Psalm\Type\Atomic\TNull;
use Psalm\Type\Atomic\TEmpty;
use Psalm\Type\Atomic\TArray;
use Psalm\Type\Atomic\TMixed;
use Psalm\Type\Atomic\TObject;
use Psalm\Type\Atomic\TResource;
use Psalm\Type\Atomic\TCallable;
use Psalm\Type\Atomic\TNamedObject;
use Psalm\Type\Atomic\TGenericObject;
use Psalm\Type\Atomic\TNumericString;
2016-10-22 19:23:18 +02:00
class ExpressionChecker
{
2016-11-02 07:29:00 +01:00
/**
* @var array<string,array<int,string>>
*/
2016-10-22 19:23:18 +02:00
protected static $reflection_functions = [];
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr $stmt
* @param Context $context
* @param bool $array_assignment
* @param Type\Union|null $assignment_key_type
* @param Type\Union|null $assignment_value_type
* @param string|null $assignment_key_value
* @return false|null
2016-10-22 19:23:18 +02:00
*/
public static function analyze(
2016-10-22 19:23:18 +02:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr $stmt,
Context $context,
$array_assignment = false,
Type\Union $assignment_key_type = null,
Type\Union $assignment_value_type = null,
$assignment_key_value = null
) {
if ($stmt instanceof PhpParser\Node\Expr\Variable) {
if (self::analyzeVariable($statements_checker, $stmt, $context, false, null, $array_assignment) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Assign) {
$assignment_type = AssignmentChecker::analyze(
2016-11-01 16:37:58 +01:00
$statements_checker,
$stmt->var,
$stmt->expr,
null,
2016-11-01 16:37:58 +01:00
$context,
(string)$stmt->getDocComment()
);
if ($assignment_type === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-01 16:37:58 +01:00
$stmt->inferredType = $assignment_type;
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\AssignOp) {
if (AssignmentChecker::analyzeAssignmentOperation($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\MethodCall) {
if (CallChecker::analyzeMethodCall($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\StaticCall) {
if (CallChecker::analyzeStaticCall($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\ConstFetch) {
if (FetchChecker::analyzeConstFetch($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\String_) {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = Type::getString();
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\EncapsedStringPart) {
2016-10-22 19:23:18 +02:00
// do nothing
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\MagicConst) {
2016-10-22 19:23:18 +02:00
// do nothing
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\LNumber) {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = Type::getInt();
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\DNumber) {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = Type::getFloat();
2017-01-26 04:02:19 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\UnaryMinus ||
$stmt instanceof PhpParser\Node\Expr\UnaryPlus
) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-07 21:29:47 +01:00
2017-01-26 04:02:19 +01:00
if (!isset($stmt->expr->inferredType)) {
$stmt->inferredType = new Type\Union([new TInt, new TFloat]);
} elseif ($stmt->expr->inferredType->isMixed()) {
$stmt->inferredType = Type::getMixed();
2017-01-26 04:02:19 +01:00
} else {
2017-01-26 04:23:06 +01:00
$acceptable_types = [];
foreach ($stmt->expr->inferredType->types as $type_part) {
if ($type_part instanceof TInt || $type_part instanceof TFloat) {
$acceptable_types[] = $type_part;
} elseif ($type_part instanceof TString) {
$acceptable_types[] = new TInt;
$acceptable_types[] = new TFloat;
} else {
$acceptable_types[] = new TInt;
2017-01-26 04:02:19 +01:00
}
2017-01-26 04:23:06 +01:00
}
$stmt->inferredType = new Type\Union($acceptable_types);
2017-01-26 04:02:19 +01:00
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Isset_) {
self::analyzeIsset($statements_checker, $stmt, $context);
$stmt->inferredType = Type::getBool();
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\ClassConstFetch) {
if (FetchChecker::analyzeClassConstFetch($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\PropertyFetch) {
2017-02-12 01:30:06 +01:00
if (FetchChecker::analyzePropertyFetch($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\StaticPropertyFetch) {
if (FetchChecker::analyzeStaticPropertyFetch($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\BitwiseNot) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp) {
if (self::analyzeBinaryOp($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-12-17 06:48:31 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\PostInc ||
$stmt instanceof PhpParser\Node\Expr\PostDec ||
$stmt instanceof PhpParser\Node\Expr\PreInc ||
$stmt instanceof PhpParser\Node\Expr\PreDec
) {
if (self::analyze($statements_checker, $stmt->var, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-12-25 00:44:11 +01:00
if (isset($stmt->var->inferredType)) {
$stmt->inferredType = clone $stmt->var->inferredType;
} else {
$stmt->inferredType = Type::getMixed();
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\New_) {
if (CallChecker::analyzeNew($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Array_) {
if (self::analyzeArray($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\Encapsed) {
if (self::analyzeEncapsulatedString($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\FuncCall) {
if (CallChecker::analyzeFunctionCall($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Ternary) {
if (self::analyzeTernary($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\BooleanNot) {
if (self::analyzeBooleanNot($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Empty_) {
if (self::analyzeEmpty($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Closure) {
2016-10-22 19:23:18 +02:00
$closure_checker = new ClosureChecker($stmt, $statements_checker->getSource());
if (self::analyzeClosureUses($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$use_context = new Context($context->self);
$use_context->count_references = $statements_checker->getFileChecker()->project_checker->count_references;
2016-10-22 19:23:18 +02:00
if (!$statements_checker->isStatic()) {
2017-01-12 06:54:41 +01:00
if ($context->collect_mutations &&
$context->self &&
ClassChecker::classExtends(
2017-01-12 06:54:41 +01:00
$context->self,
2017-01-07 20:35:07 +01:00
(string)$statements_checker->getFQCLN()
)
2017-01-12 06:54:41 +01:00
) {
$use_context->vars_in_scope['$this'] = clone $context->vars_in_scope['$this'];
} elseif ($context->self) {
$use_context->vars_in_scope['$this'] = new Type\Union([new TNamedObject($context->self)]);
2016-10-22 19:23:18 +02:00
}
}
foreach ($context->vars_in_scope as $var => $type) {
if (strpos($var, '$this->') === 0) {
$use_context->vars_in_scope[$var] = clone $type;
}
}
foreach ($context->vars_possibly_in_scope as $var => $type) {
if (strpos($var, '$this->') === 0) {
$use_context->vars_possibly_in_scope[$var] = true;
}
}
foreach ($stmt->uses as $use) {
// insert the ref into the current context if passed by ref, as whatever we're passing
// the closure to could execute it straight away.
if (!$context->hasVariable('$' . $use->var) && $use->byRef) {
$context->vars_in_scope['$' . $use->var] = Type::getMixed();
}
$use_context->vars_in_scope['$' . $use->var] = $context->hasVariable('$' . $use->var)
2016-11-02 07:29:00 +01:00
? clone $context->vars_in_scope['$' . $use->var]
: Type::getMixed();
2016-10-22 19:23:18 +02:00
$use_context->vars_possibly_in_scope['$' . $use->var] = true;
}
$closure_checker->analyze($use_context);
2016-10-22 19:23:18 +02:00
2016-12-07 20:13:39 +01:00
if (!isset($stmt->inferredType)) {
$stmt->inferredType = Type::getClosure();
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch) {
if (FetchChecker::analyzeArrayAccess(
2016-11-02 07:29:00 +01:00
$statements_checker,
$stmt,
$context,
$array_assignment,
$assignment_key_type,
$assignment_value_type,
$assignment_key_value
) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Int_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
$stmt->inferredType = Type::getInt();
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Double) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
$stmt->inferredType = Type::getFloat();
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Bool_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
$stmt->inferredType = Type::getBool();
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\String_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
$stmt->inferredType = Type::getString();
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Object_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
$stmt->inferredType = Type::getObject();
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Array_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
$stmt->inferredType = Type::getArray();
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Unset_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
$stmt->inferredType = Type::getNull();
} elseif ($stmt instanceof PhpParser\Node\Expr\Clone_) {
self::analyzeClone($statements_checker, $stmt, $context);
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Instanceof_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
if ($stmt->class instanceof PhpParser\Node\Name &&
!in_array($stmt->class->parts[0], ['self', 'static', 'parent'])
) {
2016-10-22 19:23:18 +02:00
if ($context->check_classes) {
2016-11-08 01:16:51 +01:00
$fq_class_name = ClassLikeChecker::getFQCLNFromNameObject(
2016-10-22 19:23:18 +02:00
$stmt->class,
2017-01-07 20:35:07 +01:00
$statements_checker
2016-10-22 19:23:18 +02:00
);
2016-11-08 01:16:51 +01:00
if (ClassLikeChecker::checkFullyQualifiedClassLikeName(
$fq_class_name,
$statements_checker->getFileChecker(),
new CodeLocation($statements_checker->getSource(), $stmt->class),
2016-11-02 07:29:00 +01:00
$statements_checker->getSuppressedIssues()
) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
}
}
2016-11-05 22:57:14 +01:00
$stmt->inferredType = Type::getBool();
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Exit_) {
2017-02-12 01:42:12 +01:00
if ($stmt->expr) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
return false;
}
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Include_) {
$statements_checker->analyzeInclude($stmt, $context);
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Eval_) {
2016-10-22 19:23:18 +02:00
$context->check_classes = false;
$context->check_variables = false;
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\AssignRef) {
if (AssignmentChecker::analyzeAssignmentRef($statements_checker, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\ErrorSuppress) {
2016-10-22 19:23:18 +02:00
// do nothing
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\ShellExec) {
2016-10-22 19:23:18 +02:00
if (IssueBuffer::accepts(
new ForbiddenCode(
'Use of shell_exec',
new CodeLocation($statements_checker->getSource(), $stmt)
),
2016-10-22 19:23:18 +02:00
$statements_checker->getSuppressedIssues()
)) {
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Print_) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Yield_) {
self::analyzeYield($statements_checker, $stmt, $context);
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\YieldFrom) {
self::analyzeYieldFrom($statements_checker, $stmt, $context);
2016-11-02 07:29:00 +01:00
} else {
2016-11-06 01:53:39 +01:00
if (IssueBuffer::accepts(
new UnrecognizedExpression(
'Psalm does not understand ' . get_class($stmt),
new CodeLocation($statements_checker->getSource(), $stmt)
2016-11-06 01:53:39 +01:00
),
$statements_checker->getSuppressedIssues()
)) {
return false;
}
2016-10-22 19:23:18 +02:00
}
$plugins = Config::getInstance()->getPlugins();
if ($plugins) {
$code_location = new CodeLocation($statements_checker->getSource(), $stmt);
foreach ($plugins as $plugin) {
if ($plugin->checkExpression(
$statements_checker,
$stmt,
$context,
$code_location,
$statements_checker->getSuppressedIssues()
) === false) {
return false;
}
2016-10-22 19:23:18 +02:00
}
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Variable $stmt
* @param Context $context
* @param bool $passed_by_reference
* @param Type\Union|null $by_ref_type
* @param bool $array_assignment
* @return false|null
2016-10-22 19:23:18 +02:00
*/
public static function analyzeVariable(
2016-10-22 19:23:18 +02:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Variable $stmt,
Context $context,
$passed_by_reference = false,
Type\Union $by_ref_type = null,
$array_assignment = false
) {
if ($stmt->name === 'this') {
if ($statements_checker->isStatic()) {
if (IssueBuffer::accepts(
new InvalidStaticVariable(
'Invalid reference to $this in a static context',
new CodeLocation($statements_checker->getSource(), $stmt)
),
$statements_checker->getSuppressedIssues()
)) {
return false;
}
return null;
} elseif (!isset($context->vars_in_scope['$this'])) {
if (IssueBuffer::accepts(
new InvalidScope(
'Invalid reference to $this in a non-class context',
new CodeLocation($statements_checker->getSource(), $stmt)
),
$statements_checker->getSuppressedIssues()
)) {
return false;
}
return null;
2016-10-22 19:23:18 +02:00
}
$stmt->inferredType = clone $context->vars_in_scope['$this'];
return null;
2016-10-22 19:23:18 +02:00
}
if (!$context->check_variables) {
$stmt->inferredType = Type::getMixed();
if (is_string($stmt->name) && !$context->hasVariable('$' . $stmt->name)) {
2016-10-22 19:23:18 +02:00
$context->vars_in_scope['$' . $stmt->name] = Type::getMixed();
$context->vars_possibly_in_scope['$' . $stmt->name] = true;
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
if (in_array(
$stmt->name,
[
'_SERVER',
'_GET',
'_POST',
'_COOKIE',
'_REQUEST',
'_FILES',
'_ENV',
'_SESSION',
'GLOBALS',
'argv',
'argc',
]
)
) {
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
if (!is_string($stmt->name)) {
return self::analyze($statements_checker, $stmt->name, $context);
2016-10-22 19:23:18 +02:00
}
if ($passed_by_reference && $by_ref_type) {
self::assignByRefParam($statements_checker, $stmt, $by_ref_type, $context);
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
$var_name = '$' . $stmt->name;
if (!$context->hasVariable($var_name)) {
2016-11-02 07:29:00 +01:00
if (!isset($context->vars_possibly_in_scope[$var_name]) ||
!$statements_checker->getFirstAppearance($var_name)
) {
2016-10-22 19:23:18 +02:00
if ($array_assignment) {
// if we're in an array assignment, let's assign the variable
// because PHP allows it
$context->vars_in_scope[$var_name] = Type::getArray();
$context->vars_possibly_in_scope[$var_name] = true;
// it might have been defined first in another if/else branch
if (!$statements_checker->hasVariable($var_name)) {
$statements_checker->registerVariable(
$var_name,
new CodeLocation($statements_checker, $stmt)
);
}
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
IssueBuffer::add(
2016-11-02 07:29:00 +01:00
new UndefinedVariable(
'Cannot find referenced variable ' . $var_name,
new CodeLocation($statements_checker->getSource(), $stmt)
2016-11-02 07:29:00 +01:00
)
2016-10-22 19:23:18 +02:00
);
return false;
}
}
$first_appearance = $statements_checker->getFirstAppearance($var_name);
if ($first_appearance) {
2016-10-22 19:23:18 +02:00
if (IssueBuffer::accepts(
new PossiblyUndefinedVariable(
2016-11-02 07:29:00 +01:00
'Possibly undefined variable ' . $var_name .', first seen on line ' .
$first_appearance->getLineNumber(),
new CodeLocation($statements_checker->getSource(), $stmt)
2016-10-22 19:23:18 +02:00
),
$statements_checker->getSuppressedIssues()
)) {
return false;
}
}
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = $context->vars_in_scope[$var_name];
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr $stmt
* @param Type\Union $by_ref_type
* @param Context $context
2016-10-22 19:23:18 +02:00
* @return void
*/
2016-11-02 07:29:00 +01:00
public static function assignByRefParam(
StatementsChecker $statements_checker,
PhpParser\Node\Expr $stmt,
Type\Union $by_ref_type,
Context $context
) {
$var_id = self::getVarId(
$stmt,
2016-11-08 01:16:51 +01:00
$statements_checker->getFQCLN(),
2017-01-07 20:35:07 +01:00
$statements_checker
2016-11-02 07:29:00 +01:00
);
2016-10-22 19:23:18 +02:00
if ($var_id) {
if (!$context->hasVariable($var_id)) {
$context->vars_possibly_in_scope[$var_id] = true;
if (!$statements_checker->hasVariable($var_id)) {
$statements_checker->registerVariable($var_id, new CodeLocation($statements_checker, $stmt));
}
} else {
$existing_type = $context->vars_in_scope[$var_id];
if (TypeChecker::isContainedBy(
$existing_type,
$by_ref_type,
$statements_checker->getFileChecker()
) &&
!$existing_type->isNull() &&
(string)$existing_type !== 'array<empty, empty>'
) {
$stmt->inferredType = $context->vars_in_scope[$var_id];
return;
}
}
2016-10-22 19:23:18 +02:00
}
$stmt->inferredType = $by_ref_type;
$context->vars_in_scope[$var_id] = $by_ref_type;
}
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Array_ $stmt
* @param Context $context
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeArray(
2016-10-22 19:23:18 +02:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Array_ $stmt,
Context $context
) {
// if the array is empty, this special type allows us to match any other array type against it
if (empty($stmt->items)) {
$stmt->inferredType = Type::getEmptyArray();
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
/** @var Type\Union|null */
$item_key_type = null;
/** @var Type\Union|null */
$item_value_type = null;
/** @var array<string,Type\Union> */
$property_types = [];
$can_create_objectlike = true;
2016-10-22 19:23:18 +02:00
foreach ($stmt->items as $item) {
if ($item->key) {
if (self::analyze($statements_checker, $item->key, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if (isset($item->key->inferredType)) {
if ($item_key_type) {
/** @var Type\Union */
$item_key_type = Type::combineUnionTypes($item->key->inferredType, $item_key_type);
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
/** @var Type\Union */
$item_key_type = $item->key->inferredType;
}
}
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
$item_key_type = Type::getInt();
}
if (self::analyze($statements_checker, $item->value, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if ($item_value_type && $item_value_type->isMixed() && !$can_create_objectlike) {
continue;
}
2016-10-22 19:23:18 +02:00
if (isset($item->value->inferredType)) {
if ($item->key instanceof PhpParser\Node\Scalar\String_) {
$property_types[$item->key->value] = $item->value->inferredType;
} else {
$can_create_objectlike = false;
2016-10-22 19:23:18 +02:00
}
if ($item_value_type) {
$item_value_type = Type::combineUnionTypes($item->value->inferredType, $item_value_type);
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
$item_value_type = $item->value->inferredType;
}
} else {
$item_value_type = Type::getMixed();
2016-10-22 19:23:18 +02:00
}
}
// if this array looks like an object-like array, let's return that instead
if ($item_value_type &&
$item_key_type &&
$item_key_type->hasString() &&
!$item_key_type->hasInt() &&
$can_create_objectlike
) {
$stmt->inferredType = new Type\Union([new Type\Atomic\ObjectLike($property_types)]);
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
$stmt->inferredType = new Type\Union([
new Type\Atomic\TArray([
$item_key_type ?: new Type\Union([new TInt, new TString]),
$item_value_type ?: Type::getMixed()
])
2016-10-22 19:23:18 +02:00
]);
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\BinaryOp $stmt
* @param Context $context
* @param int $nesting
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeBinaryOp(
2016-10-22 19:23:18 +02:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\BinaryOp $stmt,
Context $context,
$nesting = 0
) {
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Concat && $nesting > 20) {
// ignore deeply-nested string concatenation
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd) {
$if_clauses = TypeChecker::getFormula(
2016-10-22 19:23:18 +02:00
$stmt->left,
2016-11-08 01:16:51 +01:00
$statements_checker->getFQCLN(),
2017-01-07 20:35:07 +01:00
$statements_checker
2016-10-22 19:23:18 +02:00
);
$simplified_clauses = TypeChecker::simplifyCNF(array_merge($context->clauses, $if_clauses));
$left_type_assertions = TypeChecker::getTruthsFromFormula($simplified_clauses);
if (self::analyze($statements_checker, $stmt->left, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$changed_vars = [];
2016-10-22 19:23:18 +02:00
// while in an and, we allow scope to boil over to support
// statements of the form if ($x && $x->foo())
$op_vars_in_scope = TypeChecker::reconcileKeyedTypes(
$left_type_assertions,
$context->vars_in_scope,
$changed_vars,
$statements_checker->getFileChecker(),
new CodeLocation($statements_checker->getSource(), $stmt),
2016-10-22 19:23:18 +02:00
$statements_checker->getSuppressedIssues()
);
if ($op_vars_in_scope === false) {
return false;
}
$op_context = clone $context;
$op_context->vars_in_scope = $op_vars_in_scope;
if (self::analyze($statements_checker, $stmt->right, $op_context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
foreach ($op_context->vars_in_scope as $var => $type) {
2017-02-11 01:10:13 +01:00
if (!isset($context->vars_in_scope[$var])) {
2016-10-22 19:23:18 +02:00
$context->vars_in_scope[$var] = $type;
continue;
}
}
$context->updateChecks($op_context);
2017-02-02 06:45:23 +01:00
$context->referenced_vars = array_merge(
$op_context->referenced_vars,
$context->referenced_vars
);
2016-11-02 07:29:00 +01:00
$context->vars_possibly_in_scope = array_merge(
$op_context->vars_possibly_in_scope,
$context->vars_possibly_in_scope
);
} elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanOr) {
$if_clauses = TypeChecker::getFormula(
2016-10-22 19:23:18 +02:00
$stmt->left,
2016-11-08 01:16:51 +01:00
$statements_checker->getFQCLN(),
2017-01-07 20:35:07 +01:00
$statements_checker
2016-10-22 19:23:18 +02:00
);
$rhs_clauses = TypeChecker::simplifyCNF(
array_merge(
$context->clauses,
TypeChecker::negateFormula($if_clauses)
)
);
$negated_type_assertions = TypeChecker::getTruthsFromFormula($rhs_clauses);
2016-10-22 19:23:18 +02:00
if (self::analyze($statements_checker, $stmt->left, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$changed_vars = [];
2016-10-22 19:23:18 +02:00
// while in an or, we allow scope to boil over to support
// statements of the form if ($x === null || $x->foo())
$op_vars_in_scope = TypeChecker::reconcileKeyedTypes(
$negated_type_assertions,
$context->vars_in_scope,
$changed_vars,
$statements_checker->getFileChecker(),
new CodeLocation($statements_checker->getSource(), $stmt),
2016-10-22 19:23:18 +02:00
$statements_checker->getSuppressedIssues()
);
if ($op_vars_in_scope === false) {
return false;
}
$op_context = clone $context;
$op_context->clauses = $rhs_clauses;
2016-10-22 19:23:18 +02:00
$op_context->vars_in_scope = $op_vars_in_scope;
if (self::analyze($statements_checker, $stmt->right, $op_context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$context->updateChecks($op_context);
2017-02-02 06:45:23 +01:00
$context->referenced_vars = array_merge(
$op_context->referenced_vars,
$context->referenced_vars
);
2016-11-02 07:29:00 +01:00
$context->vars_possibly_in_scope = array_merge(
$op_context->vars_possibly_in_scope,
$context->vars_possibly_in_scope
);
} elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Concat) {
$stmt->inferredType = Type::getString();
if (self::analyze($statements_checker, $stmt->left, $context) === false) {
return false;
2016-10-22 19:23:18 +02:00
}
if (self::analyze($statements_checker, $stmt->right, $context) === false) {
return false;
}
} else {
2016-10-22 19:23:18 +02:00
if ($stmt->left instanceof PhpParser\Node\Expr\BinaryOp) {
if (self::analyzeBinaryOp($statements_checker, $stmt->left, $context, ++$nesting) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} else {
if (self::analyze($statements_checker, $stmt->left, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
}
if ($stmt->right instanceof PhpParser\Node\Expr\BinaryOp) {
if (self::analyzeBinaryOp($statements_checker, $stmt->right, $context, ++$nesting) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-02 07:29:00 +01:00
} else {
if (self::analyze($statements_checker, $stmt->right, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
}
}
// let's do some fun type assignment
if (isset($stmt->left->inferredType) && isset($stmt->right->inferredType)) {
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Plus ||
$stmt instanceof PhpParser\Node\Expr\BinaryOp\Minus ||
$stmt instanceof PhpParser\Node\Expr\BinaryOp\Mod ||
$stmt instanceof PhpParser\Node\Expr\BinaryOp\Mul ||
$stmt instanceof PhpParser\Node\Expr\BinaryOp\Pow
) {
self::analyzeNonDivArithmenticOp(
$statements_checker,
$stmt->left,
$stmt->right,
$stmt,
2016-11-13 21:39:16 +01:00
$result_type
);
if ($result_type) {
$stmt->inferredType = $result_type;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Div
2016-12-12 02:36:27 +01:00
&& ($stmt->left->inferredType->hasInt() || $stmt->left->inferredType->hasFloat())
&& ($stmt->right->inferredType->hasInt() || $stmt->right->inferredType->hasFloat())
2016-10-22 19:23:18 +02:00
) {
$stmt->inferredType = Type::combineUnionTypes(Type::getFloat(), Type::getInt());
} elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Concat) {
self::analyzeConcatOp(
$statements_checker,
$stmt->left,
$stmt->right,
$result_type
);
if ($result_type) {
$stmt->inferredType = $result_type;
}
2016-10-22 19:23:18 +02:00
}
}
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanOr
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\Equal
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\NotEqual
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\Identical
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\NotIdentical
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\Greater
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\GreaterOrEqual
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\Smaller
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\SmallerOrEqual
) {
$stmt->inferredType = Type::getBool();
}
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Spaceship) {
$stmt->inferredType = Type::getInt();
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
2016-11-18 22:13:59 +01:00
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr $left
* @param PhpParser\Node\Expr $right
* @param PhpParser\Node $parent
2016-11-18 22:13:59 +01:00
* @param Type\Union|null &$result_type
* @return void
*/
public static function analyzeNonDivArithmenticOp(
StatementsChecker $statements_checker,
PhpParser\Node\Expr $left,
PhpParser\Node\Expr $right,
PhpParser\Node $parent,
2016-11-13 21:39:16 +01:00
Type\Union &$result_type = null
) {
$left_type = isset($left->inferredType) ? $left->inferredType : null;
$right_type = isset($right->inferredType) ? $right->inferredType : null;
$config = Config::getInstance();
if ($left_type && $right_type) {
foreach ($left_type->types as $left_type_part) {
foreach ($right_type->types as $right_type_part) {
if ($left_type_part instanceof TMixed || $right_type_part instanceof TMixed) {
if ($left_type_part instanceof TMixed) {
if (IssueBuffer::accepts(
new MixedOperand(
'Left operand cannot be mixed',
new CodeLocation($statements_checker->getSource(), $left)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
} else {
if (IssueBuffer::accepts(
new MixedOperand(
'Right operand cannot be mixed',
new CodeLocation($statements_checker->getSource(), $right)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
$result_type = Type::getMixed();
2016-11-13 21:55:06 +01:00
return;
}
2016-11-13 21:39:16 +01:00
if ($left_type_part instanceof TArray || $right_type_part instanceof TArray) {
if ((!$right_type_part instanceof TArray && !$right_type_part instanceof ObjectLike) ||
(!$left_type_part instanceof TArray && !$left_type_part instanceof ObjectLike)) {
if (!$left_type_part instanceof TArray && !$left_type_part instanceof ObjectLike) {
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot add an array to a non-array ' . $left_type_part,
new CodeLocation($statements_checker->getSource(), $left)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
} else {
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot add an array to a non-array ' . $right_type_part,
new CodeLocation($statements_checker->getSource(), $right)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
$result_type = Type::getArray();
return;
}
$result_type_member = Type::combineTypes([$left_type_part, $right_type_part]);
if (!$result_type) {
$result_type = $result_type_member;
} else {
$result_type = Type::combineUnionTypes($result_type_member, $result_type);
}
continue;
2016-11-13 21:55:06 +01:00
}
2016-11-13 21:39:16 +01:00
if ($left_type_part->isNumericType() || $right_type_part->isNumericType()) {
if ($left_type_part instanceof TInt && $right_type_part instanceof TInt) {
if (!$result_type) {
$result_type = Type::getInt();
} else {
$result_type = Type::combineUnionTypes(Type::getInt(), $result_type);
}
continue;
}
if ($left_type_part instanceof TFloat && $right_type_part instanceof TFloat) {
if (!$result_type) {
$result_type = Type::getFloat();
} else {
$result_type = Type::combineUnionTypes(Type::getFloat(), $result_type);
}
continue;
}
if (($left_type_part instanceof TFloat && $right_type_part instanceof TInt) ||
($left_type_part instanceof TInt && $right_type_part instanceof TFloat)
) {
if ($config->strict_binary_operands) {
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot add ints to floats',
new CodeLocation($statements_checker->getSource(), $parent)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
if (!$result_type) {
$result_type = Type::getFloat();
} else {
$result_type = Type::combineUnionTypes(Type::getFloat(), $result_type);
}
continue;
}
if ($left_type_part->isNumericType() && $right_type_part->isNumericType()) {
if ($config->strict_binary_operands) {
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot add numeric types together, please cast explicitly',
new CodeLocation($statements_checker->getSource(), $parent)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
if (!$result_type) {
$result_type = Type::getFloat();
} else {
$result_type = Type::combineUnionTypes(Type::getFloat(), $result_type);
}
continue;
}
$non_numeric_type = $left_type_part->isNumericType() ? $right_type_part : $left_type_part;
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot add a numeric type to a non-numeric type ' . $non_numeric_type,
new CodeLocation($statements_checker->getSource(), $parent)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
2016-11-13 21:39:16 +01:00
}
}
}
}
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr $left
* @param PhpParser\Node\Expr $right
2017-02-12 01:30:06 +01:00
* @param Type\Union|null &$result_type
* @return void
*/
public static function analyzeConcatOp(
StatementsChecker $statements_checker,
PhpParser\Node\Expr $left,
PhpParser\Node\Expr $right,
Type\Union &$result_type = null
) {
2016-12-31 02:05:32 +01:00
$left_type = isset($left->inferredType) ? $left->inferredType : null;
$right_type = isset($right->inferredType) ? $right->inferredType : null;
$config = Config::getInstance();
if ($left_type && $right_type) {
2016-12-29 06:14:06 +01:00
$result_type = Type::getString();
if ($left_type->isMixed() || $right_type->isMixed()) {
if ($left_type->isMixed()) {
if (IssueBuffer::accepts(
new MixedOperand(
'Left operand cannot be mixed',
new CodeLocation($statements_checker->getSource(), $left)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
2016-12-29 06:14:06 +01:00
} else {
if (IssueBuffer::accepts(
new MixedOperand(
'Right operand cannot be mixed',
new CodeLocation($statements_checker->getSource(), $right)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
2016-12-29 06:14:06 +01:00
}
2016-12-29 06:14:06 +01:00
return;
}
2017-02-11 23:55:08 +01:00
if ($left_type->isNull()) {
if (IssueBuffer::accepts(
new NullOperand(
'Cannot concatenate with a ' . $left_type,
new CodeLocation($statements_checker->getSource(), $left)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
2017-02-11 23:55:08 +01:00
return;
}
2017-02-11 23:55:08 +01:00
if ($right_type->isNull()) {
if (IssueBuffer::accepts(
new NullOperand(
'Cannot concatenate with a ' . $right_type,
new CodeLocation($statements_checker->getSource(), $right)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
2017-02-11 23:55:08 +01:00
return;
}
if ($left_type->isNullable()) {
if (IssueBuffer::accepts(
new PossiblyNullOperand(
'Cannot concatenate with a possibly null ' . $left_type,
new CodeLocation($statements_checker->getSource(), $left)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
if ($right_type->isNullable()) {
if (IssueBuffer::accepts(
new PossiblyNullOperand(
'Cannot concatenate with a possibly null ' . $right_type,
new CodeLocation($statements_checker->getSource(), $right)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
2016-12-29 06:14:06 +01:00
$left_type_match = TypeChecker::isContainedBy(
$left_type,
Type::getString(),
$statements_checker->getFileChecker(),
true,
$left_has_scalar_match
2016-12-29 06:14:06 +01:00
);
$right_type_match = TypeChecker::isContainedBy(
$right_type,
Type::getString(),
$statements_checker->getFileChecker(),
true,
$right_has_scalar_match
2016-12-29 06:14:06 +01:00
);
if (!$left_type_match && (!$left_has_scalar_match || $config->strict_binary_operands)) {
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot concatenate with a ' . $left_type,
2016-12-29 06:14:06 +01:00
new CodeLocation($statements_checker->getSource(), $left)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
}
2016-11-13 21:39:16 +01:00
2016-12-29 06:14:06 +01:00
if (!$right_type_match && (!$right_has_scalar_match || $config->strict_binary_operands)) {
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot concatenate with a ' . $right_type,
2016-12-29 06:14:06 +01:00
new CodeLocation($statements_checker->getSource(), $right)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
2016-11-13 21:39:16 +01:00
}
}
}
2016-10-22 19:23:18 +02:00
/**
* @param PhpParser\Node\Expr $stmt
2017-01-07 20:35:07 +01:00
* @param string|null $this_class_name
* @param StatementsSource $source
* @param int|null &$nesting
2016-10-22 19:23:18 +02:00
* @return string|null
*/
public static function getVarId(
PhpParser\Node\Expr $stmt,
$this_class_name,
2017-01-07 20:35:07 +01:00
StatementsSource $source,
2016-10-22 19:23:18 +02:00
&$nesting = null
) {
if ($stmt instanceof PhpParser\Node\Expr\Variable && is_string($stmt->name)) {
return '$' . $stmt->name;
}
if ($stmt instanceof PhpParser\Node\Expr\StaticPropertyFetch
&& is_string($stmt->name)
&& $stmt->class instanceof PhpParser\Node\Name
) {
if (count($stmt->class->parts) === 1 && in_array($stmt->class->parts[0], ['self', 'static', 'parent'])) {
2017-01-07 20:35:07 +01:00
if (!$this_class_name) {
throw new \UnexpectedValueException('$this_class_name should not be null');
}
$fq_class_name = $this_class_name;
2016-11-02 07:29:00 +01:00
} else {
2016-11-08 01:16:51 +01:00
$fq_class_name = ClassLikeChecker::getFQCLNFromNameObject(
2016-11-02 07:29:00 +01:00
$stmt->class,
2017-01-07 20:35:07 +01:00
$source
2016-11-02 07:29:00 +01:00
);
2016-10-22 19:23:18 +02:00
}
return $fq_class_name . '::$' . $stmt->name;
2016-10-22 19:23:18 +02:00
}
if ($stmt instanceof PhpParser\Node\Expr\PropertyFetch && is_string($stmt->name)) {
2017-01-07 20:35:07 +01:00
$object_id = self::getVarId($stmt->var, $this_class_name, $source);
2016-10-22 19:23:18 +02:00
if (!$object_id) {
return null;
}
return $object_id . '->' . $stmt->name;
}
if ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch && $nesting !== null) {
$nesting++;
2017-01-07 20:35:07 +01:00
return self::getVarId($stmt->var, $this_class_name, $source, $nesting);
2016-10-22 19:23:18 +02:00
}
return null;
}
/**
* @param PhpParser\Node\Expr $stmt
2017-01-07 20:35:07 +01:00
* @param string|null $this_class_name
* @param StatementsSource $source
2016-10-22 19:23:18 +02:00
* @return string|null
*/
2016-11-02 07:29:00 +01:00
public static function getArrayVarId(
PhpParser\Node\Expr $stmt,
$this_class_name,
2017-01-07 20:35:07 +01:00
StatementsSource $source
2016-11-02 07:29:00 +01:00
) {
if ($stmt instanceof PhpParser\Node\Expr\Assign) {
2017-01-07 20:35:07 +01:00
return self::getArrayVarId($stmt->var, $this_class_name, $source);
}
2016-11-02 07:29:00 +01:00
if ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch &&
($stmt->dim instanceof PhpParser\Node\Scalar\String_ ||
$stmt->dim instanceof PhpParser\Node\Scalar\LNumber)
2016-11-02 07:29:00 +01:00
) {
2017-01-07 20:35:07 +01:00
$root_var_id = self::getArrayVarId($stmt->var, $this_class_name, $source);
$offset = $stmt->dim instanceof PhpParser\Node\Scalar\String_
? '\'' . $stmt->dim->value . '\''
: $stmt->dim->value;
return $root_var_id ? $root_var_id . '[' . $offset . ']' : null;
2016-10-22 19:23:18 +02:00
}
2017-01-07 20:35:07 +01:00
return self::getVarId($stmt, $this_class_name, $source);
2016-10-22 19:23:18 +02:00
}
2016-11-01 19:14:35 +01:00
/**
* @param Type\Union $return_type
* @param array<int, PhpParser\Node\Arg> $args
* @param string|null $calling_class
* @param string|null $method_id
2016-11-01 19:14:35 +01:00
* @return Type\Union
*/
public static function fleshOutTypes(Type\Union $return_type, array $args, $calling_class = null, $method_id = null)
2016-11-01 19:14:35 +01:00
{
$return_type = clone $return_type;
2016-10-22 19:23:18 +02:00
2016-11-01 19:14:35 +01:00
$new_return_type_parts = [];
2016-10-22 19:23:18 +02:00
2017-02-02 06:45:23 +01:00
foreach ($return_type->types as $return_type_part) {
2016-11-01 19:14:35 +01:00
$new_return_type_parts[] = self::fleshOutAtomicType($return_type_part, $args, $calling_class, $method_id);
2016-10-22 19:23:18 +02:00
}
$fleshed_out_type = new Type\Union($new_return_type_parts);
$fleshed_out_type->from_docblock = $return_type->from_docblock;
return $fleshed_out_type;
2016-10-22 19:23:18 +02:00
}
/**
* @param Type\Atomic &$return_type
* @param array<int, PhpParser\Node\Arg> $args
* @param string|null $calling_class
* @param string|null $method_id
2016-11-01 19:14:35 +01:00
* @return Type\Atomic
2016-10-22 19:23:18 +02:00
*/
2016-11-01 19:14:35 +01:00
protected static function fleshOutAtomicType(Type\Atomic $return_type, array $args, $calling_class, $method_id)
2016-10-22 19:23:18 +02:00
{
if ($return_type instanceof TNamedObject) {
if ($return_type->value === '$this' ||
$return_type->value === 'static' ||
$return_type->value === 'self'
) {
if (!$calling_class) {
throw new \InvalidArgumentException(
'Cannot handle ' . $return_type->value . ' when $calling_class is empty'
);
}
2016-10-22 19:23:18 +02:00
if ($return_type->value === 'static' || !$method_id) {
$return_type->value = $calling_class;
} else {
list(, $method_name) = explode('::', $method_id);
$appearing_method_id = MethodChecker::getAppearingMethodId($calling_class . '::' . $method_name);
2016-12-10 21:08:38 +01:00
$return_type->value = explode('::', (string)$appearing_method_id)[0];
}
} elseif ($return_type->value[0] === '$' && $method_id) {
$method_params = MethodChecker::getMethodParams($method_id);
2016-10-22 19:23:18 +02:00
if (!$method_params) {
throw new \InvalidArgumentException(
'Cannot get method params of ' . $method_id
);
}
foreach ($args as $i => $arg) {
$method_param = $method_params[$i];
2016-10-22 19:23:18 +02:00
if ($return_type->value === '$' . $method_param->name) {
$arg_value = $arg->value;
if ($arg_value instanceof PhpParser\Node\Scalar\String_) {
$return_type->value = preg_replace('/^\\\/', '', $arg_value->value);
}
2016-10-22 19:23:18 +02:00
}
}
if ($return_type->value[0] === '$') {
$return_type = new TMixed;
}
2016-10-22 19:23:18 +02:00
}
}
if ($return_type instanceof Type\Atomic\TArray || $return_type instanceof Type\Atomic\TGenericObject) {
2016-10-22 19:23:18 +02:00
foreach ($return_type->type_params as &$type_param) {
2016-11-01 19:14:35 +01:00
$type_param = self::fleshOutTypes($type_param, $args, $calling_class, $method_id);
2016-10-22 19:23:18 +02:00
}
}
2016-11-01 19:14:35 +01:00
return $return_type;
2016-10-22 19:23:18 +02:00
}
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Closure $stmt
* @param Context $context
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeClosureUses(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Closure $stmt,
Context $context
) {
2016-11-01 19:14:35 +01:00
foreach ($stmt->uses as $use) {
$use_var_id = '$' . $use->var;
if (!$context->hasVariable($use_var_id)) {
2016-11-01 19:14:35 +01:00
if ($use->byRef) {
$context->vars_in_scope[$use_var_id] = Type::getMixed();
$context->vars_possibly_in_scope[$use_var_id] = true;
if (!$statements_checker->hasVariable($use_var_id)) {
$statements_checker->registerVariable($use_var_id, new CodeLocation($statements_checker, $use));
}
2016-10-22 19:23:18 +02:00
return;
}
if (!isset($context->vars_possibly_in_scope[$use_var_id])) {
2016-11-01 19:14:35 +01:00
if ($context->check_variables) {
IssueBuffer::add(
2016-11-02 07:29:00 +01:00
new UndefinedVariable(
'Cannot find referenced variable ' . $use_var_id,
new CodeLocation($statements_checker->getSource(), $use)
2016-11-02 07:29:00 +01:00
)
2016-11-01 19:14:35 +01:00
);
2016-10-22 19:23:18 +02:00
2016-11-01 19:14:35 +01:00
return false;
}
}
2016-10-22 19:23:18 +02:00
$first_appearance = $statements_checker->getFirstAppearance($use_var_id);
if ($first_appearance) {
2016-11-01 19:14:35 +01:00
if (IssueBuffer::accepts(
new PossiblyUndefinedVariable(
'Possibly undefined variable ' . $use_var_id . ', first seen on line ' .
$first_appearance->getLineNumber(),
new CodeLocation($statements_checker->getSource(), $use)
2016-11-01 19:14:35 +01:00
),
$statements_checker->getSuppressedIssues()
)) {
return false;
}
2016-10-22 19:23:18 +02:00
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
2016-11-01 19:14:35 +01:00
if ($context->check_variables) {
2016-10-22 19:23:18 +02:00
IssueBuffer::add(
2016-11-02 07:29:00 +01:00
new UndefinedVariable(
'Cannot find referenced variable ' . $use_var_id,
new CodeLocation($statements_checker->getSource(), $use)
2016-11-02 07:29:00 +01:00
)
2016-10-22 19:23:18 +02:00
);
2016-11-01 19:14:35 +01:00
return false;
}
2016-10-22 19:23:18 +02:00
}
}
2016-11-01 19:14:35 +01:00
2016-11-02 07:29:00 +01:00
return null;
}
2016-11-01 19:14:35 +01:00
2016-10-22 19:23:18 +02:00
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Yield_ $stmt
* @param Context $context
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeYield(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Yield_ $stmt,
Context $context
) {
$type_in_comments = CommentChecker::getTypeFromComment(
(string) $stmt->getDocComment(),
$context,
$statements_checker->getSource()
);
2016-10-22 19:23:18 +02:00
if ($stmt->key) {
if (self::analyze($statements_checker, $stmt->key, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
}
if ($stmt->value) {
if (self::analyze($statements_checker, $stmt->value, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if ($type_in_comments) {
$stmt->inferredType = $type_in_comments;
2016-11-02 07:29:00 +01:00
} elseif (isset($stmt->value->inferredType)) {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = $stmt->value->inferredType;
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = Type::getMixed();
}
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = Type::getNull();
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\YieldFrom $stmt
* @param Context $context
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeYieldFrom(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\YieldFrom $stmt,
Context $context
) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if (isset($stmt->expr->inferredType)) {
$stmt->inferredType = $stmt->expr->inferredType;
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Ternary $stmt
* @param Context $context
* @return false|null
*/
protected static function analyzeTernary(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Ternary $stmt,
Context $context
) {
if (self::analyze($statements_checker, $stmt->cond, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$t_if_context = clone $context;
$if_clauses = TypeChecker::getFormula(
$stmt->cond,
$statements_checker->getFQCLN(),
2017-01-07 20:35:07 +01:00
$statements_checker
);
2016-11-02 07:29:00 +01:00
$ternary_clauses = TypeChecker::simplifyCNF(array_merge($context->clauses, $if_clauses));
$negated_clauses = TypeChecker::negateFormula($if_clauses);
$negated_if_types = TypeChecker::getTruthsFromFormula($negated_clauses);
$reconcilable_if_types = TypeChecker::getTruthsFromFormula($ternary_clauses);
2016-10-22 19:23:18 +02:00
$changed_vars = [];
2016-11-02 07:29:00 +01:00
$t_if_vars_in_scope_reconciled = TypeChecker::reconcileKeyedTypes(
$reconcilable_if_types,
$t_if_context->vars_in_scope,
$changed_vars,
$statements_checker->getFileChecker(),
new CodeLocation($statements_checker->getSource(), $stmt->cond),
2016-11-02 07:29:00 +01:00
$statements_checker->getSuppressedIssues()
);
2016-10-22 19:23:18 +02:00
if ($t_if_vars_in_scope_reconciled === false) {
return false;
}
$t_if_context->vars_in_scope = $t_if_vars_in_scope_reconciled;
if ($stmt->if) {
if (self::analyze($statements_checker, $stmt->if, $t_if_context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if ($context->count_references) {
$context->referenced_vars = array_merge(
$context->referenced_vars,
$t_if_context->referenced_vars
);
}
2016-10-22 19:23:18 +02:00
}
$t_else_context = clone $context;
if ($negated_if_types) {
2016-10-22 19:23:18 +02:00
$t_else_vars_in_scope_reconciled = TypeChecker::reconcileKeyedTypes(
$negated_if_types,
$t_else_context->vars_in_scope,
$changed_vars,
$statements_checker->getFileChecker(),
new CodeLocation($statements_checker->getSource(), $stmt->else),
2016-10-22 19:23:18 +02:00
$statements_checker->getSuppressedIssues()
);
if ($t_else_vars_in_scope_reconciled === false) {
return false;
}
2016-11-02 07:29:00 +01:00
2016-10-22 19:23:18 +02:00
$t_else_context->vars_in_scope = $t_else_vars_in_scope_reconciled;
}
if (self::analyze($statements_checker, $stmt->else, $t_else_context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if ($context->count_references) {
$context->referenced_vars = array_merge(
$context->referenced_vars,
$t_else_context->referenced_vars
);
}
2016-10-22 19:23:18 +02:00
$lhs_type = null;
if ($stmt->if) {
if (isset($stmt->if->inferredType)) {
$lhs_type = $stmt->if->inferredType;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt->cond) {
2016-10-22 19:23:18 +02:00
if (isset($stmt->cond->inferredType)) {
2016-11-02 07:29:00 +01:00
$if_return_type_reconciled = TypeChecker::reconcileTypes(
'!empty',
$stmt->cond->inferredType,
'',
$statements_checker->getFileChecker(),
new CodeLocation($statements_checker->getSource(), $stmt),
2016-11-02 07:29:00 +01:00
$statements_checker->getSuppressedIssues()
);
2016-10-22 19:23:18 +02:00
if ($if_return_type_reconciled === false) {
return false;
}
$lhs_type = $if_return_type_reconciled;
}
}
if (!$lhs_type || !isset($stmt->else->inferredType)) {
$stmt->inferredType = Type::getMixed();
2016-11-02 07:29:00 +01:00
} else {
2016-10-22 19:23:18 +02:00
$stmt->inferredType = Type::combineUnionTypes($lhs_type, $stmt->else->inferredType);
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\BooleanNot $stmt
* @param Context $context
* @return false|null
*/
protected static function analyzeBooleanNot(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\BooleanNot $stmt,
Context $context
) {
$stmt->inferredType = Type::getBool();
2016-12-28 22:29:50 +01:00
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
2016-12-28 22:29:50 +01:00
return false;
}
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Empty_ $stmt
* @param Context $context
* @return false|null
*/
protected static function analyzeEmpty(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Empty_ $stmt,
Context $context
) {
return self::analyze($statements_checker, $stmt->expr, $context);
2016-10-22 19:23:18 +02:00
}
/**
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Scalar\Encapsed $stmt
* @param Context $context
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeEncapsulatedString(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Scalar\Encapsed $stmt,
Context $context
) {
2016-10-22 19:23:18 +02:00
/** @var PhpParser\Node\Expr $part */
foreach ($stmt->parts as $part) {
if (self::analyze($statements_checker, $part, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
}
$stmt->inferredType = Type::getString();
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Isset_ $stmt
* @param Context $context
* @return void
*/
protected static function analyzeIsset(
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Isset_ $stmt,
Context $context
) {
foreach ($stmt->vars as $isset_var) {
if ($isset_var instanceof PhpParser\Node\Expr\PropertyFetch &&
$isset_var->var instanceof PhpParser\Node\Expr\Variable &&
$isset_var->var->name === 'this' &&
is_string($isset_var->name)
) {
$var_id = '$this->' . $isset_var->name;
$context->vars_in_scope[$var_id] = Type::getMixed();
$context->vars_possibly_in_scope[$var_id] = true;
}
self::analyzeIssetVar($statements_checker, $isset_var, $context);
}
}
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr $stmt
* @param Context $context
* @return void
*/
protected static function analyzeIssetVar(
StatementsChecker $statements_checker,
PhpParser\Node\Expr $stmt,
Context $context
) {
if ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch) {
self::analyzeIssetVar($statements_checker, $stmt->var, $context);
if (isset($stmt->dim)) {
self::analyze($statements_checker, $stmt->dim, $context);
}
} elseif ($stmt instanceof PhpParser\Node\Expr\PropertyFetch) {
self::analyzeIssetVar($statements_checker, $stmt->var, $context);
} elseif ($stmt instanceof PhpParser\Node\Expr\Variable) {
if ($context->count_references && is_string($stmt->name)) {
$context->hasVariable('$' . $stmt->name);
}
}
}
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Expr\Clone_ $stmt
* @param Context $context
* @return false|null
*/
protected static function analyzeClone(
StatementsChecker $statements_checker,
PhpParser\Node\Expr\Clone_ $stmt,
Context $context
) {
if (self::analyze($statements_checker, $stmt->expr, $context) === false) {
return false;
}
if (isset($stmt->expr->inferredType)) {
foreach ($stmt->expr->inferredType->types as $clone_type_part) {
2017-01-19 21:00:05 +01:00
if (!$clone_type_part instanceof TNamedObject &&
!$clone_type_part instanceof TObject &&
!$clone_type_part instanceof TMixed
) {
if (IssueBuffer::accepts(
new InvalidClone(
'Cannot clone ' . $clone_type_part,
new CodeLocation($statements_checker->getSource(), $stmt)
),
$statements_checker->getSuppressedIssues()
)) {
return false;
}
return;
}
}
$stmt->inferredType = $stmt->expr->inferredType;
}
}
2016-10-22 19:23:18 +02:00
/**
* @param string $fq_class_name
2016-10-22 19:23:18 +02:00
* @return boolean
*/
public static function isMock($fq_class_name)
2016-10-22 19:23:18 +02:00
{
return in_array($fq_class_name, Config::getInstance()->getMockClasses());
2016-10-22 19:23:18 +02:00
}
2016-10-22 23:35:59 +02:00
2016-11-02 07:29:00 +01:00
/**
* @return void
*/
2016-10-22 23:35:59 +02:00
public static function clearCache()
{
self::$reflection_functions = [];
}
2016-10-22 19:23:18 +02:00
}