1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-15 10:57:08 +01:00
psalm/src/Psalm/Internal/Analyzer/Statements/ExpressionAnalyzer.php

1954 lines
76 KiB
PHP
Raw Normal View History

2016-10-22 19:23:18 +02:00
<?php
2018-11-06 03:57:36 +01:00
namespace Psalm\Internal\Analyzer\Statements;
2016-10-22 19:23:18 +02:00
use PhpParser;
2018-11-06 03:57:36 +01:00
use Psalm\Codebase;
use Psalm\Internal\Analyzer\ClassAnalyzer;
2018-11-06 03:57:36 +01:00
use Psalm\Internal\Analyzer\ClassLikeAnalyzer;
use Psalm\Internal\Analyzer\ClosureAnalyzer;
use Psalm\Internal\Analyzer\CommentAnalyzer;
use Psalm\Internal\Analyzer\FunctionLikeAnalyzer;
2019-02-06 21:52:43 +01:00
use Psalm\Internal\Analyzer\TraitAnalyzer;
2018-11-06 03:57:36 +01:00
use Psalm\Internal\Analyzer\Statements\Expression\ArrayAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\AssertionFinder;
use Psalm\Internal\Analyzer\Statements\Expression\AssignmentAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\BinaryOpAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\CallAnalyzer;
2018-11-06 03:57:36 +01:00
use Psalm\Internal\Analyzer\Statements\Expression\Call\FunctionCallAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Call\MethodCallAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Call\NewAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Call\StaticCallAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\ArrayFetchAnalyzer;
2019-09-14 19:12:54 +02:00
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\ClassConstFetchAnalyzer;
2018-11-06 03:57:36 +01:00
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\ConstFetchAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\PropertyFetchAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\VariableFetchAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\IncludeAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\TernaryAnalyzer;
use Psalm\Internal\Analyzer\StatementsAnalyzer;
use Psalm\CodeLocation;
2016-10-22 19:23:18 +02:00
use Psalm\Config;
use Psalm\Context;
use Psalm\Exception\DocblockParseException;
use Psalm\Internal\FileManipulation\FileManipulationBuffer;
use Psalm\FileSource;
2019-01-30 17:36:21 +01:00
use Psalm\Issue\DuplicateParam;
2016-10-22 19:23:18 +02:00
use Psalm\Issue\ForbiddenCode;
use Psalm\Issue\ImpurePropertyAssignment;
use Psalm\Issue\InvalidCast;
use Psalm\Issue\InvalidClone;
use Psalm\Issue\InvalidDocblock;
2019-04-13 16:11:25 +02:00
use Psalm\Issue\InvalidOperand;
use Psalm\Issue\PossiblyInvalidCast;
2019-04-13 16:11:25 +02:00
use Psalm\Issue\PossiblyInvalidOperand;
2016-10-22 19:23:18 +02:00
use Psalm\Issue\PossiblyUndefinedVariable;
use Psalm\Issue\UndefinedConstant;
2016-10-22 19:23:18 +02:00
use Psalm\Issue\UndefinedVariable;
use Psalm\Issue\UnnecessaryVarAnnotation;
2016-11-06 01:53:39 +01:00
use Psalm\Issue\UnrecognizedExpression;
2016-11-02 07:29:00 +01:00
use Psalm\IssueBuffer;
use Psalm\Storage\FunctionLikeParameter;
2016-10-22 19:23:18 +02:00
use Psalm\Type;
use Psalm\Type\Atomic\ObjectLike;
use Psalm\Type\Atomic\Scalar;
use Psalm\Type\Atomic\TArray;
use Psalm\Type\Atomic\TFloat;
2019-02-22 03:40:06 +01:00
use Psalm\Type\Atomic\TTemplateParam;
use Psalm\Type\Atomic\TInt;
2020-01-11 17:42:09 +01:00
use Psalm\Type\Atomic\TList;
use Psalm\Type\Atomic\TMixed;
use Psalm\Type\Atomic\TNamedObject;
use Psalm\Type\Atomic\TNull;
use Psalm\Type\Atomic\TObject;
use Psalm\Type\Atomic\TString;
use Psalm\Internal\Type\TypeCombination;
use function strpos;
use function is_string;
use function in_array;
use function strtolower;
use function get_class;
use function count;
use function implode;
use function is_array;
use function array_merge;
use function array_values;
use function array_map;
use function current;
2016-10-22 19:23:18 +02:00
/**
* @internal
*/
2018-11-06 03:57:36 +01:00
class ExpressionAnalyzer
2016-10-22 19:23:18 +02:00
{
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Expr $stmt
* @param Context $context
* @param bool $array_assignment
2017-05-27 02:16:18 +02:00
*
2016-11-02 07:29:00 +01:00
* @return false|null
2016-10-22 19:23:18 +02:00
*/
public static function analyze(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-10-22 19:23:18 +02:00
PhpParser\Node\Expr $stmt,
Context $context,
$array_assignment = false,
Context $global_context = null,
bool $from_stmt = false
2016-10-22 19:23:18 +02:00
) {
2018-11-11 18:01:14 +01:00
$codebase = $statements_analyzer->getCodebase();
2018-11-06 03:57:36 +01:00
2016-10-22 19:23:18 +02:00
if ($stmt instanceof PhpParser\Node\Expr\Variable) {
2018-11-06 03:57:36 +01:00
if (VariableFetchAnalyzer::analyze(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
2018-01-14 18:09:40 +01:00
$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) {
2018-11-06 03:57:36 +01:00
$assignment_type = AssignmentAnalyzer::analyze(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
2016-11-01 16:37:58 +01:00
$stmt->var,
$stmt->expr,
null,
2016-11-01 16:37:58 +01:00
$context,
$stmt->getDocComment()
2016-11-01 16:37:58 +01:00
);
if ($assignment_type === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-01 16:37:58 +01:00
if (!$from_stmt) {
$statements_analyzer->node_data->setType($stmt, $assignment_type);
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\AssignOp) {
2018-11-11 18:01:14 +01:00
if (AssignmentAnalyzer::analyzeAssignmentOperation($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (MethodCallAnalyzer::analyze($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (StaticCallAnalyzer::analyze($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
ConstFetchAnalyzer::analyze($statements_analyzer, $stmt, $context);
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\String_) {
$statements_analyzer->node_data->setType($stmt, Type::getString($stmt->value));
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) {
if ($stmt instanceof PhpParser\Node\Scalar\MagicConst\Line) {
$statements_analyzer->node_data->setType($stmt, Type::getInt());
} elseif ($stmt instanceof PhpParser\Node\Scalar\MagicConst\Class_) {
if (!$context->self) {
if (IssueBuffer::accepts(
new UndefinedConstant(
'Cannot get __class__ outside a class',
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
$statements_analyzer->node_data->setType($stmt, Type::getClassString());
} else {
if ($codebase->alter_code) {
2019-06-14 21:55:23 +02:00
$codebase->classlikes->handleClassLikeReferenceInMigration(
$codebase,
$statements_analyzer,
$stmt,
$context->self,
$context->calling_method_id
);
}
$statements_analyzer->node_data->setType($stmt, Type::getLiteralClassString($context->self));
}
} elseif ($stmt instanceof PhpParser\Node\Scalar\MagicConst\Namespace_) {
$namespace = $statements_analyzer->getNamespace();
if ($namespace === null
&& IssueBuffer::accepts(
2019-02-18 19:14:08 +01:00
new UndefinedConstant(
'Cannot get __namespace__ outside a namespace',
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
$statements_analyzer->getSuppressedIssues()
)
) {
// fall through
}
2019-02-18 19:14:08 +01:00
$statements_analyzer->node_data->setType($stmt, Type::getString($namespace));
} elseif ($stmt instanceof PhpParser\Node\Scalar\MagicConst\Method
|| $stmt instanceof PhpParser\Node\Scalar\MagicConst\Function_
) {
$source = $statements_analyzer->getSource();
if ($source instanceof FunctionLikeAnalyzer) {
$statements_analyzer->node_data->setType($stmt, Type::getString($source->getId()));
} else {
$statements_analyzer->node_data->setType($stmt, new Type\Union([new Type\Atomic\TCallableString]));
}
} elseif ($stmt instanceof PhpParser\Node\Scalar\MagicConst\File
|| $stmt instanceof PhpParser\Node\Scalar\MagicConst\Dir
|| $stmt instanceof PhpParser\Node\Scalar\MagicConst\Trait_
) {
$statements_analyzer->node_data->setType($stmt, Type::getString());
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\LNumber) {
$statements_analyzer->node_data->setType($stmt, Type::getInt(false, $stmt->value));
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Scalar\DNumber) {
$statements_analyzer->node_data->setType($stmt, Type::getFloat($stmt->value));
2017-01-26 04:02:19 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\UnaryMinus ||
$stmt instanceof PhpParser\Node\Expr\UnaryPlus
) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2016-11-07 21:29:47 +01:00
if (!($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr))) {
$statements_analyzer->node_data->setType($stmt, new Type\Union([new TInt, new TFloat]));
} elseif ($stmt_expr_type->isMixed()) {
$statements_analyzer->node_data->setType($stmt, Type::getMixed());
2017-01-26 04:02:19 +01:00
} else {
2017-01-26 04:23:06 +01:00
$acceptable_types = [];
foreach ($stmt_expr_type->getAtomicTypes() as $type_part) {
2017-01-26 04:23:06 +01:00
if ($type_part instanceof TInt || $type_part instanceof TFloat) {
if ($type_part instanceof Type\Atomic\TLiteralInt
&& $stmt instanceof PhpParser\Node\Expr\UnaryMinus
) {
$type_part->value = -$type_part->value;
} elseif ($type_part instanceof Type\Atomic\TLiteralFloat
&& $stmt instanceof PhpParser\Node\Expr\UnaryMinus
) {
$type_part->value = -$type_part->value;
}
2017-01-26 04:23:06 +01:00
$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
}
$statements_analyzer->node_data->setType($stmt, 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_) {
2018-11-11 18:01:14 +01:00
self::analyzeIsset($statements_analyzer, $stmt, $context);
$statements_analyzer->node_data->setType($stmt, Type::getBool());
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\ClassConstFetch) {
2019-09-14 19:12:54 +02:00
if (ClassConstFetchAnalyzer::analyze($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (PropertyFetchAnalyzer::analyzeInstance($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (PropertyFetchAnalyzer::analyzeStatic($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2019-04-13 16:11:25 +02:00
if (!($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr))) {
$statements_analyzer->node_data->setType($stmt, new Type\Union([new TInt(), new TString()]));
} elseif ($stmt_expr_type->isMixed()) {
$statements_analyzer->node_data->setType($stmt, Type::getMixed());
2019-04-13 16:11:25 +02:00
} else {
$acceptable_types = [];
$unacceptable_type = null;
$has_valid_operand = false;
foreach ($stmt_expr_type->getAtomicTypes() as $type_string => $type_part) {
2019-04-13 16:11:25 +02:00
if ($type_part instanceof TInt || $type_part instanceof TString) {
if ($type_part instanceof Type\Atomic\TLiteralInt) {
$type_part->value = ~$type_part->value;
} elseif ($type_part instanceof Type\Atomic\TLiteralString) {
$type_part->value = ~$type_part->value;
}
$acceptable_types[] = $type_part;
$has_valid_operand = true;
} elseif ($type_part instanceof TFloat) {
$type_part = ($type_part instanceof Type\Atomic\TLiteralFloat) ?
new Type\Atomic\TLiteralInt(~$type_part->value) :
new TInt;
$stmt_expr_type->removeType($type_string);
$stmt_expr_type->addType($type_part);
2019-04-13 16:11:25 +02:00
$acceptable_types[] = $type_part;
$has_valid_operand = true;
} elseif (!$unacceptable_type) {
$unacceptable_type = $type_part;
}
}
if ($unacceptable_type || !$acceptable_types) {
2019-04-13 16:11:25 +02:00
$message = 'Cannot negate a non-numeric non-string type ' . $unacceptable_type;
if ($has_valid_operand) {
if (IssueBuffer::accepts(
new PossiblyInvalidOperand(
$message,
new CodeLocation($statements_analyzer, $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
} else {
if (IssueBuffer::accepts(
new InvalidOperand(
$message,
new CodeLocation($statements_analyzer, $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
$statements_analyzer->node_data->setType($stmt, Type::getMixed());
2019-04-13 16:11:25 +02:00
} else {
$statements_analyzer->node_data->setType($stmt, new Type\Union($acceptable_types));
2019-04-13 16:11:25 +02:00
}
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp) {
2018-11-06 03:57:36 +01:00
if (BinaryOpAnalyzer::analyze(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
2017-02-13 00:06:18 +01:00
$stmt,
$context,
0,
$from_stmt
2017-02-13 00:06:18 +01:00
) === 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
) {
$was_inside_assignment = $context->inside_assignment;
$context->inside_assignment = true;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->var, $context) === false) {
if (!$was_inside_assignment) {
$context->inside_assignment = false;
}
2016-10-22 19:23:18 +02:00
return false;
}
2016-12-25 00:44:11 +01:00
if (!$was_inside_assignment) {
$context->inside_assignment = false;
}
if ($stmt_var_type = $statements_analyzer->node_data->getType($stmt->var)) {
$return_type = null;
$fake_right_expr = new PhpParser\Node\Scalar\LNumber(1, $stmt->getAttributes());
$statements_analyzer->node_data->setType($fake_right_expr, Type::getInt());
BinaryOpAnalyzer::analyzeNonDivArithmeticOp(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
$statements_analyzer->node_data,
$stmt->var,
$fake_right_expr,
$stmt,
$return_type,
$context
);
$stmt_type = clone $stmt_var_type;
$statements_analyzer->node_data->setType($stmt, $stmt_type);
$stmt_type->from_calculation = true;
foreach ($stmt_type->getAtomicTypes() as $atomic_type) {
2018-05-08 23:42:02 +02:00
if ($atomic_type instanceof Type\Atomic\TLiteralInt) {
$stmt_type->addType(new Type\Atomic\TInt);
2018-05-08 23:42:02 +02:00
} elseif ($atomic_type instanceof Type\Atomic\TLiteralFloat) {
$stmt_type->addType(new Type\Atomic\TFloat);
}
}
$var_id = self::getArrayVarId($stmt->var, null);
2019-08-30 18:36:35 +02:00
if ($var_id && $context->mutation_free && strpos($var_id, '->')) {
if (IssueBuffer::accepts(
new ImpurePropertyAssignment(
2019-08-30 18:36:35 +02:00
'Cannot assign to a property from a mutation-free context',
new CodeLocation($statements_analyzer, $stmt->var)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
if ($var_id && isset($context->vars_in_scope[$var_id])) {
$context->vars_in_scope[$var_id] = $stmt_type;
if ($codebase->find_unused_variables && $stmt->var instanceof PhpParser\Node\Expr\Variable) {
2018-11-11 18:01:14 +01:00
$location = new CodeLocation($statements_analyzer, $stmt->var);
$context->assigned_var_ids[$var_id] = true;
$context->possibly_assigned_var_ids[$var_id] = true;
2018-11-11 18:01:14 +01:00
$statements_analyzer->registerVariableAssignment(
$var_id,
$location
);
$context->unreferenced_vars[$var_id] = [$location->getHash() => $location];
}
// removes dependent vars from $context
$context->removeDescendents(
$var_id,
$context->vars_in_scope[$var_id],
$return_type,
$statements_analyzer
);
}
2016-12-25 00:44:11 +01:00
} else {
$statements_analyzer->node_data->setType($stmt, Type::getMixed());
2016-12-25 00:44:11 +01:00
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\New_) {
2018-11-11 18:01:14 +01:00
if (NewAnalyzer::analyze($statements_analyzer, $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_) {
2018-11-11 18:01:14 +01:00
if (ArrayAnalyzer::analyze($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (self::analyzeEncapsulatedString($statements_analyzer, $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) {
2018-11-06 03:57:36 +01:00
if (FunctionCallAnalyzer::analyze(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
$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) {
2018-11-11 18:01:14 +01:00
if (TernaryAnalyzer::analyze($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (self::analyzeBooleanNot($statements_analyzer, $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_) {
2018-11-11 18:01:14 +01:00
self::analyzeEmpty($statements_analyzer, $stmt, $context);
} elseif ($stmt instanceof PhpParser\Node\Expr\Closure
|| $stmt instanceof PhpParser\Node\Expr\ArrowFunction
) {
2018-11-11 18:01:14 +01:00
$closure_analyzer = new ClosureAnalyzer($stmt, $statements_analyzer);
2016-10-22 19:23:18 +02:00
if ($stmt instanceof PhpParser\Node\Expr\Closure
&& self::analyzeClosureUses($statements_analyzer, $stmt, $context) === false
) {
2016-10-22 19:23:18 +02:00
return false;
}
$use_context = new Context($context->self);
$use_context->mutation_free = $context->mutation_free;
$use_context->external_mutation_free = $context->external_mutation_free;
$use_context->pure = $context->pure;
2016-10-22 19:23:18 +02:00
2018-11-11 18:01:14 +01:00
if (!$statements_analyzer->isStatic()) {
2017-01-12 06:54:41 +01:00
if ($context->collect_mutations &&
$context->self &&
2018-02-01 06:50:01 +01:00
$codebase->classExtends(
2017-01-12 06:54:41 +01:00
$context->self,
2018-11-11 18:01:14 +01:00
(string)$statements_analyzer->getFQCLN()
)
2017-01-12 06:54:41 +01:00
) {
2019-11-11 16:11:42 +01:00
/** @psalm-suppress PossiblyUndefinedStringArrayOffset */
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;
}
}
if ($context->self) {
$self_class_storage = $codebase->classlike_storage_provider->get($context->self);
ClassAnalyzer::addContextProperties(
$statements_analyzer,
$self_class_storage,
$use_context,
2020-03-22 01:53:46 +01:00
$context->self,
$statements_analyzer->getParentFQCLN()
);
}
2018-01-28 18:43:19 +01:00
foreach ($context->vars_possibly_in_scope as $var => $_) {
2016-10-22 19:23:18 +02:00
if (strpos($var, '$this->') === 0) {
$use_context->vars_possibly_in_scope[$var] = true;
}
}
$byref_uses = [];
if ($stmt instanceof PhpParser\Node\Expr\Closure) {
foreach ($stmt->uses as $use) {
if (!is_string($use->var->name)) {
continue;
}
$use_var_id = '$' . $use->var->name;
if ($use->byRef) {
$byref_uses[$use_var_id] = true;
}
// 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_id, $statements_analyzer) && $use->byRef) {
$context->vars_in_scope[$use_var_id] = Type::getMixed();
}
$use_context->vars_in_scope[$use_var_id] =
$context->hasVariable($use_var_id, $statements_analyzer) && !$use->byRef
? clone $context->vars_in_scope[$use_var_id]
: Type::getMixed();
$use_context->vars_possibly_in_scope[$use_var_id] = true;
}
} else {
$traverser = new PhpParser\NodeTraverser;
2020-03-15 04:54:42 +01:00
$short_closure_visitor = new \Psalm\Internal\PhpVisitor\ShortClosureVisitor();
$traverser->addVisitor($short_closure_visitor);
$traverser->traverse($stmt->getStmts());
foreach ($short_closure_visitor->getUsedVariables() as $use_var_id => $_) {
$use_context->vars_in_scope[$use_var_id] =
$context->hasVariable($use_var_id, $statements_analyzer)
? clone $context->vars_in_scope[$use_var_id]
: Type::getMixed();
$use_context->vars_possibly_in_scope[$use_var_id] = true;
}
2016-10-22 19:23:18 +02:00
}
$use_context->calling_method_id = $context->calling_method_id;
$closure_analyzer->analyze($use_context, $statements_analyzer->node_data, $context, false, $byref_uses);
2016-10-22 19:23:18 +02:00
if (!$statements_analyzer->node_data->getType($stmt)) {
$statements_analyzer->node_data->setType($stmt, Type::getClosure());
2016-12-07 20:13:39 +01:00
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch) {
2018-11-06 03:57:36 +01:00
if (ArrayFetchAnalyzer::analyze(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
2016-11-02 07:29:00 +01:00
$stmt,
$context
2016-11-02 07:29:00 +01:00
) === 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_) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$as_int = true;
$maybe_type = $statements_analyzer->node_data->getType($stmt->expr);
if (null !== $maybe_type) {
$maybe = $maybe_type->getAtomicTypes();
if (1 === count($maybe) && current($maybe) instanceof Type\Atomic\TBool) {
$as_int = false;
$statements_analyzer->node_data->setType($stmt, new Type\Union([
new Type\Atomic\TLiteralInt(0),
new Type\Atomic\TLiteralInt(1),
]));
}
}
if ($as_int) {
$statements_analyzer->node_data->setType($stmt, Type::getInt());
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Double) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$statements_analyzer->node_data->setType($stmt, Type::getFloat());
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Bool_) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$statements_analyzer->node_data->setType($stmt, Type::getBool());
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\String_) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if ($statements_analyzer->node_data->getType($stmt->expr)) {
$stmt_type = self::castStringAttempt($statements_analyzer, $context, $stmt->expr, true);
} else {
$stmt_type = Type::getString();
}
$statements_analyzer->node_data->setType($stmt, $stmt_type);
if (($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr))
&& $stmt_expr_type->tainted
) {
$stmt_type->tainted = $stmt_expr_type->tainted;
$stmt_type->sources = $stmt_expr_type->sources;
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Object_) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$statements_analyzer->node_data->setType($stmt, new Type\Union([new TNamedObject('stdClass')]));
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Array_) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$permissible_atomic_types = [];
$all_permissible = false;
if ($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr)) {
$all_permissible = true;
foreach ($stmt_expr_type->getAtomicTypes() as $type) {
if ($type instanceof Scalar) {
$permissible_atomic_types[] = new ObjectLike([new Type\Union([$type])]);
} elseif ($type instanceof TNull) {
$permissible_atomic_types[] = new TArray([Type::getEmpty(), Type::getEmpty()]);
2020-01-11 17:42:09 +01:00
} elseif ($type instanceof TArray
|| $type instanceof TList
|| $type instanceof ObjectLike
) {
$permissible_atomic_types[] = clone $type;
} else {
$all_permissible = false;
break;
}
}
}
if ($permissible_atomic_types && $all_permissible) {
$statements_analyzer->node_data->setType(
$stmt,
TypeCombination::combineTypes($permissible_atomic_types)
);
} else {
$statements_analyzer->node_data->setType($stmt, Type::getArray());
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Cast\Unset_) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$statements_analyzer->node_data->setType($stmt, Type::getNull());
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Clone_) {
2018-11-11 18:01:14 +01:00
self::analyzeClone($statements_analyzer, $stmt, $context);
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Instanceof_) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if ($stmt->class instanceof PhpParser\Node\Expr) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->class, $context) === false) {
return false;
}
} elseif (!in_array(strtolower($stmt->class->parts[0]), ['self', 'static', 'parent'], true)
2016-11-02 07:29:00 +01:00
) {
2016-10-22 19:23:18 +02:00
if ($context->check_classes) {
2018-11-06 03:57:36 +01:00
$fq_class_name = ClassLikeAnalyzer::getFQCLNFromNameObject(
2016-10-22 19:23:18 +02:00
$stmt->class,
2018-11-11 18:01:14 +01:00
$statements_analyzer->getAliases()
2016-10-22 19:23:18 +02:00
);
if ($codebase->store_node_types
&& $fq_class_name
&& !$context->collect_initializations
&& !$context->collect_mutations
) {
$codebase->analyzer->addNodeReference(
$statements_analyzer->getFilePath(),
$stmt->class,
$codebase->classlikes->classOrInterfaceExists($fq_class_name)
? $fq_class_name
: '*' . implode('\\', $stmt->class->parts)
);
}
2018-11-06 03:57:36 +01:00
if (ClassLikeAnalyzer::checkFullyQualifiedClassLikeName(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
$fq_class_name,
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $stmt->class),
$context->self,
$context->calling_method_id,
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues(),
false
2016-11-02 07:29:00 +01:00
) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
2019-02-24 07:33:25 +01:00
2019-06-04 22:36:32 +02:00
if ($codebase->alter_code) {
$codebase->classlikes->handleClassLikeReferenceInMigration(
$codebase,
$statements_analyzer,
$stmt->class,
$fq_class_name,
$context->calling_method_id
2019-06-04 22:36:32 +02:00
);
}
2016-10-22 19:23:18 +02:00
}
}
2016-11-05 22:57:14 +01:00
$statements_analyzer->node_data->setType($stmt, 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) {
$context->inside_call = true;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2017-02-12 01:42:12 +01:00
return false;
}
$context->inside_call = false;
2017-02-12 01:42:12 +01:00
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Include_) {
2018-11-11 18:01:14 +01:00
IncludeAnalyzer::analyze($statements_analyzer, $stmt, $context, $global_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;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $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) {
2018-11-11 18:01:14 +01:00
if (AssignmentAnalyzer::analyzeAssignmentRef($statements_analyzer, $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) {
$context->error_suppressing = true;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
return false;
}
$context->error_suppressing = false;
$expr_type = $statements_analyzer->node_data->getType($stmt->expr);
if ($expr_type) {
$statements_analyzer->node_data->setType($stmt, $expr_type);
}
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',
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues()
2016-10-22 19:23:18 +02:00
)) {
2019-03-23 15:58:26 +01:00
// continue
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Print_) {
$was_inside_call = $context->inside_call;
$context->inside_call = true;
if (self::analyzePrint($statements_analyzer, $stmt, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$context->inside_call = $was_inside_call;
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Yield_) {
2018-11-11 18:01:14 +01:00
self::analyzeYield($statements_analyzer, $stmt, $context);
2016-11-02 07:29:00 +01:00
} elseif ($stmt instanceof PhpParser\Node\Expr\YieldFrom) {
2018-11-11 18:01:14 +01:00
self::analyzeYieldFrom($statements_analyzer, $stmt, $context);
2018-10-07 02:11:19 +02:00
} elseif ($stmt instanceof PhpParser\Node\Expr\Error) {
// do nothing
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),
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $stmt)
2016-11-06 01:53:39 +01:00
),
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues()
2016-11-06 01:53:39 +01:00
)) {
return false;
}
2016-10-22 19:23:18 +02:00
}
if (!$context->inside_conditional
&& ($stmt instanceof PhpParser\Node\Expr\BinaryOp
|| $stmt instanceof PhpParser\Node\Expr\Instanceof_
|| $stmt instanceof PhpParser\Node\Expr\Assign
|| $stmt instanceof PhpParser\Node\Expr\BooleanNot
|| $stmt instanceof PhpParser\Node\Expr\Empty_
|| $stmt instanceof PhpParser\Node\Expr\Isset_
|| $stmt instanceof PhpParser\Node\Expr\FuncCall)
) {
$assertions = $statements_analyzer->node_data->getAssertions($stmt);
if ($assertions === null) {
AssertionFinder::scrapeAssertions(
$stmt,
$context->self,
$statements_analyzer,
$codebase
);
}
}
2018-11-06 03:57:36 +01:00
$plugin_classes = $codebase->config->after_expression_checks;
if ($plugin_classes) {
$file_manipulations = [];
foreach ($plugin_classes as $plugin_fq_class_name) {
2018-11-06 03:57:36 +01:00
if ($plugin_fq_class_name::afterExpressionAnalysis(
$stmt,
$context,
$statements_analyzer,
2018-11-06 03:57:36 +01:00
$codebase,
$file_manipulations
) === false) {
return false;
}
2016-10-22 19:23:18 +02:00
}
if ($file_manipulations) {
2018-11-11 18:01:14 +01:00
FileManipulationBuffer::add($statements_analyzer->getFilePath(), $file_manipulations);
}
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
}
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Expr $stmt
* @param Type\Union $by_ref_type
* @param Context $context
* @param bool $constrain_type
2017-05-27 02:16:18 +02:00
*
2016-10-22 19:23:18 +02:00
* @return void
*/
2016-11-02 07:29:00 +01:00
public static function assignByRefParam(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Expr $stmt,
Type\Union $by_ref_type,
Type\Union $by_ref_out_type,
2017-10-28 21:33:29 +02:00
Context $context,
bool $constrain_type = true,
bool $prevent_null = false
2016-11-02 07:29:00 +01:00
) {
if ($stmt instanceof PhpParser\Node\Expr\PropertyFetch && $stmt->name instanceof PhpParser\Node\Identifier) {
$prop_name = $stmt->name->name;
Expression\Assignment\PropertyAssignmentAnalyzer::analyzeInstance(
$statements_analyzer,
$stmt,
$prop_name,
null,
$by_ref_out_type,
$context
);
return;
}
2016-11-02 07:29:00 +01:00
$var_id = self::getVarId(
$stmt,
2018-11-11 18:01:14 +01:00
$statements_analyzer->getFQCLN(),
$statements_analyzer
2016-11-02 07:29:00 +01:00
);
2016-10-22 19:23:18 +02:00
if ($var_id) {
if (!$by_ref_type->hasMixed() && $constrain_type) {
2018-11-06 03:57:36 +01:00
$context->byref_constraints[$var_id] = new \Psalm\Internal\ReferenceConstraint($by_ref_type);
}
2018-11-11 18:01:14 +01:00
if (!$context->hasVariable($var_id, $statements_analyzer)) {
$context->vars_possibly_in_scope[$var_id] = true;
2018-11-11 18:01:14 +01:00
if (!$statements_analyzer->hasVariable($var_id)) {
$location = new CodeLocation($statements_analyzer, $stmt);
$statements_analyzer->registerVariable($var_id, $location, null);
if ($constrain_type
&& $prevent_null
&& !$by_ref_type->isMixed()
&& !$by_ref_type->isNullable()
&& !strpos($var_id, '->')
&& !strpos($var_id, '::')
) {
if (IssueBuffer::accepts(
new \Psalm\Issue\NullReference(
'Not expecting null argument passed by reference',
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
$codebase = $statements_analyzer->getCodebase();
if ($codebase->find_unused_variables) {
2018-06-17 02:01:33 +02:00
$context->unreferenced_vars[$var_id] = [$location->getHash() => $location];
}
2018-11-11 18:01:14 +01:00
$context->hasVariable($var_id, $statements_analyzer);
}
} elseif ($var_id === '$this') {
// don't allow changing $this
return;
} else {
$existing_type = $context->vars_in_scope[$var_id];
2017-04-02 23:37:56 +02:00
// removes dependent vars from $context
2017-04-02 23:37:56 +02:00
$context->removeDescendents(
$var_id,
$existing_type,
$by_ref_type,
2018-11-11 18:01:14 +01:00
$statements_analyzer
2017-04-02 23:37:56 +02:00
);
if ($existing_type->getId() !== 'array<empty, empty>') {
$context->vars_in_scope[$var_id] = clone $by_ref_out_type;
if (!($stmt_type = $statements_analyzer->node_data->getType($stmt))
|| $stmt_type->isEmpty()
) {
$statements_analyzer->node_data->setType($stmt, clone $by_ref_type);
}
2017-05-25 04:07:49 +02:00
return;
}
}
2019-12-07 21:58:05 +01:00
$context->assigned_var_ids[$var_id] = true;
$context->vars_in_scope[$var_id] = $by_ref_out_type;
2016-10-22 19:23:18 +02:00
if (!($stmt_type = $statements_analyzer->node_data->getType($stmt)) || $stmt_type->isEmpty()) {
$statements_analyzer->node_data->setType($stmt, clone $by_ref_type);
}
}
2016-10-22 19:23:18 +02:00
}
/**
2018-01-14 18:09:40 +01:00
* @param PhpParser\Node\Expr $stmt
* @param string|null $this_class_name
* @param FileSource|null $source
2018-01-14 18:09:40 +01:00
* @param int|null &$nesting
2017-05-27 02:16:18 +02:00
*
2018-01-14 18:09:40 +01:00
* @return string|null
2016-10-22 19:23:18 +02:00
*/
2018-01-14 18:09:40 +01:00
public static function getVarId(
PhpParser\Node\Expr $stmt,
$this_class_name,
FileSource $source = null,
2018-01-14 18:09:40 +01:00
&$nesting = null
2016-10-22 19:23:18 +02:00
) {
2018-01-14 18:09:40 +01:00
if ($stmt instanceof PhpParser\Node\Expr\Variable && is_string($stmt->name)) {
return '$' . $stmt->name;
2016-10-22 19:23:18 +02:00
}
2018-01-14 18:09:40 +01:00
if ($stmt instanceof PhpParser\Node\Expr\StaticPropertyFetch
&& $stmt->name instanceof PhpParser\Node\Identifier
2018-01-14 18:09:40 +01:00
&& $stmt->class instanceof PhpParser\Node\Name
) {
if (count($stmt->class->parts) === 1
&& in_array(strtolower($stmt->class->parts[0]), ['self', 'static', 'parent'], true)
) {
if (!$this_class_name) {
$fq_class_name = $stmt->class->parts[0];
} else {
$fq_class_name = $this_class_name;
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
} else {
2018-01-14 18:09:40 +01:00
$fq_class_name = $source
2018-11-06 03:57:36 +01:00
? ClassLikeAnalyzer::getFQCLNFromNameObject(
2018-01-14 18:09:40 +01:00
$stmt->class,
$source->getAliases()
)
: implode('\\', $stmt->class->parts);
}
return $fq_class_name . '::$' . $stmt->name->name;
2018-01-14 18:09:40 +01:00
}
2016-10-22 19:23:18 +02:00
if ($stmt instanceof PhpParser\Node\Expr\PropertyFetch && $stmt->name instanceof PhpParser\Node\Identifier) {
2018-01-14 18:09:40 +01:00
$object_id = self::getVarId($stmt->var, $this_class_name, $source);
2017-03-20 07:05:58 +01:00
2018-01-14 18:09:40 +01:00
if (!$object_id) {
return null;
2016-10-22 19:23:18 +02:00
}
2018-01-14 18:09:40 +01:00
return $object_id . '->' . $stmt->name->name;
2016-10-22 19:23:18 +02:00
}
2018-01-14 18:09:40 +01:00
if ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch && $nesting !== null) {
++$nesting;
2017-05-25 04:07:49 +02:00
2018-01-14 18:09:40 +01:00
return self::getVarId($stmt->var, $this_class_name, $source, $nesting);
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
}
/**
2018-01-14 18:09:40 +01:00
* @param PhpParser\Node\Expr $stmt
* @param string|null $this_class_name
* @param FileSource|null $source
2017-05-27 02:16:18 +02:00
*
2018-01-14 18:09:40 +01:00
* @return string|null
2016-10-22 19:23:18 +02:00
*/
2018-01-14 18:09:40 +01:00
public static function getRootVarId(
PhpParser\Node\Expr $stmt,
$this_class_name,
FileSource $source = null
2016-10-22 19:23:18 +02:00
) {
2018-01-14 18:09:40 +01:00
if ($stmt instanceof PhpParser\Node\Expr\Variable
|| $stmt instanceof PhpParser\Node\Expr\StaticPropertyFetch
) {
2018-01-14 18:09:40 +01:00
return self::getVarId($stmt, $this_class_name, $source);
}
if ($stmt instanceof PhpParser\Node\Expr\PropertyFetch && $stmt->name instanceof PhpParser\Node\Identifier) {
2018-01-14 18:09:40 +01:00
$property_root = self::getRootVarId($stmt->var, $this_class_name, $source);
2018-01-14 18:09:40 +01:00
if ($property_root) {
return $property_root . '->' . $stmt->name->name;
2016-10-22 19:23:18 +02:00
}
2018-01-14 18:09:40 +01:00
}
2016-10-22 19:23:18 +02:00
2018-01-14 18:09:40 +01:00
if ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch) {
return self::getRootVarId($stmt->var, $this_class_name, $source);
}
2018-01-14 18:09:40 +01:00
return null;
}
2018-01-14 18:09:40 +01:00
/**
* @param PhpParser\Node\Expr $stmt
* @param string|null $this_class_name
* @param FileSource|null $source
2018-01-14 18:09:40 +01:00
*
* @return string|null
*/
public static function getArrayVarId(
PhpParser\Node\Expr $stmt,
$this_class_name,
FileSource $source = null
2018-01-14 18:09:40 +01:00
) {
if ($stmt instanceof PhpParser\Node\Expr\Assign) {
return self::getArrayVarId($stmt->var, $this_class_name, $source);
}
if ($stmt instanceof PhpParser\Node\Expr\ArrayDimFetch) {
2018-01-14 18:09:40 +01:00
$root_var_id = self::getArrayVarId($stmt->var, $this_class_name, $source);
$offset = null;
if ($root_var_id) {
if ($stmt->dim instanceof PhpParser\Node\Scalar\String_
|| $stmt->dim instanceof PhpParser\Node\Scalar\LNumber
) {
$offset = $stmt->dim instanceof PhpParser\Node\Scalar\String_
? '\'' . $stmt->dim->value . '\''
: $stmt->dim->value;
} elseif ($stmt->dim instanceof PhpParser\Node\Expr\Variable
&& is_string($stmt->dim->name)
) {
$offset = '$' . $stmt->dim->name;
} elseif ($stmt->dim instanceof PhpParser\Node\Expr\ConstFetch) {
$offset = implode('\\', $stmt->dim->name->parts);
} elseif ($stmt->dim instanceof PhpParser\Node\Expr\PropertyFetch) {
$object_id = self::getArrayVarId($stmt->dim->var, $this_class_name, $source);
if ($object_id && $stmt->dim->name instanceof PhpParser\Node\Identifier) {
$offset = $object_id . '->' . $stmt->dim->name;
}
} elseif ($stmt->dim instanceof PhpParser\Node\Expr\ClassConstFetch
&& $stmt->dim->name instanceof PhpParser\Node\Identifier
&& $stmt->dim->class instanceof PhpParser\Node\Name
&& $stmt->dim->class->parts[0] === 'static'
) {
$offset = 'static::' . $stmt->dim->name;
} elseif ($stmt->dim
&& $source instanceof StatementsAnalyzer
2020-04-13 03:47:50 +02:00
&& ($stmt_dim_type = $source->node_data->getType($stmt->dim))
&& (!$stmt->dim instanceof PhpParser\Node\Expr\ClassConstFetch
|| !$stmt->dim->name instanceof PhpParser\Node\Identifier
|| $stmt->dim->name->name !== 'class'
)
) {
if ($stmt_dim_type->isSingleStringLiteral()) {
$offset = '\'' . $stmt_dim_type->getSingleStringLiteral()->value . '\'';
} elseif ($stmt_dim_type->isSingleIntLiteral()) {
$offset = $stmt_dim_type->getSingleIntLiteral()->value;
}
} elseif ($stmt->dim instanceof PhpParser\Node\Expr\ClassConstFetch
&& $stmt->dim->name instanceof PhpParser\Node\Identifier
) {
/** @var string|null */
$resolved_name = $stmt->dim->class->getAttribute('resolvedName');
if ($resolved_name) {
$offset = $resolved_name . '::' . $stmt->dim->name;
}
}
return $offset !== null ? $root_var_id . '[' . $offset . ']' : null;
}
2018-01-14 18:09:40 +01:00
}
if ($stmt instanceof PhpParser\Node\Expr\PropertyFetch) {
$object_id = self::getArrayVarId($stmt->var, $this_class_name, $source);
if (!$object_id) {
return null;
}
if ($stmt->name instanceof PhpParser\Node\Identifier) {
return $object_id . '->' . $stmt->name;
} elseif ($source instanceof StatementsAnalyzer
&& ($stmt_name_type = $source->node_data->getType($stmt->name))
&& $stmt_name_type->isSingleStringLiteral()
) {
return $object_id . '->' . $stmt_name_type->getSingleStringLiteral()->value;
} else {
return null;
}
}
if ($stmt instanceof PhpParser\Node\Expr\ClassConstFetch
&& $stmt->name instanceof PhpParser\Node\Identifier
) {
/** @var string|null */
$resolved_name = $stmt->class->getAttribute('resolvedName');
if ($resolved_name) {
if (($resolved_name === 'self' || $resolved_name === 'static') && $this_class_name) {
$resolved_name = $this_class_name;
}
return $resolved_name . '::' . $stmt->name;
}
}
2018-04-28 19:25:44 +02:00
if ($stmt instanceof PhpParser\Node\Expr\MethodCall
&& $stmt->name instanceof PhpParser\Node\Identifier
&& !$stmt->args
) {
$config = \Psalm\Config::getInstance();
if ($config->memoize_method_calls || isset($stmt->pure)) {
2018-04-28 19:25:44 +02:00
$lhs_var_name = self::getArrayVarId(
$stmt->var,
$this_class_name,
$source
);
if (!$lhs_var_name) {
return null;
}
return $lhs_var_name . '->' . strtolower($stmt->name->name) . '()';
}
}
2018-01-14 18:09:40 +01:00
return self::getVarId($stmt, $this_class_name, $source);
}
2016-10-22 19:23:18 +02:00
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Expr\Closure $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
2016-11-02 07:29:00 +01:00
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeClosureUses(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Expr\Closure $stmt,
Context $context
) {
2019-01-30 17:36:21 +01:00
$param_names = array_map(
function (PhpParser\Node\Param $p) : string {
if (!$p->var instanceof PhpParser\Node\Expr\Variable
|| !is_string($p->var->name)
) {
return '';
}
return $p->var->name;
},
$stmt->params
);
2016-11-01 19:14:35 +01:00
foreach ($stmt->uses as $use) {
2018-05-08 23:42:02 +02:00
if (!is_string($use->var->name)) {
continue;
}
$use_var_id = '$' . $use->var->name;
2019-01-30 17:36:21 +01:00
if (in_array($use->var->name, $param_names)) {
if (IssueBuffer::accepts(
new DuplicateParam(
'Closure use duplicates param name ' . $use_var_id,
new CodeLocation($statements_analyzer->getSource(), $use->var)
),
$statements_analyzer->getSuppressedIssues()
)) {
return false;
}
}
2018-11-11 18:01:14 +01:00
if (!$context->hasVariable($use_var_id, $statements_analyzer)) {
2018-05-08 23:43:26 +02:00
if ($use_var_id === '$argv' || $use_var_id === '$argc') {
continue;
}
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;
2018-11-11 18:01:14 +01:00
if (!$statements_analyzer->hasVariable($use_var_id)) {
$statements_analyzer->registerVariable(
$use_var_id,
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer, $use->var),
null
);
}
2017-05-25 04:07:49 +02:00
2016-10-22 19:23:18 +02:00
return;
}
2018-05-08 23:43:26 +02:00
if (!isset($context->vars_possibly_in_scope[$use_var_id])) {
2016-11-01 19:14:35 +01:00
if ($context->check_variables) {
if (IssueBuffer::accepts(
2016-11-02 07:29:00 +01:00
new UndefinedVariable(
'Cannot find referenced variable ' . $use_var_id,
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $use->var)
),
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues()
)) {
return false;
}
2016-10-22 19:23:18 +02:00
2018-05-08 23:43:26 +02:00
return null;
}
}
2018-11-11 18:01:14 +01:00
$first_appearance = $statements_analyzer->getFirstAppearance($use_var_id);
2018-05-08 23:43:26 +02:00
if ($first_appearance) {
if (IssueBuffer::accepts(
new PossiblyUndefinedVariable(
'Possibly undefined variable ' . $use_var_id . ', first seen on line ' .
$first_appearance->getLineNumber(),
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $use->var)
2018-05-08 23:43:26 +02:00
),
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues()
2018-05-08 23:43:26 +02:00
)) {
return false;
}
continue;
}
if ($context->check_variables) {
if (IssueBuffer::accepts(
new UndefinedVariable(
'Cannot find referenced variable ' . $use_var_id,
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $use->var)
2018-05-08 23:43:26 +02:00
),
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues()
2018-05-08 23:43:26 +02:00
)) {
return false;
2016-11-01 19:14:35 +01:00
}
2018-05-08 23:43:26 +02:00
continue;
2016-11-01 19:14:35 +01:00
}
} elseif ($use->byRef) {
$context->remove($use_var_id);
$context->vars_in_scope[$use_var_id] = Type::getMixed();
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
/**
* @param StatementsAnalyzer $statements_analyzer
* @param PhpParser\Node\Expr\Print_ $stmt
* @param Context $context
*
* @return false|null
*/
protected static function analyzePrint(
StatementsAnalyzer $statements_analyzer,
PhpParser\Node\Expr\Print_ $stmt,
Context $context
) {
$codebase = $statements_analyzer->getCodebase();
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
return false;
}
if ($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr)) {
if (CallAnalyzer::checkFunctionArgumentType(
$statements_analyzer,
$stmt_expr_type,
Type::getString(),
null,
'print',
0,
new CodeLocation($statements_analyzer->getSource(), $stmt->expr),
$stmt->expr,
$context,
new FunctionLikeParameter('var', false),
false,
2020-03-17 22:34:45 +01:00
null,
false,
true,
new CodeLocation($statements_analyzer->getSource(), $stmt)
) === false) {
return false;
}
}
if (isset($codebase->config->forbidden_functions['print'])) {
if (IssueBuffer::accepts(
new ForbiddenCode(
'You have forbidden the use of print',
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// continue
}
}
$statements_analyzer->node_data->setType($stmt, Type::getInt(false, 1));
return null;
}
2016-10-22 19:23:18 +02:00
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Expr\Yield_ $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
2016-11-02 07:29:00 +01:00
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeYield(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Expr\Yield_ $stmt,
Context $context
) {
$doc_comment = $stmt->getDocComment();
2017-03-02 04:27:52 +01:00
$var_comments = [];
$var_comment_type = null;
2018-11-11 18:01:14 +01:00
$codebase = $statements_analyzer->getCodebase();
2018-11-06 03:57:36 +01:00
if ($doc_comment) {
try {
2018-11-06 03:57:36 +01:00
$var_comments = CommentAnalyzer::getTypeFromComment(
$doc_comment,
2018-11-11 18:01:14 +01:00
$statements_analyzer,
2019-06-01 18:25:57 +02:00
$statements_analyzer->getAliases()
);
} catch (DocblockParseException $e) {
if (IssueBuffer::accepts(
new InvalidDocblock(
(string)$e->getMessage(),
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $stmt)
)
)) {
// fall through
}
}
foreach ($var_comments as $var_comment) {
if (!$var_comment->type) {
continue;
}
2020-05-11 04:45:01 +02:00
$comment_type = \Psalm\Internal\Type\TypeExpander::expandUnion(
2018-11-06 03:57:36 +01:00
$codebase,
$var_comment->type,
2018-01-26 19:51:00 +01:00
$context->self,
$context->self ? new Type\Atomic\TNamedObject($context->self) : null,
$statements_analyzer->getParentFQCLN()
);
$type_location = null;
if ($var_comment->type_start
&& $var_comment->type_end
&& $var_comment->line_number
) {
$type_location = new CodeLocation\DocblockTypeLocation(
$statements_analyzer,
$var_comment->type_start,
$var_comment->type_end,
$var_comment->line_number
);
}
if (!$var_comment->var_id) {
$var_comment_type = $comment_type;
continue;
}
if ($codebase->find_unused_variables
&& $type_location
&& isset($context->vars_in_scope[$var_comment->var_id])
&& $context->vars_in_scope[$var_comment->var_id]->getId() === $comment_type->getId()
) {
2019-09-26 21:53:41 +02:00
$project_analyzer = $statements_analyzer->getProjectAnalyzer();
if ($codebase->alter_code
&& isset($project_analyzer->getIssuesToFix()['UnnecessaryVarAnnotation'])
) {
FileManipulationBuffer::addVarAnnotationToRemove($type_location);
} elseif (IssueBuffer::accepts(
new UnnecessaryVarAnnotation(
'The @var annotation for ' . $var_comment->var_id . ' is unnecessary',
$type_location
),
[],
true
)) {
// fall through
}
}
$context->vars_in_scope[$var_comment->var_id] = $comment_type;
}
2017-03-02 04:27:52 +01:00
}
2016-10-22 19:23:18 +02:00
if ($stmt->key) {
$context->inside_call = true;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->key, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$context->inside_call = false;
2016-10-22 19:23:18 +02:00
}
if ($stmt->value) {
$context->inside_call = true;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->value, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
$context->inside_call = false;
2016-10-22 19:23:18 +02:00
if ($var_comment_type) {
$expression_type = clone $var_comment_type;
} elseif ($stmt_var_type = $statements_analyzer->node_data->getType($stmt->value)) {
$expression_type = clone $stmt_var_type;
2016-11-02 07:29:00 +01:00
} else {
$expression_type = Type::getMixed();
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
} else {
$expression_type = Type::getEmpty();
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
foreach ($expression_type->getAtomicTypes() as $expression_atomic_type) {
if ($expression_atomic_type instanceof Type\Atomic\TNamedObject) {
$classlike_storage = $codebase->classlike_storage_provider->get($expression_atomic_type->value);
if ($classlike_storage->yield) {
if ($expression_atomic_type instanceof Type\Atomic\TGenericObject) {
$yield_type = PropertyFetchAnalyzer::localizePropertyType(
$codebase,
2020-04-05 05:54:41 +02:00
clone $classlike_storage->yield,
$expression_atomic_type,
$classlike_storage,
$classlike_storage
);
2020-04-05 05:54:41 +02:00
} else {
$yield_type = Type::getMixed();
}
$expression_type->substitute($expression_type, $yield_type);
}
}
}
$statements_analyzer->node_data->setType($stmt, $expression_type);
2019-02-06 21:52:43 +01:00
$source = $statements_analyzer->getSource();
if ($source instanceof FunctionLikeAnalyzer
&& !($source->getSource() instanceof TraitAnalyzer)
) {
$source->examineParamTypes($statements_analyzer, $context, $codebase, $stmt);
2019-02-06 21:52:43 +01:00
$storage = $source->getFunctionLikeStorage($statements_analyzer);
if ($storage->return_type) {
foreach ($storage->return_type->getAtomicTypes() as $atomic_return_type) {
if ($atomic_return_type instanceof Type\Atomic\TNamedObject
2019-02-06 21:52:43 +01:00
&& $atomic_return_type->value === 'Generator'
) {
if ($atomic_return_type instanceof Type\Atomic\TGenericObject) {
if (!$atomic_return_type->type_params[2]->isVoid()) {
$statements_analyzer->node_data->setType(
$stmt,
Type::combineUnionTypes(
clone $atomic_return_type->type_params[2],
$expression_type,
$codebase
)
);
}
} else {
$statements_analyzer->node_data->setType(
$stmt,
Type::combineUnionTypes(
Type::getMixed(),
$expression_type
)
);
2019-02-06 21:52:43 +01:00
}
}
}
}
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Expr\YieldFrom $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
2016-11-02 07:29:00 +01:00
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeYieldFrom(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Expr\YieldFrom $stmt,
Context $context
) {
$was_inside_call = $context->inside_call;
$context->inside_call = true;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
$context->inside_call = $was_inside_call;
2016-10-22 19:23:18 +02:00
return false;
}
if ($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr)) {
$yield_from_type = null;
foreach ($stmt_expr_type->getAtomicTypes() as $atomic_type) {
if ($yield_from_type === null) {
if ($atomic_type instanceof Type\Atomic\TGenericObject
&& strtolower($atomic_type->value) === 'generator'
&& isset($atomic_type->type_params[3])
) {
$yield_from_type = clone $atomic_type->type_params[3];
} elseif ($atomic_type instanceof Type\Atomic\TArray) {
$yield_from_type = clone $atomic_type->type_params[1];
} elseif ($atomic_type instanceof Type\Atomic\ObjectLike) {
$yield_from_type = $atomic_type->getGenericValueType();
}
} else {
$yield_from_type = Type::getMixed();
}
}
// this should be whatever the generator above returns, but *not* the return type
$statements_analyzer->node_data->setType($stmt, $yield_from_type ?: Type::getMixed());
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
$context->inside_call = $was_inside_call;
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
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Expr\BooleanNot $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
2016-11-02 07:29:00 +01:00
* @return false|null
*/
protected static function analyzeBooleanNot(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Expr\BooleanNot $stmt,
Context $context
) {
$statements_analyzer->node_data->setType($stmt, Type::getBool());
2016-12-28 22:29:50 +01:00
2019-03-16 02:50:16 +01:00
$inside_negation = $context->inside_negation;
$context->inside_negation = !$inside_negation;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
2016-12-28 22:29:50 +01:00
return false;
}
2019-03-16 02:50:16 +01:00
$context->inside_negation = $inside_negation;
2016-10-22 19:23:18 +02:00
}
2016-11-02 07:29:00 +01:00
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Expr\Empty_ $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
* @return void
2016-11-02 07:29:00 +01:00
*/
protected static function analyzeEmpty(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Expr\Empty_ $stmt,
Context $context
) {
2018-11-11 18:01:14 +01:00
self::analyzeIssetVar($statements_analyzer, $stmt->expr, $context);
if (($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr))
&& $stmt_expr_type->hasBool()
&& $stmt_expr_type->isSingle()
&& !$stmt_expr_type->from_docblock
) {
if (IssueBuffer::accepts(
new \Psalm\Issue\InvalidArgument(
'Calling empty on a boolean value is almost certainly unintended',
new CodeLocation($statements_analyzer->getSource(), $stmt->expr),
'empty'
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
$statements_analyzer->node_data->setType($stmt, Type::getBool());
2016-10-22 19:23:18 +02:00
}
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
2016-11-02 07:29:00 +01:00
* @param PhpParser\Node\Scalar\Encapsed $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
2016-11-02 07:29:00 +01:00
* @return false|null
2016-10-22 19:23:18 +02:00
*/
protected static function analyzeEncapsulatedString(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Scalar\Encapsed $stmt,
Context $context
) {
2016-10-22 19:23:18 +02:00
foreach ($stmt->parts as $part) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $part, $context) === false) {
2016-10-22 19:23:18 +02:00
return false;
}
if ($statements_analyzer->node_data->getType($part)) {
self::castStringAttempt($statements_analyzer, $context, $part);
}
2016-10-22 19:23:18 +02:00
}
$statements_analyzer->node_data->setType($stmt, Type::getString());
2017-05-25 04:07:49 +02:00
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 19:23:18 +02:00
}
private static function castStringAttempt(
StatementsAnalyzer $statements_analyzer,
Context $context,
PhpParser\Node\Expr $stmt,
bool $explicit_cast = false
) : Type\Union {
$codebase = $statements_analyzer->getCodebase();
if (!($stmt_type = $statements_analyzer->node_data->getType($stmt))) {
return Type::getString();
}
$invalid_casts = [];
$valid_strings = [];
$castable_types = [];
$atomic_types = $stmt_type->getAtomicTypes();
while ($atomic_types) {
2020-02-21 21:44:26 +01:00
$atomic_type = \array_pop($atomic_types);
if ($atomic_type instanceof TString) {
$valid_strings[] = $atomic_type;
continue;
}
if ($atomic_type instanceof TMixed
|| $atomic_type instanceof Type\Atomic\TResource
|| $atomic_type instanceof Type\Atomic\TNull
|| $atomic_type instanceof Type\Atomic\Scalar
) {
$castable_types[] = new TString();
continue;
}
if ($atomic_type instanceof TNamedObject
2020-04-27 02:21:24 +02:00
|| $atomic_type instanceof Type\Atomic\TObjectWithProperties
) {
$intersection_types = [$atomic_type];
if ($atomic_type->extra_types) {
$intersection_types = array_merge($intersection_types, $atomic_type->extra_types);
}
foreach ($intersection_types as $intersection_type) {
if ($intersection_type instanceof TNamedObject) {
$intersection_method_id = new \Psalm\Internal\MethodIdentifier(
$intersection_type->value,
'__tostring'
);
if ($codebase->methods->methodExists(
$intersection_method_id,
$context->calling_method_id,
new CodeLocation($statements_analyzer->getSource(), $stmt)
)) {
$return_type = $codebase->methods->getMethodReturnType(
$intersection_method_id,
$self_class
);
if ($return_type) {
$castable_types = array_merge(
$castable_types,
array_values($return_type->getAtomicTypes())
);
} else {
$castable_types[] = new TString();
}
continue 2;
}
}
if ($intersection_type instanceof Type\Atomic\TObjectWithProperties
&& isset($intersection_type->methods['__toString'])
) {
$castable_types[] = new TString();
continue 2;
}
}
}
if ($atomic_type instanceof Type\Atomic\TTemplateParam) {
$atomic_types = array_merge($atomic_types, $atomic_type->as->getAtomicTypes());
continue;
}
$invalid_casts[] = $atomic_type->getId();
}
if ($invalid_casts) {
if ($valid_strings || $castable_types) {
if (IssueBuffer::accepts(
new PossiblyInvalidCast(
$invalid_casts[0] . ' cannot be cast to string',
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
} else {
if (IssueBuffer::accepts(
new InvalidCast(
$invalid_casts[0] . ' cannot be cast to string',
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
} elseif ($explicit_cast && !$castable_types) {
// todo: emit error here
}
$valid_types = array_merge($valid_strings, $castable_types);
if (!$valid_types) {
return Type::getString();
}
return \Psalm\Internal\Type\TypeCombination::combineTypes(
$valid_types,
$codebase
);
}
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
* @param PhpParser\Node\Expr\Isset_ $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
* @return void
*/
protected static function analyzeIsset(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
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'
&& $isset_var->name instanceof PhpParser\Node\Identifier
) {
$var_id = '$this->' . $isset_var->name->name;
if (!isset($context->vars_in_scope[$var_id])) {
$context->vars_in_scope[$var_id] = Type::getMixed();
$context->vars_possibly_in_scope[$var_id] = true;
}
}
2018-11-11 18:01:14 +01:00
self::analyzeIssetVar($statements_analyzer, $isset_var, $context);
}
2018-04-09 16:20:13 +02:00
$statements_analyzer->node_data->setType($stmt, Type::getBool());
}
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
* @param PhpParser\Node\Expr $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
* @return false|null
*/
protected static function analyzeIssetVar(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
PhpParser\Node\Expr $stmt,
Context $context
) {
2018-05-01 04:13:13 +02:00
$context->inside_isset = true;
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt, $context) === false) {
return false;
}
$context->inside_isset = false;
}
/**
2018-11-11 18:01:14 +01:00
* @param StatementsAnalyzer $statements_analyzer
* @param PhpParser\Node\Expr\Clone_ $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
* @return false|null
*/
protected static function analyzeClone(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
PhpParser\Node\Expr\Clone_ $stmt,
Context $context
) {
2018-11-11 18:01:14 +01:00
if (self::analyze($statements_analyzer, $stmt->expr, $context) === false) {
return false;
}
$stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr);
if ($stmt_expr_type) {
$clone_type = $stmt_expr_type;
$immutable_cloned = false;
foreach ($clone_type->getAtomicTypes() as $clone_type_part) {
if (!$clone_type_part instanceof TNamedObject
&& !$clone_type_part instanceof TObject
&& !$clone_type_part instanceof TMixed
2019-02-22 03:40:06 +01:00
&& !$clone_type_part instanceof TTemplateParam
2017-01-19 21:00:05 +01:00
) {
if ($clone_type_part instanceof Type\Atomic\TFalse
&& $clone_type->ignore_falsable_issues
) {
continue;
}
if ($clone_type_part instanceof Type\Atomic\TNull
&& $clone_type->ignore_nullable_issues
) {
continue;
}
if (IssueBuffer::accepts(
new InvalidClone(
'Cannot clone ' . $clone_type_part,
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $stmt)
),
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues()
)) {
return false;
}
return;
}
2020-03-14 06:09:12 +01:00
if ($clone_type_part instanceof TNamedObject) {
$immutable_cloned = true;
}
}
$statements_analyzer->node_data->setType($stmt, $stmt_expr_type);
if ($immutable_cloned) {
$stmt_expr_type = clone $stmt_expr_type;
$statements_analyzer->node_data->setType($stmt, $stmt_expr_type);
2020-03-14 06:09:12 +01:00
$stmt_expr_type->reference_free = true;
$stmt_expr_type->allow_mutations = true;
}
}
}
2016-10-22 19:23:18 +02:00
/**
* @param string $fq_class_name
2017-05-27 02:16:18 +02:00
*
2017-05-25 04:07:49 +02:00
* @return bool
2016-10-22 19:23:18 +02:00
*/
public static function isMock($fq_class_name)
2016-10-22 19:23:18 +02:00
{
return in_array(strtolower($fq_class_name), Config::getInstance()->getMockClasses(), true);
2016-10-22 19:23:18 +02:00
}
}