Skip to content

Commit

Permalink
Fix false positive for unused variable in try (fixes #7613).
Browse files Browse the repository at this point in the history
  • Loading branch information
AndrolGenhald committed Feb 13, 2022
1 parent 7f304be commit 7b1599d
Show file tree
Hide file tree
Showing 5 changed files with 58 additions and 0 deletions.
7 changes: 7 additions & 0 deletions src/Psalm/Context.php
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,13 @@ class Context
*/
public $inside_assignment = false;

/**
* Whether or not we're inside a try block.
*
* @var bool
*/
public $inside_try = false;

/**
* @var null|CodeLocation
*/
Expand Down
2 changes: 2 additions & 0 deletions src/Psalm/Internal/Analyzer/Statements/Block/TryAnalyzer.php
Original file line number Diff line number Diff line change
Expand Up @@ -86,9 +86,11 @@ public static function analyze(

$old_referenced_var_ids = $try_context->referenced_var_ids;

$context->inside_try = true;
if ($statements_analyzer->analyze($stmt->stmts, $context) === false) {
return false;
}
$context->inside_try = false;

if ($try_context->finally_scope) {
foreach ($context->vars_in_scope as $var_id => $type) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -315,6 +315,14 @@ public static function analyze(
$assign_value_type->parent_nodes = [
$assignment_node->id => $assignment_node
];

if ($context->inside_try) {
// Copy previous assignment's parent nodes inside a try. Since an exception could be thrown at any
// point this is a workaround to ensure that use of a variable also uses all previous assignments.
if (isset($context->vars_in_scope[$array_var_id])) {
$assign_value_type->parent_nodes += $context->vars_in_scope[$array_var_id]->parent_nodes;
}
}
}

if ($array_var_id && isset($context->vars_in_scope[$array_var_id])) {
Expand Down
19 changes: 19 additions & 0 deletions tests/TaintTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -2285,6 +2285,25 @@ function sinkNotWorking($sink) : string {}
echo sinkNotWorking($_GET["taint"]);',
'error_message' => 'TaintedHtml',
],
'taintEscapedInTryMightNotWork' => [
'<?php
/** @psalm-taint-escape html */
function escapeHtml(string $arg): string
{
return htmlspecialchars($arg);
}
$tainted = $_GET["foo"];
try {
$tainted = escapeHtml($tainted);
} catch (Throwable $_) {
}
echo $tainted;
',
'error_message' => 'TaintedHtml',
],
];
}

Expand Down
22 changes: 22 additions & 0 deletions tests/UnusedVariableTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -2436,6 +2436,28 @@ function setProxySettingsFromEnv(): void {
$a = false;
}'
],
'usedInCatchIsAlwaysUsedInTry' => [
'<?php
$step = 0;
try {
$step = 1;
$step = 2;
} catch (Throwable $_) {
echo $step;
}
',
],
'usedInFinallyIsAlwaysUsedInTry' => [
'<?php
$step = 0;
try {
$step = 1;
$step = 2;
} finally {
echo $step;
}
',
],
];
}

Expand Down

0 comments on commit 7b1599d

Please sign in to comment.