2020-08-30 22:08:22 +02:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2020-08-30 22:08:22 +02:00
|
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression;
|
|
|
|
|
|
|
|
use PhpParser;
|
2021-12-03 20:11:20 +01:00
|
|
|
use Psalm\CodeLocation;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Context;
|
2021-12-03 20:11:20 +01:00
|
|
|
use Psalm\Internal\Algebra;
|
2020-11-03 22:15:44 +01:00
|
|
|
use Psalm\Internal\Algebra\FormulaGenerator;
|
2020-08-30 22:08:22 +02:00
|
|
|
use Psalm\Internal\Analyzer\Statements\ExpressionAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
2020-09-01 04:59:47 +02:00
|
|
|
use Psalm\Issue\UnhandledMatchCondition;
|
2021-12-03 20:11:20 +01:00
|
|
|
use Psalm\IssueBuffer;
|
2021-02-15 22:18:41 +01:00
|
|
|
use Psalm\Node\Expr\BinaryOp\VirtualIdentical;
|
|
|
|
use Psalm\Node\Expr\VirtualArray;
|
|
|
|
use Psalm\Node\Expr\VirtualArrayItem;
|
|
|
|
use Psalm\Node\Expr\VirtualConstFetch;
|
|
|
|
use Psalm\Node\Expr\VirtualFuncCall;
|
|
|
|
use Psalm\Node\Expr\VirtualNew;
|
|
|
|
use Psalm\Node\Expr\VirtualTernary;
|
|
|
|
use Psalm\Node\Expr\VirtualThrow;
|
|
|
|
use Psalm\Node\Expr\VirtualVariable;
|
|
|
|
use Psalm\Node\Name\VirtualFullyQualified;
|
|
|
|
use Psalm\Node\VirtualArg;
|
2020-08-30 22:08:22 +02:00
|
|
|
use Psalm\Type;
|
2021-12-13 04:45:57 +01:00
|
|
|
use Psalm\Type\Atomic\TEnumCase;
|
|
|
|
use Psalm\Type\Atomic\TLiteralFloat;
|
|
|
|
use Psalm\Type\Atomic\TLiteralInt;
|
|
|
|
use Psalm\Type\Atomic\TLiteralString;
|
2021-12-03 20:11:20 +01:00
|
|
|
use Psalm\Type\Reconciler;
|
2021-12-03 21:40:18 +01:00
|
|
|
use UnexpectedValueException;
|
2020-09-22 07:10:46 +02:00
|
|
|
|
2021-12-03 21:07:25 +01:00
|
|
|
use function array_filter;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function array_map;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function array_merge;
|
2020-08-31 00:29:28 +02:00
|
|
|
use function array_reverse;
|
|
|
|
use function array_shift;
|
|
|
|
use function count;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function in_array;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function spl_object_id;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function substr;
|
2020-08-30 22:08:22 +02:00
|
|
|
|
2022-01-03 07:55:32 +01:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2020-08-30 22:08:22 +02:00
|
|
|
class MatchAnalyzer
|
|
|
|
{
|
|
|
|
public static function analyze(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
PhpParser\Node\Expr\Match_ $stmt,
|
|
|
|
Context $context
|
2021-12-05 18:51:26 +01:00
|
|
|
): bool {
|
2020-08-30 22:08:22 +02:00
|
|
|
$was_inside_call = $context->inside_call;
|
|
|
|
|
|
|
|
$context->inside_call = true;
|
|
|
|
|
|
|
|
$was_inside_conditional = $context->inside_conditional;
|
|
|
|
|
|
|
|
$context->inside_conditional = true;
|
|
|
|
|
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $stmt->cond, $context) === false) {
|
|
|
|
$context->inside_conditional = $was_inside_conditional;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$context->inside_conditional = $was_inside_conditional;
|
|
|
|
|
2022-02-04 18:49:12 +01:00
|
|
|
$switch_var_id = ExpressionIdentifier::getExtendedVarId(
|
2020-08-30 22:08:22 +02:00
|
|
|
$stmt->cond,
|
|
|
|
null,
|
|
|
|
$statements_analyzer
|
|
|
|
);
|
|
|
|
|
|
|
|
$match_condition = $stmt->cond;
|
|
|
|
|
2020-11-13 17:55:42 +01:00
|
|
|
if (!$switch_var_id) {
|
|
|
|
if ($stmt->cond instanceof PhpParser\Node\Expr\FuncCall
|
|
|
|
&& $stmt->cond->name instanceof PhpParser\Node\Name
|
|
|
|
&& ($stmt->cond->name->parts === ['get_class']
|
|
|
|
|| $stmt->cond->name->parts === ['gettype']
|
2022-02-13 22:38:38 +01:00
|
|
|
|| $stmt->cond->name->parts === ['get_debug_type']
|
|
|
|
|| $stmt->cond->name->parts === ['count'])
|
2021-10-09 23:37:04 +02:00
|
|
|
&& $stmt->cond->getArgs()
|
2020-11-13 17:55:42 +01:00
|
|
|
) {
|
2021-10-09 23:37:04 +02:00
|
|
|
$first_arg = $stmt->cond->getArgs()[0];
|
2020-11-13 17:55:42 +01:00
|
|
|
|
|
|
|
if (!$first_arg->value instanceof PhpParser\Node\Expr\Variable) {
|
|
|
|
$switch_var_id = '$__tmp_switch__' . (int) $first_arg->value->getAttribute('startFilePos');
|
|
|
|
|
2021-10-13 18:35:16 +02:00
|
|
|
$condition_type = $statements_analyzer->node_data->getType($first_arg->value) ?? Type::getMixed();
|
2020-11-13 17:55:42 +01:00
|
|
|
|
|
|
|
$context->vars_in_scope[$switch_var_id] = $condition_type;
|
|
|
|
|
2021-02-15 22:18:41 +01:00
|
|
|
$match_condition = new VirtualFuncCall(
|
2020-11-13 17:55:42 +01:00
|
|
|
$stmt->cond->name,
|
|
|
|
[
|
2021-02-15 22:18:41 +01:00
|
|
|
new VirtualArg(
|
|
|
|
new VirtualVariable(
|
2020-11-13 17:55:42 +01:00
|
|
|
substr($switch_var_id, 1),
|
|
|
|
$first_arg->value->getAttributes()
|
|
|
|
),
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
$first_arg->getAttributes()
|
|
|
|
)
|
|
|
|
],
|
|
|
|
$stmt->cond->getAttributes()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} elseif ($stmt->cond instanceof PhpParser\Node\Expr\FuncCall
|
2020-08-30 22:08:22 +02:00
|
|
|
|| $stmt->cond instanceof PhpParser\Node\Expr\MethodCall
|
|
|
|
|| $stmt->cond instanceof PhpParser\Node\Expr\StaticCall
|
2020-11-13 17:55:42 +01:00
|
|
|
) {
|
|
|
|
$switch_var_id = '$__tmp_switch__' . (int) $stmt->cond->getAttribute('startFilePos');
|
2020-08-30 22:08:22 +02:00
|
|
|
|
2021-10-13 18:35:16 +02:00
|
|
|
$condition_type = $statements_analyzer->node_data->getType($stmt->cond) ?? Type::getMixed();
|
2020-08-30 22:08:22 +02:00
|
|
|
|
2020-11-13 17:55:42 +01:00
|
|
|
$context->vars_in_scope[$switch_var_id] = $condition_type;
|
2020-08-30 22:08:22 +02:00
|
|
|
|
2021-02-15 22:18:41 +01:00
|
|
|
$match_condition = new VirtualVariable(
|
2020-11-13 17:55:42 +01:00
|
|
|
substr($switch_var_id, 1),
|
|
|
|
$stmt->cond->getAttributes()
|
|
|
|
);
|
|
|
|
}
|
2020-08-30 22:08:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$arms = $stmt->arms;
|
|
|
|
|
|
|
|
foreach ($arms as $i => $arm) {
|
|
|
|
// move default to the end
|
|
|
|
if ($arm->conds === null) {
|
|
|
|
unset($arms[$i]);
|
|
|
|
$arms[] = $arm;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$arms = array_reverse($arms);
|
|
|
|
|
|
|
|
$last_arm = array_shift($arms);
|
|
|
|
|
2020-11-09 14:36:59 +01:00
|
|
|
if (!$last_arm) {
|
2021-12-03 20:11:20 +01:00
|
|
|
IssueBuffer::maybeAdd(
|
2020-11-09 14:36:59 +01:00
|
|
|
new UnhandledMatchCondition(
|
|
|
|
'This match expression does not match anything',
|
2021-12-03 20:11:20 +01:00
|
|
|
new CodeLocation($statements_analyzer->getSource(), $match_condition)
|
2020-11-09 14:36:59 +01:00
|
|
|
),
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
);
|
2020-11-09 14:36:59 +01:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-08-30 22:08:22 +02:00
|
|
|
$old_node_data = $statements_analyzer->node_data;
|
|
|
|
|
|
|
|
$statements_analyzer->node_data = clone $statements_analyzer->node_data;
|
|
|
|
|
2020-08-30 22:30:43 +02:00
|
|
|
if (!$last_arm->conds) {
|
2020-08-30 22:08:22 +02:00
|
|
|
$ternary = $last_arm->body;
|
|
|
|
} else {
|
2021-02-15 22:18:41 +01:00
|
|
|
$ternary = new VirtualTernary(
|
2020-08-30 22:08:22 +02:00
|
|
|
self::convertCondsToConditional($last_arm->conds, $match_condition, $last_arm->getAttributes()),
|
|
|
|
$last_arm->body,
|
2021-02-15 22:18:41 +01:00
|
|
|
new VirtualThrow(
|
|
|
|
new VirtualNew(
|
|
|
|
new VirtualFullyQualified(
|
2020-11-09 16:00:53 +01:00
|
|
|
'UnhandledMatchError',
|
|
|
|
$stmt->getAttributes()
|
|
|
|
),
|
|
|
|
[],
|
|
|
|
$stmt->getAttributes()
|
2020-08-30 22:08:22 +02:00
|
|
|
)
|
2020-11-09 16:00:53 +01:00
|
|
|
),
|
|
|
|
$stmt->getAttributes()
|
2020-08-30 22:08:22 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($arms as $arm) {
|
|
|
|
if (!$arm->conds) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-02-15 22:18:41 +01:00
|
|
|
$ternary = new VirtualTernary(
|
2020-08-30 22:08:22 +02:00
|
|
|
self::convertCondsToConditional($arm->conds, $match_condition, $arm->getAttributes()),
|
|
|
|
$arm->body,
|
|
|
|
$ternary,
|
|
|
|
$arm->getAttributes()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-08-30 22:23:53 +02:00
|
|
|
$suppressed_issues = $statements_analyzer->getSuppressedIssues();
|
|
|
|
|
|
|
|
if (!in_array('RedundantCondition', $suppressed_issues, true)) {
|
|
|
|
$statements_analyzer->addSuppressedIssues(['RedundantCondition']);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!in_array('RedundantConditionGivenDocblockType', $suppressed_issues, true)) {
|
|
|
|
$statements_analyzer->addSuppressedIssues(['RedundantConditionGivenDocblockType']);
|
|
|
|
}
|
|
|
|
|
2020-08-30 22:08:22 +02:00
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $ternary, $context) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-08-30 22:23:53 +02:00
|
|
|
if (!in_array('RedundantCondition', $suppressed_issues, true)) {
|
|
|
|
$statements_analyzer->removeSuppressedIssues(['RedundantCondition']);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!in_array('RedundantConditionGivenDocblockType', $suppressed_issues, true)) {
|
|
|
|
$statements_analyzer->removeSuppressedIssues(['RedundantConditionGivenDocblockType']);
|
|
|
|
}
|
|
|
|
|
2020-09-01 05:03:36 +02:00
|
|
|
if ($switch_var_id && $last_arm->conds) {
|
2020-09-01 04:59:47 +02:00
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
|
2020-09-01 05:03:36 +02:00
|
|
|
$all_conds = $last_arm->conds;
|
2020-09-01 04:59:47 +02:00
|
|
|
|
|
|
|
foreach ($arms as $arm) {
|
2020-09-01 05:03:36 +02:00
|
|
|
if (!$arm->conds) {
|
2021-12-03 21:40:18 +01:00
|
|
|
throw new UnexpectedValueException('bad');
|
2020-09-01 05:03:36 +02:00
|
|
|
}
|
|
|
|
|
2021-12-03 21:07:25 +01:00
|
|
|
$all_conds = array_merge($arm->conds, $all_conds);
|
2020-09-01 04:59:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$all_match_condition = self::convertCondsToConditional(
|
2021-11-26 13:48:38 +01:00
|
|
|
$all_conds,
|
2020-09-01 04:59:47 +02:00
|
|
|
$match_condition,
|
|
|
|
$match_condition->getAttributes()
|
|
|
|
);
|
|
|
|
|
2020-09-01 05:23:24 +02:00
|
|
|
ExpressionAnalyzer::analyze($statements_analyzer, $all_match_condition, $context);
|
|
|
|
|
2020-11-03 22:15:44 +01:00
|
|
|
$clauses = FormulaGenerator::getFormula(
|
2021-12-03 21:07:25 +01:00
|
|
|
spl_object_id($all_match_condition),
|
|
|
|
spl_object_id($all_match_condition),
|
2020-09-01 04:59:47 +02:00
|
|
|
$all_match_condition,
|
|
|
|
$context->self,
|
|
|
|
$statements_analyzer,
|
|
|
|
$codebase,
|
|
|
|
false,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
$reconcilable_types = Algebra::getTruthsFromFormula(
|
|
|
|
Algebra::negateFormula($clauses)
|
2020-09-01 04:59:47 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
// if the if has an || in the conditional, we cannot easily reason about it
|
|
|
|
if ($reconcilable_types) {
|
|
|
|
$changed_var_ids = [];
|
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
$vars_in_scope_reconciled = Reconciler::reconcileKeyedTypes(
|
2020-09-01 04:59:47 +02:00
|
|
|
$reconcilable_types,
|
|
|
|
[],
|
|
|
|
$context->vars_in_scope,
|
2022-01-11 00:45:29 +01:00
|
|
|
$context->references_in_scope,
|
2020-09-01 04:59:47 +02:00
|
|
|
$changed_var_ids,
|
|
|
|
[],
|
|
|
|
$statements_analyzer,
|
|
|
|
[],
|
|
|
|
$context->inside_loop,
|
|
|
|
null
|
|
|
|
);
|
|
|
|
|
|
|
|
if (isset($vars_in_scope_reconciled[$switch_var_id])) {
|
2021-12-03 21:07:25 +01:00
|
|
|
$array_literal_types = array_filter(
|
2021-05-03 23:54:09 +02:00
|
|
|
$vars_in_scope_reconciled[$switch_var_id]->getAtomicTypes(),
|
2022-01-05 23:45:11 +01:00
|
|
|
fn($type) => $type instanceof TLiteralInt
|
|
|
|
|| $type instanceof TLiteralString
|
|
|
|
|| $type instanceof TLiteralFloat
|
|
|
|
|| $type instanceof TEnumCase
|
2021-05-03 23:54:09 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
if ($array_literal_types) {
|
2021-12-03 20:11:20 +01:00
|
|
|
IssueBuffer::maybeAdd(
|
2020-09-01 04:59:47 +02:00
|
|
|
new UnhandledMatchCondition(
|
|
|
|
'This match expression is not exhaustive - consider values '
|
|
|
|
. $vars_in_scope_reconciled[$switch_var_id]->getId(),
|
2021-12-03 20:11:20 +01:00
|
|
|
new CodeLocation($statements_analyzer->getSource(), $match_condition)
|
2020-09-01 04:59:47 +02:00
|
|
|
),
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
);
|
2020-09-01 04:59:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-30 22:30:43 +02:00
|
|
|
$stmt_expr_type = $statements_analyzer->node_data->getType($ternary);
|
|
|
|
|
2021-10-13 18:35:16 +02:00
|
|
|
$old_node_data->setType($stmt, $stmt_expr_type ?? Type::getMixed());
|
2020-08-30 22:08:22 +02:00
|
|
|
|
|
|
|
$statements_analyzer->node_data = $old_node_data;
|
|
|
|
|
|
|
|
$context->inside_call = $was_inside_call;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param non-empty-list<PhpParser\Node\Expr> $conds
|
|
|
|
*/
|
|
|
|
private static function convertCondsToConditional(
|
|
|
|
array $conds,
|
|
|
|
PhpParser\Node\Expr $match_condition,
|
|
|
|
array $attributes
|
2021-12-05 18:51:26 +01:00
|
|
|
): PhpParser\Node\Expr {
|
2020-08-30 22:08:22 +02:00
|
|
|
if (count($conds) === 1) {
|
2021-02-15 22:18:41 +01:00
|
|
|
return new VirtualIdentical(
|
2020-08-30 22:08:22 +02:00
|
|
|
$match_condition,
|
|
|
|
$conds[0],
|
|
|
|
$attributes
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$array_items = array_map(
|
2022-01-05 23:45:11 +01:00
|
|
|
fn($cond): PhpParser\Node\Expr\ArrayItem =>
|
|
|
|
new VirtualArrayItem($cond, null, false, $cond->getAttributes()),
|
2020-08-30 22:08:22 +02:00
|
|
|
$conds
|
|
|
|
);
|
|
|
|
|
2021-02-15 22:18:41 +01:00
|
|
|
return new VirtualFuncCall(
|
|
|
|
new VirtualFullyQualified(['in_array']),
|
2020-08-30 22:08:22 +02:00
|
|
|
[
|
2021-02-15 22:18:41 +01:00
|
|
|
new VirtualArg(
|
2020-11-09 16:00:53 +01:00
|
|
|
$match_condition,
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
$attributes
|
2020-08-30 22:08:22 +02:00
|
|
|
),
|
2021-02-15 22:18:41 +01:00
|
|
|
new VirtualArg(
|
|
|
|
new VirtualArray(
|
2020-11-09 16:00:53 +01:00
|
|
|
$array_items,
|
|
|
|
$attributes
|
|
|
|
),
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
$attributes
|
2020-08-30 22:08:22 +02:00
|
|
|
),
|
2021-02-15 22:18:41 +01:00
|
|
|
new VirtualArg(
|
|
|
|
new VirtualConstFetch(
|
|
|
|
new VirtualFullyQualified(['true']),
|
2020-11-09 16:00:53 +01:00
|
|
|
$attributes
|
|
|
|
),
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
$attributes
|
2020-08-30 22:08:22 +02:00
|
|
|
),
|
|
|
|
],
|
|
|
|
$attributes
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|