1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-16 03:17:02 +01:00
psalm/src/Psalm/Internal/Analyzer/Statements/Block/TryAnalyzer.php

540 lines
21 KiB
PHP
Raw Normal View History

2016-10-22 23:35:59 +02:00
<?php
2018-11-06 03:57:36 +01:00
namespace Psalm\Internal\Analyzer\Statements\Block;
2016-10-22 23:35:59 +02:00
use PhpParser;
2018-11-06 03:57:36 +01:00
use Psalm\Internal\Analyzer\ClassLikeAnalyzer;
use Psalm\Internal\Analyzer\ScopeAnalyzer;
use Psalm\Internal\Analyzer\StatementsAnalyzer;
use Psalm\CodeLocation;
2016-11-02 07:29:00 +01:00
use Psalm\Context;
use Psalm\Issue\InvalidCatch;
use Psalm\IssueBuffer;
2016-10-22 23:35:59 +02:00
use Psalm\Type;
use Psalm\Type\Atomic\TNamedObject;
use Psalm\Type\Union;
use function in_array;
use function array_merge;
use function array_diff;
use function array_intersect_key;
use function array_diff_key;
use function is_string;
use function strtolower;
use function array_map;
use function version_compare;
use const PHP_VERSION;
2016-10-22 23:35:59 +02:00
/**
* @internal
*/
2018-11-06 03:57:36 +01:00
class TryAnalyzer
2016-10-22 23:35:59 +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\Stmt\TryCatch $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 23:35:59 +02:00
*/
public static function analyze(
2018-11-11 18:01:14 +01:00
StatementsAnalyzer $statements_analyzer,
2016-11-02 07:29:00 +01:00
PhpParser\Node\Stmt\TryCatch $stmt,
Context $context
2016-11-02 07:29:00 +01:00
) {
$catch_actions = [];
$all_catches_leave = true;
2018-11-11 18:01:14 +01:00
$codebase = $statements_analyzer->getCodebase();
/** @var int $i */
foreach ($stmt->catches as $i => $catch) {
2018-11-06 03:57:36 +01:00
$catch_actions[$i] = ScopeAnalyzer::getFinalControlActions(
$catch->stmts,
$statements_analyzer->node_data,
2018-11-06 03:57:36 +01:00
$codebase->config->exit_functions
);
$all_catches_leave = $all_catches_leave && !in_array(ScopeAnalyzer::ACTION_NONE, $catch_actions[$i], true);
}
$existing_thrown_exceptions = $context->possibly_thrown_exceptions;
/**
2019-04-03 01:42:23 +02:00
* @var array<string, array<array-key, CodeLocation>>
*/
$context->possibly_thrown_exceptions = [];
$old_context = clone $context;
if ($all_catches_leave && !$stmt->finally) {
$try_context = $context;
} else {
$try_context = clone $context;
2018-11-06 03:57:36 +01:00
if ($codebase->alter_code) {
$try_context->branch_point = $try_context->branch_point ?: (int) $stmt->getAttribute('startFilePos');
}
}
$assigned_var_ids = $try_context->assigned_var_ids;
$context->assigned_var_ids = [];
$old_referenced_var_ids = $try_context->referenced_var_ids;
$old_unreferenced_vars = $try_context->unreferenced_vars;
2019-11-11 22:52:55 +01:00
$newly_unreferenced_vars = [];
2018-11-11 18:01:14 +01:00
if ($statements_analyzer->analyze($stmt->stmts, $context) === false) {
return false;
}
$stmt_control_actions = ScopeAnalyzer::getFinalControlActions(
$stmt->stmts,
$statements_analyzer->node_data,
$codebase->config->exit_functions,
$context->break_types
);
/** @var array<string, bool> */
$newly_assigned_var_ids = $context->assigned_var_ids;
$context->assigned_var_ids = array_merge(
$assigned_var_ids,
$newly_assigned_var_ids
);
$possibly_referenced_var_ids = array_merge(
$context->referenced_var_ids,
$old_referenced_var_ids
);
if ($try_context !== $context) {
2017-12-11 00:09:38 +01:00
foreach ($context->vars_in_scope as $var_id => $type) {
if (!isset($try_context->vars_in_scope[$var_id])) {
$try_context->vars_in_scope[$var_id] = clone $type;
2017-12-11 00:09:38 +01:00
$try_context->vars_in_scope[$var_id]->from_docblock = true;
} else {
$try_context->vars_in_scope[$var_id] = Type::combineUnionTypes(
2017-12-11 00:09:38 +01:00
$try_context->vars_in_scope[$var_id],
$type
);
}
}
2017-12-11 00:34:22 +01:00
$try_context->vars_possibly_in_scope = $context->vars_possibly_in_scope;
$context->referenced_var_ids = array_intersect_key(
$try_context->referenced_var_ids,
$context->referenced_var_ids
);
if ($context->collect_references) {
$newly_unreferenced_vars = array_merge(
$newly_unreferenced_vars,
array_diff_key(
$context->unreferenced_vars,
$old_unreferenced_vars
)
);
}
}
$try_leaves_loop = $context->loop_scope
&& $context->loop_scope->final_actions
2018-11-06 03:57:36 +01:00
&& !in_array(ScopeAnalyzer::ACTION_NONE, $context->loop_scope->final_actions, true);
2016-10-22 23:35:59 +02:00
if (!$all_catches_leave) {
foreach ($newly_assigned_var_ids as $assigned_var_id => $_) {
$context->removeVarFromConflictingClauses($assigned_var_id);
}
} else {
foreach ($newly_assigned_var_ids as $assigned_var_id => $_) {
$try_context->removeVarFromConflictingClauses($assigned_var_id);
}
}
// at this point we have two contexts $context, in which it is assumed that everything was fine,
// and $try_context - which allows all variables to have the union of the values before and after
// the try was applied
$original_context = clone $try_context;
2016-10-22 23:35:59 +02:00
$issues_to_suppress = [
'RedundantCondition',
'RedundantConditionGivenDocblockType',
'TypeDoesNotContainNull',
'TypeDoesNotContainType',
];
$definitely_newly_assigned_var_ids = $newly_assigned_var_ids;
/** @var int $i */
foreach ($stmt->catches as $i => $catch) {
2016-10-22 23:35:59 +02:00
$catch_context = clone $original_context;
foreach ($catch_context->vars_in_scope as $var_id => $type) {
if (!isset($old_context->vars_in_scope[$var_id])) {
$type = clone $type;
$type->possibly_undefined_from_try = true;
$catch_context->vars_in_scope[$var_id] = $type;
} else {
$catch_context->vars_in_scope[$var_id] = Type::combineUnionTypes(
$type,
$old_context->vars_in_scope[$var_id]
);
}
2019-11-11 22:52:55 +01:00
if (isset($old_context->unreferenced_vars[$var_id])) {
if (!isset($catch_context->unreferenced_vars[$var_id])) {
$catch_context->unreferenced_vars[$var_id] = $old_context->unreferenced_vars[$var_id];
} else {
$catch_context->unreferenced_vars[$var_id] += $old_context->unreferenced_vars[$var_id];
}
}
}
$fq_catch_classes = [];
2016-10-22 23:35:59 +02:00
$catch_var_name = $catch->var->name;
if (!is_string($catch_var_name)) {
throw new \UnexpectedValueException('Catch var name must be a string');
}
if (!$catch->types) {
throw new \UnexpectedValueException('Very bad');
}
foreach ($catch->types as $catch_type) {
2018-11-06 03:57:36 +01:00
$fq_catch_class = ClassLikeAnalyzer::getFQCLNFromNameObject(
$catch_type,
2018-11-11 18:01:14 +01:00
$statements_analyzer->getAliases()
);
2016-10-22 23:35:59 +02:00
2019-06-05 17:33:04 +02:00
if ($codebase->alter_code && $fq_catch_class) {
$codebase->classlikes->handleClassLikeReferenceInMigration(
$codebase,
$statements_analyzer,
$catch_type,
$fq_catch_class,
$context->calling_function_id
2019-06-05 17:33:04 +02:00
);
}
if ($original_context->check_classes) {
2018-11-06 03:57:36 +01:00
if (ClassLikeAnalyzer::checkFullyQualifiedClassLikeName(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
$fq_catch_class,
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $catch_type, $context->include_location),
$statements_analyzer->getSuppressedIssues(),
false
) === false) {
return false;
}
2016-10-22 23:35:59 +02:00
}
2016-12-08 23:15:51 +01:00
2018-02-01 06:50:01 +01:00
if (($codebase->classExists($fq_catch_class)
&& strtolower($fq_catch_class) !== 'exception'
2018-02-01 06:50:01 +01:00
&& !($codebase->classExtends($fq_catch_class, 'Exception')
|| $codebase->classImplements($fq_catch_class, 'Throwable')))
|| ($codebase->interfaceExists($fq_catch_class)
&& strtolower($fq_catch_class) !== 'throwable'
2018-02-01 06:50:01 +01:00
&& !$codebase->interfaceExtends($fq_catch_class, 'Throwable'))
) {
if (IssueBuffer::accepts(
new InvalidCatch(
'Class/interface ' . $fq_catch_class . ' cannot be caught',
2018-11-11 18:01:14 +01:00
new CodeLocation($statements_analyzer->getSource(), $stmt),
$fq_catch_class
),
2018-11-11 18:01:14 +01:00
$statements_analyzer->getSuppressedIssues()
)) {
return false;
}
}
2016-12-08 23:15:51 +01:00
$fq_catch_classes[] = $fq_catch_class;
2016-10-22 23:35:59 +02:00
}
if ($catch_context->collect_exceptions) {
foreach ($fq_catch_classes as $fq_catch_class) {
$fq_catch_class_lower = strtolower($fq_catch_class);
foreach ($context->possibly_thrown_exceptions as $exception_fqcln => $_) {
$exception_fqcln_lower = strtolower($exception_fqcln);
if ($exception_fqcln_lower === $fq_catch_class_lower) {
unset($context->possibly_thrown_exceptions[$exception_fqcln]);
unset($catch_context->possibly_thrown_exceptions[$exception_fqcln]);
continue;
}
if ($codebase->classExists($exception_fqcln)
&& $codebase->classExtendsOrImplements(
$exception_fqcln,
$fq_catch_class
)
) {
unset($context->possibly_thrown_exceptions[$exception_fqcln]);
unset($catch_context->possibly_thrown_exceptions[$exception_fqcln]);
continue;
}
if ($codebase->interfaceExists($exception_fqcln)
&& $codebase->interfaceExtends(
$exception_fqcln,
$fq_catch_class
)
) {
unset($context->possibly_thrown_exceptions[$exception_fqcln]);
unset($catch_context->possibly_thrown_exceptions[$exception_fqcln]);
continue;
}
}
}
$context->mergeExceptions($catch_context);
}
$catch_var_id = '$' . $catch_var_name;
$catch_context->vars_in_scope[$catch_var_id] = new Union(
array_map(
2016-12-07 01:41:52 +01:00
/**
* @param string $fq_catch_class
2017-05-27 02:16:18 +02:00
*
2016-12-07 20:13:39 +01:00
* @return Type\Atomic
2016-12-07 01:41:52 +01:00
*/
2018-02-01 06:50:01 +01:00
function ($fq_catch_class) use ($codebase) {
$catch_class_type = new TNamedObject($fq_catch_class);
if (version_compare(PHP_VERSION, '7.0.0dev', '>=')
2018-02-01 06:50:01 +01:00
&& $codebase->interfaceExists($fq_catch_class)
&& !$codebase->interfaceExtends($fq_catch_class, 'Throwable')
) {
$catch_class_type->addIntersectionType(new TNamedObject('Throwable'));
}
return $catch_class_type;
},
$fq_catch_classes
)
);
2016-10-22 23:35:59 +02:00
// discard all clauses because crazy stuff may have happened in try block
$catch_context->clauses = [];
$catch_context->vars_possibly_in_scope[$catch_var_id] = true;
2018-11-11 18:01:14 +01:00
if (!$statements_analyzer->hasVariable($catch_var_id)) {
$location = new CodeLocation(
2018-11-11 18:01:14 +01:00
$statements_analyzer,
$catch->var,
$context->include_location
);
2018-11-11 18:01:14 +01:00
$statements_analyzer->registerVariable(
$catch_var_id,
$location,
$try_context->branch_point
);
2018-06-17 02:01:33 +02:00
$catch_context->unreferenced_vars[$catch_var_id] = [$location->getHash() => $location];
}
2016-10-22 23:35:59 +02:00
// this registers the variable to avoid unfair deadcode issues
2018-11-11 18:01:14 +01:00
$catch_context->hasVariable($catch_var_id, $statements_analyzer);
2016-10-22 23:35:59 +02:00
2018-11-11 18:01:14 +01:00
$suppressed_issues = $statements_analyzer->getSuppressedIssues();
foreach ($issues_to_suppress as $issue_to_suppress) {
if (!in_array($issue_to_suppress, $suppressed_issues, true)) {
$statements_analyzer->addSuppressedIssues([$issue_to_suppress]);
}
}
$old_catch_assigned_var_ids = $catch_context->referenced_var_ids;
$catch_context->assigned_var_ids = [];
2018-11-11 18:01:14 +01:00
$statements_analyzer->analyze($catch->stmts, $catch_context);
2016-10-22 23:35:59 +02:00
// recalculate in case there's a no-return clause
$catch_actions[$i] = ScopeAnalyzer::getFinalControlActions(
$catch->stmts,
$statements_analyzer->node_data,
$codebase->config->exit_functions,
$context->break_types
);
foreach ($issues_to_suppress as $issue_to_suppress) {
if (!in_array($issue_to_suppress, $suppressed_issues, true)) {
$statements_analyzer->removeSuppressedIssues([$issue_to_suppress]);
}
}
/** @var array<string, bool> */
$new_catch_assigned_var_ids = $catch_context->assigned_var_ids;
$catch_context->assigned_var_ids += $old_catch_assigned_var_ids;
$context->referenced_var_ids = array_intersect_key(
$catch_context->referenced_var_ids,
$context->referenced_var_ids
);
2017-02-02 06:45:23 +01:00
$possibly_referenced_var_ids = array_merge(
$catch_context->referenced_var_ids,
$possibly_referenced_var_ids
);
2018-11-06 03:57:36 +01:00
if ($context->collect_references && $catch_actions[$i] !== [ScopeAnalyzer::ACTION_END]) {
$newly_unreferenced_vars = array_merge(
$newly_unreferenced_vars,
array_diff_key(
$catch_context->unreferenced_vars,
$old_unreferenced_vars
)
);
2018-06-17 02:01:33 +02:00
foreach ($catch_context->unreferenced_vars as $var_id => $locations) {
if (!isset($old_unreferenced_vars[$var_id])
&& (isset($context->unreferenced_vars[$var_id])
|| isset($newly_assigned_var_ids[$var_id]))
) {
2018-11-11 18:01:14 +01:00
$statements_analyzer->registerVariableUses($locations);
} elseif (isset($old_unreferenced_vars[$var_id])
2018-06-17 02:01:33 +02:00
&& $old_unreferenced_vars[$var_id] !== $locations
) {
2018-11-11 18:01:14 +01:00
$statements_analyzer->registerVariableUses($locations);
} elseif (isset($newly_unreferenced_vars[$var_id])) {
$context->unreferenced_vars[$var_id] = $newly_unreferenced_vars[$var_id];
}
}
}
if ($catch_context->collect_exceptions) {
$context->mergeExceptions($catch_context);
}
if ($catch_actions[$i] !== [ScopeAnalyzer::ACTION_END]
&& $catch_actions[$i] !== [ScopeAnalyzer::ACTION_CONTINUE]
&& $catch_actions[$i] !== [ScopeAnalyzer::ACTION_BREAK]
) {
$definitely_newly_assigned_var_ids = array_intersect_key(
$new_catch_assigned_var_ids,
$definitely_newly_assigned_var_ids
);
2017-12-11 00:09:38 +01:00
foreach ($catch_context->vars_in_scope as $var_id => $type) {
2019-07-07 20:00:49 +02:00
if ($stmt_control_actions === [ScopeAnalyzer::ACTION_END]) {
$context->vars_in_scope[$var_id] = $type;
} elseif (isset($context->vars_in_scope[$var_id])
&& !$context->vars_in_scope[$var_id]->equals($type)
2016-11-02 07:29:00 +01:00
) {
2017-12-11 00:09:38 +01:00
$context->vars_in_scope[$var_id] = Type::combineUnionTypes(
$context->vars_in_scope[$var_id],
2016-11-02 07:29:00 +01:00
$type
);
2016-10-22 23:35:59 +02:00
}
}
$context->vars_possibly_in_scope = array_merge(
$catch_context->vars_possibly_in_scope,
$context->vars_possibly_in_scope
);
} else {
if ($stmt->finally) {
$context->vars_possibly_in_scope = array_merge(
$catch_context->vars_possibly_in_scope,
$context->vars_possibly_in_scope
);
}
}
if ($stmt->finally) {
$suppressed_issues = $statements_analyzer->getSuppressedIssues();
foreach ($issues_to_suppress as $issue_to_suppress) {
if (!in_array($issue_to_suppress, $suppressed_issues, true)) {
$statements_analyzer->addSuppressedIssues([$issue_to_suppress]);
}
}
$statements_analyzer->analyze($stmt->finally->stmts, $catch_context);
foreach ($issues_to_suppress as $issue_to_suppress) {
if (!in_array($issue_to_suppress, $suppressed_issues, true)) {
$statements_analyzer->removeSuppressedIssues([$issue_to_suppress]);
}
}
}
}
foreach ($definitely_newly_assigned_var_ids as $var_id => $_) {
if (isset($context->vars_in_scope[$var_id])) {
$new_type = clone $context->vars_in_scope[$var_id];
$new_type->possibly_undefined_from_try = false;
$context->vars_in_scope[$var_id] = $new_type;
2016-10-22 23:35:59 +02:00
}
}
if ($context->loop_scope
2017-12-03 00:28:18 +01:00
&& !$try_leaves_loop
2018-11-06 03:57:36 +01:00
&& !in_array(ScopeAnalyzer::ACTION_NONE, $context->loop_scope->final_actions, true)
2017-12-03 00:28:18 +01:00
) {
2018-11-06 03:57:36 +01:00
$context->loop_scope->final_actions[] = ScopeAnalyzer::ACTION_NONE;
2016-10-22 23:35:59 +02:00
}
2016-11-02 07:29:00 +01:00
$newly_referenced_var_ids = array_diff_key(
$context->referenced_var_ids,
$old_referenced_var_ids
);
if ($context->collect_references) {
foreach ($old_unreferenced_vars as $var_id => $locations) {
if ((isset($context->unreferenced_vars[$var_id]) && $context->unreferenced_vars[$var_id] !== $locations)
|| (!isset($newly_referenced_var_ids[$var_id]) && isset($possibly_referenced_var_ids[$var_id]))
) {
$statements_analyzer->registerVariableUses($locations);
}
}
$newly_unreferenced_vars = array_merge(
$newly_unreferenced_vars,
array_diff_key(
$try_context->unreferenced_vars,
$old_unreferenced_vars
)
);
foreach ($newly_unreferenced_vars as $var_id => $locations) {
if (!isset($context->unreferenced_vars[$var_id])) {
$context->unreferenced_vars[$var_id] = $locations;
}
}
}
if ($stmt->finally) {
$suppressed_issues = $statements_analyzer->getSuppressedIssues();
foreach ($issues_to_suppress as $issue_to_suppress) {
if (!in_array($issue_to_suppress, $suppressed_issues, true)) {
$statements_analyzer->addSuppressedIssues([$issue_to_suppress]);
}
}
2018-11-11 18:01:14 +01:00
$statements_analyzer->analyze($stmt->finally->stmts, $context);
foreach ($issues_to_suppress as $issue_to_suppress) {
if (!in_array($issue_to_suppress, $suppressed_issues, true)) {
$statements_analyzer->removeSuppressedIssues([$issue_to_suppress]);
}
}
}
2019-04-03 01:42:23 +02:00
foreach ($existing_thrown_exceptions as $possibly_thrown_exception => $codelocations) {
foreach ($codelocations as $hash => $codelocation) {
$context->possibly_thrown_exceptions[$possibly_thrown_exception][$hash] = $codelocation;
}
}
2016-11-02 07:29:00 +01:00
return null;
2016-10-22 23:35:59 +02:00
}
}