1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-22 05:41:20 +01:00

136 lines
4.7 KiB
PHP
Raw Normal View History

2016-10-22 17:35:59 -04:00
<?php
namespace Psalm\Checker\Statements\Block;
use PhpParser;
use Psalm\Checker\ClassLikeChecker;
use Psalm\Checker\ScopeChecker;
use Psalm\Checker\StatementsChecker;
use Psalm\CodeLocation;
2016-11-02 02:29:00 -04:00
use Psalm\Context;
2017-12-02 18:28:18 -05:00
use Psalm\Scope\LoopScope;
2016-10-22 17:35:59 -04:00
use Psalm\Type;
use Psalm\Type\Atomic\TNamedObject;
2016-10-22 17:35:59 -04:00
class TryChecker
{
/**
2016-11-02 02:29:00 -04:00
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Stmt\TryCatch $stmt
* @param Context $context
2017-05-26 20:16:18 -04:00
*
2016-11-02 02:29:00 -04:00
* @return false|null
2016-10-22 17:35:59 -04:00
*/
public static function analyze(
2016-11-02 02:29:00 -04:00
StatementsChecker $statements_checker,
PhpParser\Node\Stmt\TryCatch $stmt,
Context $context,
2017-12-02 18:28:18 -05:00
LoopScope $loop_scope = null
2016-11-02 02:29:00 -04:00
) {
2017-12-02 18:28:18 -05:00
$statements_checker->analyze($stmt->stmts, $context, $loop_scope);
$try_leaves_loop = $loop_scope
&& $loop_scope->final_actions
&& !in_array(ScopeChecker::ACTION_NONE, $loop_scope->final_actions, true);
2016-10-22 17:35:59 -04:00
// clone context for catches after running the try block, as
// we optimistically assume it only failed at the very end
$original_context = clone $context;
foreach ($stmt->catches as $catch) {
$catch_context = clone $original_context;
$fq_catch_classes = [];
2016-10-22 17:35:59 -04:00
foreach ($catch->types as $catch_type) {
2016-12-08 17:15:51 -05:00
$fq_catch_class = ClassLikeChecker::getFQCLNFromNameObject(
$catch_type,
Refactor scanning and analysis, introducing multithreading (#191) * Add failing test * Add visitor to soup up classlike references * Move a whole bunch of code into the visitor * Move some methods back, move onto analysis stage * Use the getAliases method everywhere * Fix refs * Fix more refs * Fix some tests * Fix more tests * Fix include tests * Shift config class finding to project checker and fix bugs * Fix a few more tests * transition test to new syntax * Remove var_dump * Delete a bunch of code and fix mutation test * Remove unnecessary visitation * Transition to better mocked out file provider, breaking some cached statement loading * Use different scheme for naming anonymous classes * Fix anonymous class issues * Refactor file/statement loading * Add specific property types * Fix mapped property assignment * Improve how we deal with traits * Fix trait checking * Pass Psalm checks * Add multi-process support * Delay console output until the end * Remove PHP 7 syntax * Update file storage with classes * Fix scanning individual files and add reflection return types * Always turn XDebug off * Add quicker method of getting method mutations * Queue return types for crawling * Interpret all strings as possible classes once we see a `get_class` call * Check invalid return types again * Fix template namespacing issues * Default to class-insensitive file names for includes * Don’t overwrite existing issues data * Add var docblocks for scanning * Add null check * Fix loading of external classes in templates * Only try to populate class when we haven’t yet seen it’s not a class * Fix trait property accessibility * Only ever improve docblock param type * Make param replacement more robust * Fix static const missing inferred type * Fix a few more tests * Register constant definitions * Fix trait aliasing * Skip constant type tests for now * Fix linting issues * Make sure caching is off for tests * Remove unnecessary return * Use emulative parser if on PHP 5.6 * Cache parser for faster first-time parse * Fix constant resolution when scanning classes * Remove test that’s beyond a practical scope * Add back --diff support * Add --help for --threads * Remove unused vars
2017-07-25 16:11:02 -04:00
$statements_checker->getAliases()
);
2016-10-22 17:35:59 -04:00
2016-12-08 17:15:51 -05:00
if ($context->check_classes) {
if (ClassLikeChecker::checkFullyQualifiedClassLikeName(
$statements_checker->getFileChecker()->project_checker,
$fq_catch_class,
new CodeLocation($statements_checker->getSource(), $catch_type, $context->include_location),
$statements_checker->getSuppressedIssues()
) === false) {
return false;
}
2016-10-22 17:35:59 -04:00
}
2016-12-08 17:15:51 -05:00
$fq_catch_classes[] = $fq_catch_class;
2016-10-22 17:35:59 -04:00
}
$catch_var_id = '$' . $catch->var;
$catch_context->vars_in_scope[$catch_var_id] = new Type\Union(
array_map(
2016-12-06 19:41:52 -05:00
/**
* @param string $fq_catch_class
2017-05-26 20:16:18 -04:00
*
2016-12-07 14:13:39 -05:00
* @return Type\Atomic
2016-12-06 19:41:52 -05:00
*/
function ($fq_catch_class) {
return new TNamedObject($fq_catch_class);
},
$fq_catch_classes
)
);
2016-10-22 17:35:59 -04: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;
if (!$statements_checker->hasVariable($catch_var_id)) {
$statements_checker->registerVariable(
$catch_var_id,
new CodeLocation($statements_checker, $catch, $context->include_location, true)
);
}
2016-10-22 17:35:59 -04:00
// this registers the variable to avoid unfair deadcode issues
$catch_context->hasVariable($catch_var_id);
2016-10-22 17:35:59 -04:00
2017-12-02 18:28:18 -05:00
$statements_checker->analyze($catch->stmts, $catch_context, $loop_scope);
2016-10-22 17:35:59 -04:00
$context->referenced_var_ids = array_merge(
$catch_context->referenced_var_ids,
$context->referenced_var_ids
);
2017-02-02 00:45:23 -05:00
if (ScopeChecker::getFinalControlActions($catch->stmts) !== [ScopeChecker::ACTION_END]) {
2016-10-22 17:35:59 -04:00
foreach ($catch_context->vars_in_scope as $catch_var => $type) {
2016-11-02 02:29:00 -04:00
if ($catch->var !== $catch_var &&
$context->hasVariable($catch_var) &&
$context->vars_in_scope[$catch_var]->getId() !== $type->getId()
2016-11-02 02:29:00 -04:00
) {
$context->vars_in_scope[$catch_var] = Type::combineUnionTypes(
$context->vars_in_scope[$catch_var],
$type
);
2016-10-22 17:35:59 -04:00
}
}
2016-11-02 02:29:00 -04:00
$context->vars_possibly_in_scope = array_merge(
$catch_context->vars_possibly_in_scope,
$context->vars_possibly_in_scope
);
2016-10-22 17:35:59 -04:00
}
}
if ($stmt->finally) {
2017-12-02 18:28:18 -05:00
$statements_checker->analyze($stmt->finally->stmts, $context, $loop_scope);
}
if ($loop_scope
&& !$try_leaves_loop
&& !in_array(ScopeChecker::ACTION_NONE, $loop_scope->final_actions, true)
) {
$loop_scope->final_actions[] = ScopeChecker::ACTION_NONE;
2016-10-22 17:35:59 -04:00
}
2016-11-02 02:29:00 -04:00
return null;
2016-10-22 17:35:59 -04:00
}
}