1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-10 15:09:04 +01:00
psalm/src/Psalm/Internal/Analyzer/Statements/Expression/BinaryOp/CoalesceAnalyzer.php

90 lines
2.7 KiB
PHP

<?php
namespace Psalm\Internal\Analyzer\Statements\Expression\BinaryOp;
use PhpParser;
use Psalm\Context;
use Psalm\Internal\Analyzer\Statements\ExpressionAnalyzer;
use Psalm\Internal\Analyzer\StatementsAnalyzer;
use Psalm\Node\Expr\VirtualIsset;
use Psalm\Node\Expr\VirtualTernary;
use Psalm\Node\Expr\VirtualVariable;
use Psalm\Type;
use function substr;
/**
* @internal
*/
class CoalesceAnalyzer
{
public static function analyze(
StatementsAnalyzer $statements_analyzer,
PhpParser\Node\Expr\BinaryOp\Coalesce $stmt,
Context $context
): bool {
$left_expr = $stmt->left;
$root_expr = $left_expr;
while ($root_expr instanceof PhpParser\Node\Expr\ArrayDimFetch
|| $root_expr instanceof PhpParser\Node\Expr\PropertyFetch
) {
$root_expr = $root_expr->var;
}
if ($root_expr instanceof PhpParser\Node\Expr\FuncCall
|| $root_expr instanceof PhpParser\Node\Expr\MethodCall
|| $root_expr instanceof PhpParser\Node\Expr\StaticCall
|| $root_expr instanceof PhpParser\Node\Expr\Cast
|| $root_expr instanceof PhpParser\Node\Expr\NullsafePropertyFetch
|| $root_expr instanceof PhpParser\Node\Expr\NullsafeMethodCall
|| $root_expr instanceof PhpParser\Node\Expr\Ternary
) {
$left_var_id = '$<tmp coalesce var>' . (int) $left_expr->getAttribute('startFilePos');
$cloned = clone $context;
$cloned->inside_isset = true;
ExpressionAnalyzer::analyze($statements_analyzer, $left_expr, $cloned);
$condition_type = $statements_analyzer->node_data->getType($left_expr) ?? Type::getMixed();
if ($root_expr !== $left_expr) {
$condition_type->possibly_undefined = true;
}
$context->vars_in_scope[$left_var_id] = $condition_type;
$left_expr = new VirtualVariable(
substr($left_var_id, 1),
$left_expr->getAttributes()
);
}
$ternary = new VirtualTernary(
new VirtualIsset(
[$left_expr],
$stmt->left->getAttributes()
),
$left_expr,
$stmt->right,
$stmt->getAttributes()
);
$old_node_data = $statements_analyzer->node_data;
$statements_analyzer->node_data = clone $statements_analyzer->node_data;
ExpressionAnalyzer::analyze($statements_analyzer, $ternary, $context);
$ternary_type = $statements_analyzer->node_data->getType($ternary) ?? Type::getMixed();
$statements_analyzer->node_data = $old_node_data;
$statements_analyzer->node_data->setType($stmt, $ternary_type);
return true;
}
}