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/Expression/BinaryOpAnalyzer.php

486 lines
20 KiB
PHP
Raw Normal View History

2018-01-14 18:09:40 +01:00
<?php
2018-11-06 03:57:36 +01:00
namespace Psalm\Internal\Analyzer\Statements\Expression;
2018-01-14 18:09:40 +01:00
use PhpParser;
2018-11-06 03:57:36 +01:00
use Psalm\Internal\Analyzer\Statements\ExpressionAnalyzer;
use Psalm\Internal\Analyzer\StatementsAnalyzer;
use Psalm\Internal\Codebase\VariableUseGraph;
use Psalm\Internal\ControlFlow\ControlFlowNode;
2018-01-14 18:09:40 +01:00
use Psalm\CodeLocation;
use Psalm\Context;
use Psalm\Issue\ImpureMethodCall;
2020-07-30 17:25:47 +02:00
use Psalm\Issue\InvalidOperand;
2018-01-14 18:09:40 +01:00
use Psalm\IssueBuffer;
use Psalm\Type;
2018-05-05 18:59:30 +02:00
use Psalm\Type\Atomic\TNamedObject;
2018-01-14 18:09:40 +01:00
/**
* @internal
*/
2018-11-06 03:57:36 +01:00
class BinaryOpAnalyzer
2018-01-14 18:09:40 +01:00
{
public static function analyze(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2018-01-14 18:09:40 +01:00
PhpParser\Node\Expr\BinaryOp $stmt,
Context $context,
int $nesting = 0,
bool $from_stmt = false
2020-05-18 21:13:27 +02:00
) : bool {
2020-05-23 03:37:18 +02:00
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Concat && $nesting > 100) {
$statements_analyzer->node_data->setType($stmt, Type::getString());
2020-05-23 03:37:18 +02:00
2018-01-14 18:09:40 +01:00
// ignore deeply-nested string concatenation
2020-05-19 00:57:02 +02:00
return true;
}
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd ||
2018-01-14 18:09:40 +01:00
$stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalAnd
) {
$was_inside_use = $context->inside_use;
$context->inside_use = true;
2020-05-19 05:00:53 +02:00
$expr_result = BinaryOp\AndAnalyzer::analyze(
$statements_analyzer,
2020-05-19 00:57:02 +02:00
$stmt,
$context,
$from_stmt
2018-01-14 18:09:40 +01:00
);
2020-05-19 05:00:53 +02:00
$context->inside_use = $was_inside_use;
2020-05-19 05:00:53 +02:00
$statements_analyzer->node_data->setType($stmt, Type::getBool());
return $expr_result;
2020-05-19 00:57:02 +02:00
}
2018-01-14 18:09:40 +01:00
2020-05-19 00:57:02 +02:00
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanOr ||
2018-01-14 18:09:40 +01:00
$stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalOr
) {
$was_inside_use = $context->inside_use;
$context->inside_use = true;
2020-05-19 05:00:53 +02:00
$expr_result = BinaryOp\OrAnalyzer::analyze(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
2020-05-19 00:57:02 +02:00
$stmt,
$context,
$from_stmt
2018-01-14 18:09:40 +01:00
);
2020-05-19 05:00:53 +02:00
$context->inside_use = $was_inside_use;
2020-05-19 05:00:53 +02:00
$statements_analyzer->node_data->setType($stmt, Type::getBool());
return $expr_result;
2020-05-19 00:57:02 +02:00
}
2018-01-14 18:09:40 +01:00
2020-05-19 00:57:02 +02:00
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Coalesce) {
$expr_result = BinaryOp\CoalesceAnalyzer::analyze(
2020-05-19 00:57:02 +02:00
$statements_analyzer,
$stmt,
$context
);
self::addControlFlow(
$statements_analyzer,
$stmt,
$stmt->left,
$stmt->right,
'coalesce'
);
return $expr_result;
2020-05-19 00:57:02 +02:00
}
2018-01-14 18:09:40 +01:00
2020-05-19 00:57:02 +02:00
if ($stmt->left instanceof PhpParser\Node\Expr\BinaryOp) {
if (self::analyze($statements_analyzer, $stmt->left, $context, $nesting + 1) === false) {
2018-01-14 18:09:40 +01:00
return false;
}
2020-05-19 00:57:02 +02:00
} else {
2018-11-11 18:01:14 +01:00
if (ExpressionAnalyzer::analyze($statements_analyzer, $stmt->left, $context) === false) {
2018-01-14 18:09:40 +01:00
return false;
}
2020-05-19 00:57:02 +02:00
}
2018-01-14 18:09:40 +01:00
2020-05-19 00:57:02 +02:00
if ($stmt->right instanceof PhpParser\Node\Expr\BinaryOp) {
if (self::analyze($statements_analyzer, $stmt->right, $context, $nesting + 1) === false) {
2020-05-19 00:57:02 +02:00
return false;
}
} else {
2018-11-11 18:01:14 +01:00
if (ExpressionAnalyzer::analyze($statements_analyzer, $stmt->right, $context) === false) {
2018-01-14 18:09:40 +01:00
return false;
}
2020-05-19 00:57:02 +02:00
}
2020-05-19 00:57:02 +02:00
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Concat) {
2020-05-19 05:00:53 +02:00
$stmt_type = Type::getString();
2020-05-19 00:57:02 +02:00
BinaryOp\ConcatAnalyzer::analyze(
$statements_analyzer,
$stmt->left,
$stmt->right,
2020-05-19 05:00:53 +02:00
$context,
$result_type
);
2020-05-19 05:00:53 +02:00
if ($result_type) {
$stmt_type = $result_type;
}
if ($statements_analyzer->control_flow_graph
&& ($statements_analyzer->control_flow_graph instanceof VariableUseGraph
|| !\in_array('TaintedInput', $statements_analyzer->getSuppressedIssues()))
2020-05-26 05:28:11 +02:00
) {
2020-05-23 07:11:16 +02:00
$stmt_left_type = $statements_analyzer->node_data->getType($stmt->left);
$stmt_right_type = $statements_analyzer->node_data->getType($stmt->right);
2020-05-24 03:38:09 +02:00
$var_location = new CodeLocation($statements_analyzer, $stmt);
2020-05-23 07:11:16 +02:00
$new_parent_node = ControlFlowNode::getForAssignment('concat', $var_location);
$statements_analyzer->control_flow_graph->addNode($new_parent_node);
2020-05-24 03:38:09 +02:00
$stmt_type->parent_nodes = [
$new_parent_node->id => $new_parent_node
];
2020-05-23 07:11:16 +02:00
2020-05-24 03:38:09 +02:00
if ($stmt_left_type && $stmt_left_type->parent_nodes) {
foreach ($stmt_left_type->parent_nodes as $parent_node) {
$statements_analyzer->control_flow_graph->addPath($parent_node, $new_parent_node, 'concat');
2020-05-24 03:38:09 +02:00
}
2020-05-23 07:11:16 +02:00
}
2020-05-24 03:38:09 +02:00
if ($stmt_right_type && $stmt_right_type->parent_nodes) {
foreach ($stmt_right_type->parent_nodes as $parent_node) {
$statements_analyzer->control_flow_graph->addPath($parent_node, $new_parent_node, 'concat');
2020-05-24 03:38:09 +02:00
}
2020-05-23 07:11:16 +02:00
}
}
2020-05-19 05:00:53 +02:00
$statements_analyzer->node_data->setType($stmt, $stmt_type);
2020-05-19 00:57:02 +02:00
return true;
}
2018-01-14 18:09:40 +01:00
2020-05-19 00:57:02 +02:00
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Spaceship) {
$statements_analyzer->node_data->setType($stmt, Type::getInt());
2018-01-14 18:09:40 +01:00
self::addControlFlow(
$statements_analyzer,
$stmt,
$stmt->left,
$stmt->right,
'<=>'
);
2020-05-19 00:57:02 +02:00
return true;
}
2018-01-14 18:09:40 +01:00
2020-05-19 00:57:02 +02:00
if ($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
) {
$statements_analyzer->node_data->setType($stmt, Type::getBool());
2018-01-14 18:09:40 +01:00
2020-05-19 00:57:02 +02:00
$stmt_left_type = $statements_analyzer->node_data->getType($stmt->left);
$stmt_right_type = $statements_analyzer->node_data->getType($stmt->right);
2018-01-14 18:09:40 +01:00
2020-07-30 17:25:47 +02:00
if (($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)
&& $statements_analyzer->getCodebase()->config->strict_binary_operands
&& $stmt_left_type
&& $stmt_right_type
&& (($stmt_left_type->isSingle() && $stmt_left_type->hasBool())
|| ($stmt_right_type->isSingle() && $stmt_right_type->hasBool()))
) {
if (IssueBuffer::accepts(
new InvalidOperand(
'Cannot compare ' . $stmt_left_type->getId() . ' to ' . $stmt_right_type->getId(),
new CodeLocation($statements_analyzer, $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
if (($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->left instanceof PhpParser\Node\Expr\FuncCall
&& $stmt->left->name instanceof PhpParser\Node\Name
&& $stmt->left->name->parts === ['substr']
&& isset($stmt->left->args[1])
&& $stmt_right_type
&& $stmt_right_type->hasLiteralString()
) {
$from_type = $statements_analyzer->node_data->getType($stmt->left->args[1]->value);
$length_type = isset($stmt->left->args[2])
? ($statements_analyzer->node_data->getType($stmt->left->args[2]->value) ?: Type::getMixed())
: null;
$string_length = null;
if ($from_type && $from_type->isSingleIntLiteral() && $length_type === null) {
$string_length = -$from_type->getSingleIntLiteral()->value;
} elseif ($length_type && $length_type->isSingleIntLiteral()) {
$string_length = $length_type->getSingleIntLiteral()->value;
}
if ($string_length > 0) {
foreach ($stmt_right_type->getAtomicTypes() as $atomic_right_type) {
if ($atomic_right_type instanceof Type\Atomic\TLiteralString) {
2020-07-24 16:51:04 +02:00
if (\strlen($atomic_right_type->value) !== $string_length) {
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Equal
|| $stmt instanceof PhpParser\Node\Expr\BinaryOp\Identical
) {
if ($atomic_right_type->from_docblock) {
if (IssueBuffer::accepts(
new \Psalm\Issue\DocblockTypeContradiction(
$atomic_right_type . ' string length is not ' . $string_length,
2020-09-11 04:44:35 +02:00
new CodeLocation($statements_analyzer, $stmt),
null
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
} else {
if (IssueBuffer::accepts(
new \Psalm\Issue\TypeDoesNotContainType(
$atomic_right_type . ' string length is not ' . $string_length,
2020-09-11 04:44:35 +02:00
new CodeLocation($statements_analyzer, $stmt),
null
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
} else {
if ($atomic_right_type->from_docblock) {
if (IssueBuffer::accepts(
new \Psalm\Issue\RedundantConditionGivenDocblockType(
$atomic_right_type . ' string length is never ' . $string_length,
2020-09-11 04:44:35 +02:00
new CodeLocation($statements_analyzer, $stmt),
null
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
} else {
if (IssueBuffer::accepts(
new \Psalm\Issue\RedundantCondition(
$atomic_right_type . ' string length is never ' . $string_length,
2020-09-11 04:44:35 +02:00
new CodeLocation($statements_analyzer, $stmt),
null
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
}
}
}
}
}
}
}
$codebase = $statements_analyzer->getCodebase();
2020-05-19 00:57:02 +02:00
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Equal
&& $stmt_left_type
&& $stmt_right_type
&& ($context->mutation_free || $codebase->alter_code)
2020-05-19 00:57:02 +02:00
) {
self::checkForImpureEqualityComparison(
$statements_analyzer,
2020-05-19 00:57:02 +02:00
$stmt,
$stmt_left_type,
$stmt_right_type
);
}
2018-01-14 18:09:40 +01:00
self::addControlFlow(
$statements_analyzer,
$stmt,
$stmt->left,
$stmt->right,
'comparison'
);
2020-05-19 00:57:02 +02:00
return true;
}
2019-08-20 04:45:24 +02:00
2020-05-19 00:57:02 +02:00
BinaryOp\NonComparisonOpAnalyzer::analyze(
$statements_analyzer,
$stmt,
$context
);
2019-08-20 04:45:24 +02:00
2020-05-19 00:57:02 +02:00
return true;
}
2019-08-20 04:45:24 +02:00
public static function addControlFlow(
StatementsAnalyzer $statements_analyzer,
PhpParser\Node\Expr $stmt,
PhpParser\Node\Expr $left,
PhpParser\Node\Expr $right,
string $type = 'binaryop'
) : void {
if ($stmt->getLine() === -1) {
throw new \UnexpectedValueException('bad');
}
$result_type = $statements_analyzer->node_data->getType($stmt);
if ($statements_analyzer->control_flow_graph
&& $result_type
) {
$stmt_left_type = $statements_analyzer->node_data->getType($left);
$stmt_right_type = $statements_analyzer->node_data->getType($right);
$var_location = new CodeLocation($statements_analyzer, $stmt);
$new_parent_node = ControlFlowNode::getForAssignment($type, $var_location);
$statements_analyzer->control_flow_graph->addNode($new_parent_node);
$result_type->parent_nodes = [
$new_parent_node->id => $new_parent_node
];
if ($stmt_left_type && $stmt_left_type->parent_nodes) {
foreach ($stmt_left_type->parent_nodes as $parent_node) {
$statements_analyzer->control_flow_graph->addPath($parent_node, $new_parent_node, $type);
}
}
if ($stmt_right_type && $stmt_right_type->parent_nodes) {
foreach ($stmt_right_type->parent_nodes as $parent_node) {
$statements_analyzer->control_flow_graph->addPath($parent_node, $new_parent_node, $type);
}
}
if ($stmt instanceof PhpParser\Node\Expr\AssignOp
&& $statements_analyzer->control_flow_graph instanceof VariableUseGraph
) {
$root_expr = $left;
while ($root_expr instanceof PhpParser\Node\Expr\ArrayDimFetch) {
$root_expr = $root_expr->var;
}
if ($left instanceof PhpParser\Node\Expr\PropertyFetch) {
$statements_analyzer->control_flow_graph->addPath(
$new_parent_node,
new ControlFlowNode('variable-use', 'variable use', null),
'used-by-instance-property'
);
} if ($left instanceof PhpParser\Node\Expr\StaticPropertyFetch) {
$statements_analyzer->control_flow_graph->addPath(
$new_parent_node,
new ControlFlowNode('variable-use', 'variable use', null),
'use-in-static-property'
);
} elseif (!$left instanceof PhpParser\Node\Expr\Variable) {
$statements_analyzer->control_flow_graph->addPath(
$new_parent_node,
new ControlFlowNode('variable-use', 'variable use', null),
'variable-use'
);
}
}
}
}
2020-05-19 00:57:02 +02:00
private static function checkForImpureEqualityComparison(
StatementsAnalyzer $statements_analyzer,
PhpParser\Node\Expr\BinaryOp\Equal $stmt,
Type\Union $stmt_left_type,
Type\Union $stmt_right_type
) : void {
$codebase = $statements_analyzer->getCodebase();
2019-08-20 04:45:24 +02:00
2020-05-19 00:57:02 +02:00
if ($stmt_left_type->hasString() && $stmt_right_type->hasObjectType()) {
foreach ($stmt_right_type->getAtomicTypes() as $atomic_type) {
if ($atomic_type instanceof TNamedObject) {
try {
$storage = $codebase->methods->getStorage(
new \Psalm\Internal\MethodIdentifier(
$atomic_type->value,
'__tostring'
)
);
} catch (\UnexpectedValueException $e) {
continue;
}
if (!$storage->mutation_free) {
if ($statements_analyzer->getSource()
instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer
&& $statements_analyzer->getSource()->track_mutations
) {
$statements_analyzer->getSource()->inferred_has_mutation = true;
$statements_analyzer->getSource()->inferred_impure = true;
} else {
if (IssueBuffer::accepts(
new ImpureMethodCall(
'Cannot call a possibly-mutating method '
. $atomic_type->value . '::__toString from a pure context',
new CodeLocation($statements_analyzer, $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
2019-08-20 04:45:24 +02:00
}
}
}
}
2020-05-19 00:57:02 +02:00
} elseif ($stmt_right_type->hasString() && $stmt_left_type->hasObjectType()) {
foreach ($stmt_left_type->getAtomicTypes() as $atomic_type) {
if ($atomic_type instanceof TNamedObject) {
try {
$storage = $codebase->methods->getStorage(
new \Psalm\Internal\MethodIdentifier(
$atomic_type->value,
'__tostring'
)
);
} catch (\UnexpectedValueException $e) {
continue;
}
if (!$storage->mutation_free) {
if ($statements_analyzer->getSource() instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer
&& $statements_analyzer->getSource()->track_mutations
) {
$statements_analyzer->getSource()->inferred_has_mutation = true;
$statements_analyzer->getSource()->inferred_impure = true;
} else {
if (IssueBuffer::accepts(
new ImpureMethodCall(
'Cannot call a possibly-mutating method '
. $atomic_type->value . '::__toString from a pure context',
new CodeLocation($statements_analyzer, $stmt)
),
$statements_analyzer->getSuppressedIssues()
)) {
// fall through
}
2020-05-19 00:57:02 +02:00
}
}
2018-01-14 18:09:40 +01:00
}
}
}
2018-01-14 18:09:40 +01:00
}
}