mirror of
https://github.com/danog/psalm.git
synced 2025-01-05 12:38:35 +01:00
117 lines
4.6 KiB
PHP
117 lines
4.6 KiB
PHP
|
<?php
|
||
|
namespace Psalm\Internal\Analyzer\Statements;
|
||
|
|
||
|
use PhpParser;
|
||
|
use Psalm\Internal\Analyzer\ScopeAnalyzer;
|
||
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
||
|
use Psalm\Context;
|
||
|
use Psalm\Type;
|
||
|
|
||
|
class BreakAnalyzer
|
||
|
{
|
||
|
public static function analyze(
|
||
|
StatementsAnalyzer $statements_analyzer,
|
||
|
PhpParser\Node\Stmt\Break_ $stmt,
|
||
|
Context $context
|
||
|
) : void {
|
||
|
$loop_scope = $context->loop_scope;
|
||
|
|
||
|
$leaving_switch = true;
|
||
|
|
||
|
$codebase = $statements_analyzer->getCodebase();
|
||
|
|
||
|
if ($loop_scope) {
|
||
|
if ($context->break_types
|
||
|
&& \end($context->break_types) === 'switch'
|
||
|
&& (!$stmt->num
|
||
|
|| !$stmt->num instanceof PhpParser\Node\Scalar\LNumber
|
||
|
|| $stmt->num->value < 2
|
||
|
)
|
||
|
) {
|
||
|
$loop_scope->final_actions[] = ScopeAnalyzer::ACTION_LEAVE_SWITCH;
|
||
|
} else {
|
||
|
$leaving_switch = false;
|
||
|
|
||
|
$loop_scope->final_actions[] = ScopeAnalyzer::ACTION_BREAK;
|
||
|
}
|
||
|
|
||
|
$redefined_vars = $context->getRedefinedVars($loop_scope->loop_parent_context->vars_in_scope);
|
||
|
|
||
|
if ($loop_scope->possibly_redefined_loop_parent_vars === null) {
|
||
|
$loop_scope->possibly_redefined_loop_parent_vars = $redefined_vars;
|
||
|
} else {
|
||
|
foreach ($redefined_vars as $var => $type) {
|
||
|
if ($type->hasMixed()) {
|
||
|
$loop_scope->possibly_redefined_loop_parent_vars[$var] = $type;
|
||
|
} elseif (isset($loop_scope->possibly_redefined_loop_parent_vars[$var])) {
|
||
|
$loop_scope->possibly_redefined_loop_parent_vars[$var] = Type::combineUnionTypes(
|
||
|
$type,
|
||
|
$loop_scope->possibly_redefined_loop_parent_vars[$var]
|
||
|
);
|
||
|
} else {
|
||
|
$loop_scope->possibly_redefined_loop_parent_vars[$var] = $type;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($loop_scope->iteration_count === 0) {
|
||
|
foreach ($context->vars_in_scope as $var_id => $type) {
|
||
|
if (!isset($loop_scope->loop_parent_context->vars_in_scope[$var_id])) {
|
||
|
if (isset($loop_scope->possibly_defined_loop_parent_vars[$var_id])) {
|
||
|
$loop_scope->possibly_defined_loop_parent_vars[$var_id] = Type::combineUnionTypes(
|
||
|
$type,
|
||
|
$loop_scope->possibly_defined_loop_parent_vars[$var_id]
|
||
|
);
|
||
|
} else {
|
||
|
$loop_scope->possibly_defined_loop_parent_vars[$var_id] = $type;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($codebase->find_unused_variables && !$leaving_switch) {
|
||
|
foreach ($context->unreferenced_vars as $var_id => $locations) {
|
||
|
if (isset($loop_scope->unreferenced_vars[$var_id])) {
|
||
|
$loop_scope->unreferenced_vars[$var_id] += $locations;
|
||
|
} else {
|
||
|
$loop_scope->unreferenced_vars[$var_id] = $locations;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$loop_scope->referenced_var_ids += $context->referenced_var_ids;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$case_scope = $context->case_scope;
|
||
|
if ($case_scope && $leaving_switch) {
|
||
|
foreach ($context->vars_in_scope as $var_id => $type) {
|
||
|
if ($case_scope->parent_context !== $context) {
|
||
|
if ($case_scope->break_vars === null) {
|
||
|
$case_scope->break_vars = [];
|
||
|
}
|
||
|
|
||
|
if (isset($case_scope->break_vars[$var_id])) {
|
||
|
$case_scope->break_vars[$var_id] = Type::combineUnionTypes(
|
||
|
$type,
|
||
|
$case_scope->break_vars[$var_id]
|
||
|
);
|
||
|
} else {
|
||
|
$case_scope->break_vars[$var_id] = $type;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($codebase->find_unused_variables) {
|
||
|
foreach ($context->unreferenced_vars as $var_id => $locations) {
|
||
|
if (isset($case_scope->unreferenced_vars[$var_id])) {
|
||
|
$case_scope->unreferenced_vars[$var_id] += $locations;
|
||
|
} else {
|
||
|
$case_scope->unreferenced_vars[$var_id] = $locations;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$context->has_returned = true;
|
||
|
}
|
||
|
}
|