1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-16 11:26:55 +01:00
psalm/src/Psalm/Checker/Statements/Block/IfChecker.php

1227 lines
45 KiB
PHP
Raw Normal View History

2016-10-22 19:23:18 +02:00
<?php
namespace Psalm\Checker\Statements\Block;
2016-11-02 07:29:00 +01:00
use PhpParser;
use Psalm\Checker\AlgebraChecker;
2016-10-22 19:23:18 +02:00
use Psalm\Checker\ScopeChecker;
2016-11-02 07:29:00 +01:00
use Psalm\Checker\Statements\ExpressionChecker;
2016-10-22 19:23:18 +02:00
use Psalm\Checker\StatementsChecker;
use Psalm\Checker\TypeChecker;
use Psalm\Clause;
use Psalm\CodeLocation;
2016-10-22 19:23:18 +02:00
use Psalm\Context;
use Psalm\Issue\ConflictingReferenceConstraint;
use Psalm\IssueBuffer;
2017-12-03 00:28:18 +01:00
use Psalm\Scope\IfScope;
use Psalm\Scope\LoopScope;
2016-10-22 19:23:18 +02:00
use Psalm\Type;
use Psalm\Type\Reconciler;
2016-10-22 19:23:18 +02:00
class IfChecker
{
/**
* System of type substitution and deletion
*
* for example
*
* x: A|null
*
* if (x)
* (x: A)
* x = B -- effects: remove A from the type of x, add B
* else
* (x: null)
* x = C -- effects: remove null from the type of x, add C
*
*
* x: A|null
*
* if (!x)
* (x: null)
* throw new Exception -- effects: remove null from the type of x
*
2016-11-02 07:29:00 +01:00
* @param StatementsChecker $statements_checker
2016-10-22 19:23:18 +02:00
* @param PhpParser\Node\Stmt\If_ $stmt
* @param Context $context
2017-05-27 02:16:18 +02:00
*
2016-10-22 19:23:18 +02:00
* @return null|false
*/
public static function analyze(
2016-11-02 07:29:00 +01:00
StatementsChecker $statements_checker,
PhpParser\Node\Stmt\If_ $stmt,
Context $context,
2017-12-03 00:28:18 +01:00
LoopScope $loop_scope = null
2016-11-02 07:29:00 +01:00
) {
2016-10-22 19:23:18 +02:00
// get the first expression in the if, which should be evaluated on its own
// this allows us to update the context of $matches in
// if (!preg_match('/a/', 'aa', $matches)) {
// exit
// }
// echo $matches[0];
2016-12-28 23:04:03 +01:00
$first_if_cond_expr = self::getDefinitelyEvaluatedExpression($stmt->cond);
2016-10-22 19:23:18 +02:00
2017-02-13 00:06:18 +01:00
$context->inside_conditional = true;
$referenced_var_ids = $context->referenced_var_ids;
$context->referenced_var_ids = [];
$pre_assigned_var_ids = $context->assigned_var_ids;
$context->assigned_var_ids = [];
2016-11-02 07:29:00 +01:00
if ($first_if_cond_expr &&
ExpressionChecker::analyze($statements_checker, $first_if_cond_expr, $context) === false
2016-11-02 07:29:00 +01:00
) {
2016-10-22 19:23:18 +02:00
return false;
}
$first_cond_assigned_var_ids = $context->assigned_var_ids;
$context->assigned_var_ids = array_merge(
$pre_assigned_var_ids,
$first_cond_assigned_var_ids
);
$first_cond_referenced_var_ids = $context->referenced_var_ids;
$context->referenced_var_ids = array_merge(
$referenced_var_ids,
$first_cond_referenced_var_ids
);
2017-02-13 00:06:18 +01:00
$context->inside_conditional = false;
$if_scope = new IfScope();
2016-10-22 19:23:18 +02:00
$if_context = clone $context;
$if_context->parent_context = $context;
2016-10-22 19:23:18 +02:00
// we need to clone the current context so our ongoing updates to $context don't mess with elseif/else blocks
$original_context = clone $context;
2017-02-13 00:06:18 +01:00
$if_context->inside_conditional = true;
$assigned_var_ids = $context->assigned_var_ids;
$if_context->assigned_var_ids = [];
$referenced_var_ids = $context->referenced_var_ids;
$if_context->referenced_var_ids = [];
2016-11-02 07:29:00 +01:00
if ($first_if_cond_expr !== $stmt->cond &&
ExpressionChecker::analyze($statements_checker, $stmt->cond, $if_context) === false
2016-11-02 07:29:00 +01:00
) {
2016-10-22 19:23:18 +02:00
return false;
}
/** @var array<string, bool> */
$more_cond_referenced_var_ids = $if_context->referenced_var_ids;
$if_context->referenced_var_ids = array_merge(
$more_cond_referenced_var_ids,
$referenced_var_ids
);
$cond_referenced_var_ids = array_merge(
$first_cond_referenced_var_ids,
$more_cond_referenced_var_ids
);
/** @var array<string, bool> */
$more_cond_assigned_var_ids = $if_context->assigned_var_ids;
$if_context->assigned_var_ids = array_merge(
$more_cond_assigned_var_ids,
$assigned_var_ids
);
$cond_assigned_var_ids = array_merge(
$first_cond_assigned_var_ids,
$more_cond_assigned_var_ids
);
// get all the var ids that were referened in the conditional, but not assigned in it
$cond_referenced_var_ids = array_diff_key($cond_referenced_var_ids, $cond_assigned_var_ids);
2017-02-13 00:06:18 +01:00
$if_context->inside_conditional = false;
$mixed_var_ids = [];
foreach ($if_context->vars_in_scope as $var_id => $type) {
if ($type->isMixed()) {
$mixed_var_ids[] = $var_id;
}
}
$if_clauses = AlgebraChecker::getFormula(
$stmt->cond,
2017-01-07 20:35:07 +01:00
$context->self,
$statements_checker
);
2016-10-22 19:23:18 +02:00
$if_clauses = array_values(
array_filter(
$if_clauses,
/** @return bool */
function (Clause $c) use ($mixed_var_ids) {
$keys = array_keys($c->possibilities);
foreach ($keys as $key) {
foreach ($mixed_var_ids as $mixed_var_id) {
if (preg_match('/^' . preg_quote($mixed_var_id, '/') . '(\[|-)/', $key)) {
return false;
}
}
}
return true;
}
)
);
if (!$if_clauses) {
$if_clauses = [new Clause([], true)];
}
2017-04-02 21:26:10 +02:00
// this will see whether any of the clauses in set A conflict with the clauses in set B
AlgebraChecker::checkForParadox(
$context->clauses,
$if_clauses,
$statements_checker,
$stmt->cond,
$cond_assigned_var_ids
);
// if we have assignments in the if, we may have duplicate clauses
if ($cond_assigned_var_ids) {
$if_clauses = AlgebraChecker::simplifyCNF($if_clauses);
}
2017-04-02 21:26:10 +02:00
$if_context->clauses = AlgebraChecker::simplifyCNF(array_merge($context->clauses, $if_clauses));
2016-10-22 19:23:18 +02:00
// define this before we alter local claues after reconciliation
$if_scope->reasonable_clauses = $if_context->clauses;
$if_scope->negated_clauses = AlgebraChecker::negateFormula($if_clauses);
$if_scope->negated_types = AlgebraChecker::getTruthsFromFormula($if_scope->negated_clauses);
2016-10-22 19:23:18 +02:00
$reconcilable_if_types = AlgebraChecker::getTruthsFromFormula($if_context->clauses);
2016-10-22 19:23:18 +02:00
// if the if has an || in the conditional, we cannot easily reason about it
if ($reconcilable_if_types) {
$changed_var_ids = [];
2016-10-22 19:23:18 +02:00
$if_vars_in_scope_reconciled =
Reconciler::reconcileKeyedTypes(
2016-10-22 19:23:18 +02:00
$reconcilable_if_types,
$if_context->vars_in_scope,
$changed_var_ids,
$cond_referenced_var_ids,
$statements_checker,
new CodeLocation($statements_checker->getSource(), $stmt->cond, $context->include_location),
2016-10-22 19:23:18 +02:00
$statements_checker->getSuppressedIssues()
);
$if_context->vars_in_scope = $if_vars_in_scope_reconciled;
2017-12-02 19:32:20 +01:00
foreach ($reconcilable_if_types as $var_id => $type) {
$if_context->vars_possibly_in_scope[$var_id] = true;
}
if ($changed_var_ids) {
$if_context->removeReconciledClauses($changed_var_ids);
}
2016-10-22 19:23:18 +02:00
}
$old_if_context = clone $if_context;
2016-11-02 07:29:00 +01:00
$context->vars_possibly_in_scope = array_merge(
$if_context->vars_possibly_in_scope,
$context->vars_possibly_in_scope
);
2016-10-22 19:23:18 +02:00
$context->referenced_var_ids = array_merge(
$if_context->referenced_var_ids,
$context->referenced_var_ids
);
$temp_else_context = clone $original_context;
2016-10-22 19:23:18 +02:00
$changed_var_ids = [];
if ($if_scope->negated_types) {
$else_vars_reconciled = Reconciler::reconcileKeyedTypes(
$if_scope->negated_types,
$temp_else_context->vars_in_scope,
$changed_var_ids,
$cond_referenced_var_ids,
$statements_checker,
new CodeLocation($statements_checker->getSource(), $stmt->cond, $context->include_location),
2016-10-22 19:23:18 +02:00
$statements_checker->getSuppressedIssues()
);
$temp_else_context->vars_in_scope = $else_vars_reconciled;
2016-10-22 19:23:18 +02:00
}
// we calculate the vars redefined in a hypothetical else statement to determine
// which vars of the if we can safely change
2017-12-03 00:28:18 +01:00
$pre_assignment_else_redefined_vars = $temp_else_context->getRedefinedVars($context->vars_in_scope);
// check the if
if (self::analyzeIfBlock(
$statements_checker,
$stmt,
$if_scope,
$if_context,
$old_if_context,
$context,
2017-12-03 00:28:18 +01:00
$pre_assignment_else_redefined_vars,
$loop_scope
) === false) {
return false;
}
2016-10-22 19:23:18 +02:00
// check the elseifs
foreach ($stmt->elseifs as $elseif) {
$elseif_context = clone $original_context;
if (self::analyzeElseIfBlock(
$statements_checker,
$elseif,
$if_scope,
$elseif_context,
2017-12-03 00:28:18 +01:00
$context,
$loop_scope
) === false) {
return false;
}
2016-10-22 19:23:18 +02:00
}
// check the else
if ($stmt->else) {
$else_context = clone $original_context;
if (self::analyzeElseBlock(
$statements_checker,
$stmt->else,
$if_scope,
$else_context,
2017-12-03 00:28:18 +01:00
$context,
$loop_scope
) === false) {
return false;
}
2017-12-03 00:28:18 +01:00
} else {
$if_scope->final_actions[] = ScopeChecker::ACTION_NONE;
}
if ($loop_scope) {
$loop_scope->final_actions = array_unique(
array_merge(
$loop_scope->final_actions,
$if_scope->final_actions
)
);
}
2016-10-22 19:23:18 +02:00
$context->vars_possibly_in_scope = array_merge(
$context->vars_possibly_in_scope,
$if_scope->new_vars_possibly_in_scope
);
2016-10-22 19:23:18 +02:00
2017-11-25 17:21:45 +01:00
$context->assigned_var_ids = array_merge(
$context->assigned_var_ids,
$if_context->assigned_var_ids
);
// vars can only be defined/redefined if there was an else (defined in every block)
if ($stmt->else) {
if ($if_scope->new_vars) {
$context->vars_in_scope = array_merge($context->vars_in_scope, $if_scope->new_vars);
}
2016-10-22 19:23:18 +02:00
if ($if_scope->redefined_vars) {
foreach ($if_scope->redefined_vars as $var => $type) {
$context->vars_in_scope[$var] = $type;
$if_scope->updated_vars[$var] = true;
2016-10-22 19:23:18 +02:00
}
}
2016-10-22 19:23:18 +02:00
if ($if_scope->possible_param_types) {
foreach ($if_scope->possible_param_types as $var => $type) {
$context->possible_param_types[$var] = $type;
}
}
} else {
if ($if_scope->forced_new_vars) {
$context->vars_in_scope = array_merge($context->vars_in_scope, $if_scope->forced_new_vars);
2016-10-22 19:23:18 +02:00
}
2017-12-03 00:28:18 +01:00
if ($loop_scope && !in_array(ScopeChecker::ACTION_NONE, $if_scope->final_actions, true)) {
$loop_scope->redefined_loop_vars = null;
}
}
2016-10-22 19:23:18 +02:00
if ($if_scope->possibly_redefined_vars) {
foreach ($if_scope->possibly_redefined_vars as $var => $type) {
2017-03-13 23:06:56 +01:00
if (!$type->failed_reconciliation &&
$context->hasVariable($var) &&
!isset($if_scope->updated_vars[$var])
) {
$context->vars_in_scope[$var] = Type::combineUnionTypes($context->vars_in_scope[$var], $type);
2016-10-22 19:23:18 +02:00
}
}
}
2016-10-22 19:23:18 +02:00
return null;
}
2016-10-22 19:23:18 +02:00
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Stmt\If_ $stmt
* @param IfScope $if_scope
* @param Context $if_context
* @param Context $old_if_context
* @param Context $outer_context
* @param array<string,Type\Union> $pre_assignment_else_redefined_vars
2017-05-27 02:16:18 +02:00
*
* @return false|null
*/
protected static function analyzeIfBlock(
StatementsChecker $statements_checker,
PhpParser\Node\Stmt\If_ $stmt,
IfScope $if_scope,
Context $if_context,
Context $old_if_context,
Context $outer_context,
2017-12-03 00:28:18 +01:00
array $pre_assignment_else_redefined_vars,
LoopScope $loop_scope = null
) {
$final_actions = ScopeChecker::getFinalControlActions($stmt->stmts);
2016-10-22 19:23:18 +02:00
$has_ending_statements = $final_actions === [ScopeChecker::ACTION_END];
$has_leaving_statements = $has_ending_statements
|| (count($final_actions) && !in_array(ScopeChecker::ACTION_NONE, $final_actions, true));
2017-12-03 00:28:18 +01:00
$has_break_statement = $final_actions === [ScopeChecker::ACTION_BREAK];
$has_continue_statement = $final_actions === [ScopeChecker::ACTION_CONTINUE];
$if_scope->final_actions = $final_actions;
$project_checker = $statements_checker->getFileChecker()->project_checker;
$assigned_var_ids = $if_context->assigned_var_ids;
$if_context->assigned_var_ids = [];
2017-12-03 00:28:18 +01:00
if ($statements_checker->analyze(
$stmt->stmts,
$if_context,
$loop_scope
) === false
) {
return false;
}
/** @var array<string, bool> */
$new_assigned_var_ids = $if_context->assigned_var_ids;
$if_context->assigned_var_ids = $assigned_var_ids;
if ($if_context->byref_constraints !== null) {
foreach ($if_context->byref_constraints as $var_id => $byref_constraint) {
if ($outer_context->byref_constraints !== null &&
isset($outer_context->byref_constraints[$var_id]) &&
!TypeChecker::isContainedBy(
$project_checker,
$byref_constraint->type,
$outer_context->byref_constraints[$var_id]->type
)
) {
if (IssueBuffer::accepts(
new ConflictingReferenceConstraint(
'There is more than one pass-by--reference constraint on ' . $var_id,
new CodeLocation($statements_checker, $stmt, $outer_context->include_location, true)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
} else {
$outer_context->byref_constraints[$var_id] = $byref_constraint;
}
}
}
if ($outer_context->collect_references) {
$outer_context->referenced_var_ids = array_merge(
$outer_context->referenced_var_ids,
$if_context->referenced_var_ids
);
}
$mic_drop = false;
2016-11-02 07:29:00 +01:00
if (!$has_leaving_statements) {
$if_scope->new_vars = array_diff_key($if_context->vars_in_scope, $outer_context->vars_in_scope);
// if we have a check like if (!isset($a)) { $a = true; } we want to make sure $a is always set
2017-02-11 01:10:13 +01:00
foreach ($if_scope->new_vars as $var_id => $_) {
if (isset($if_scope->negated_types[$var_id])
&& (
$if_scope->negated_types[$var_id] === 'isset'
|| $if_scope->negated_types[$var_id] === '^isset'
|| $if_scope->negated_types[$var_id] === '!empty'
)
) {
$if_scope->forced_new_vars[$var_id] = Type::getMixed();
}
2016-10-22 19:23:18 +02:00
}
2017-12-03 00:28:18 +01:00
$if_scope->redefined_vars = $if_context->getRedefinedVars($outer_context->vars_in_scope);
$if_scope->possibly_redefined_vars = $if_scope->redefined_vars;
$changed_var_ids = array_keys($new_assigned_var_ids);
if ($if_scope->reasonable_clauses) {
// remove all reasonable clauses that would be negated by the if stmts
foreach ($changed_var_ids as $var_id) {
$if_scope->reasonable_clauses = Context::filterClauses(
$var_id,
$if_scope->reasonable_clauses,
isset($if_context->vars_in_scope[$var_id]) ? $if_context->vars_in_scope[$var_id] : null,
$statements_checker
);
}
}
if ($project_checker->infer_types_from_usage) {
$if_scope->possible_param_types = $if_context->possible_param_types;
}
} else {
2017-12-03 00:28:18 +01:00
if (!$has_break_statement) {
$if_scope->reasonable_clauses = [];
}
}
2017-12-03 00:28:18 +01:00
if ($has_leaving_statements && !$has_break_statement && !$stmt->else && !$stmt->elseifs) {
if ($if_scope->negated_types) {
$changed_var_ids = [];
$outer_context_vars_reconciled = Reconciler::reconcileKeyedTypes(
$if_scope->negated_types,
$outer_context->vars_in_scope,
$changed_var_ids,
[],
$statements_checker,
new CodeLocation(
$statements_checker->getSource(),
$stmt->cond,
$outer_context->include_location,
false
),
$statements_checker->getSuppressedIssues()
);
2016-11-02 07:29:00 +01:00
foreach ($changed_var_ids as $changed_var_id) {
$outer_context->removeVarFromConflictingClauses($changed_var_id);
}
$changed_var_ids = array_unique(
array_merge(
$changed_var_ids,
array_keys($new_assigned_var_ids)
)
);
foreach ($changed_var_ids as $var_id) {
$if_scope->negated_clauses = Context::filterClauses(
$var_id,
$if_scope->negated_clauses
);
}
2016-12-29 00:55:16 +01:00
$outer_context->vars_in_scope = $outer_context_vars_reconciled;
$mic_drop = true;
2016-10-22 19:23:18 +02:00
}
$outer_context->clauses = AlgebraChecker::simplifyCNF(
array_merge($outer_context->clauses, $if_scope->negated_clauses)
);
}
2016-10-22 19:23:18 +02:00
// update the parent context as necessary, but only if we can safely reason about type negation.
// We only update vars that changed both at the start of the if block and then again by an assignment
// in the if statement.
if ($if_scope->negated_types && !$mic_drop) {
2016-12-29 00:55:16 +01:00
$vars_to_update = array_intersect(
array_keys($pre_assignment_else_redefined_vars),
array_keys($if_scope->negated_types)
);
$extra_vars_to_update = [];
// if there's an object-like array in there, we also need to update the root array variable
foreach ($vars_to_update as $var_id) {
$bracked_pos = strpos($var_id, '[');
if ($bracked_pos !== false) {
$extra_vars_to_update[] = substr($var_id, 0, $bracked_pos);
}
}
if ($extra_vars_to_update) {
$vars_to_update = array_unique(array_merge($extra_vars_to_update, $vars_to_update));
}
//update $if_context vars to include the pre-assignment else vars
if (!$stmt->else && !$has_leaving_statements) {
foreach ($pre_assignment_else_redefined_vars as $var_id => $type) {
if (isset($if_context->vars_in_scope[$var_id])) {
$if_context->vars_in_scope[$var_id] = Type::combineUnionTypes(
$if_context->vars_in_scope[$var_id],
$type
);
}
}
}
$outer_context->update(
$old_if_context,
$if_context,
$has_leaving_statements,
2016-12-29 00:55:16 +01:00
$vars_to_update,
$if_scope->updated_vars
2016-11-02 07:29:00 +01:00
);
}
2016-10-22 19:23:18 +02:00
if (!$has_ending_statements) {
$vars = array_diff_key($if_context->vars_possibly_in_scope, $outer_context->vars_possibly_in_scope);
2017-12-03 00:28:18 +01:00
if ($loop_scope) {
if (!$has_continue_statement && !$has_break_statement) {
$if_scope->new_vars_possibly_in_scope = $vars;
2016-10-22 19:23:18 +02:00
}
2017-12-03 00:28:18 +01:00
$loop_scope->vars_possibly_in_scope = array_merge(
$vars,
$loop_scope->vars_possibly_in_scope
);
} elseif (!$has_leaving_statements) {
$if_scope->new_vars_possibly_in_scope = $vars;
2016-10-22 19:23:18 +02:00
}
}
}
2016-10-22 19:23:18 +02:00
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Stmt\ElseIf_ $elseif
* @param IfScope $if_scope
* @param Context $elseif_context
* @param Context $outer_context
2017-05-27 02:16:18 +02:00
*
* @return false|null
*/
protected static function analyzeElseIfBlock(
StatementsChecker $statements_checker,
PhpParser\Node\Stmt\ElseIf_ $elseif,
IfScope $if_scope,
Context $elseif_context,
2017-12-03 00:28:18 +01:00
Context $outer_context,
LoopScope $loop_scope = null
) {
$project_checker = $statements_checker->getFileChecker()->project_checker;
$original_context = clone $elseif_context;
2016-10-22 19:23:18 +02:00
$entry_clauses = array_merge($original_context->clauses, $if_scope->negated_clauses);
if ($if_scope->negated_types) {
$changed_var_ids = [];
$elseif_vars_reconciled = Reconciler::reconcileKeyedTypes(
$if_scope->negated_types,
$elseif_context->vars_in_scope,
$changed_var_ids,
[],
$statements_checker,
new CodeLocation(
$statements_checker->getSource(),
$elseif->cond,
$outer_context->include_location,
false
),
$statements_checker->getSuppressedIssues()
);
2016-10-22 19:23:18 +02:00
$elseif_context->vars_in_scope = $elseif_vars_reconciled;
if ($changed_var_ids) {
$entry_clauses = array_filter(
$entry_clauses,
/** @return bool */
function (Clause $c) use ($changed_var_ids) {
return count($c->possibilities) > 1
|| !in_array(array_keys($c->possibilities)[0], $changed_var_ids, true);
}
);
}
}
$pre_conditional_context = clone $elseif_context;
2017-02-13 00:06:18 +01:00
$elseif_context->inside_conditional = true;
$pre_assigned_var_ids = $elseif_context->assigned_var_ids;
$referenced_var_ids = $elseif_context->referenced_var_ids;
$elseif_context->referenced_var_ids = [];
2016-12-09 18:06:14 +01:00
// check the elseif
if (ExpressionChecker::analyze($statements_checker, $elseif->cond, $elseif_context) === false) {
2016-12-09 18:06:14 +01:00
return false;
}
$new_referenced_var_ids = $elseif_context->referenced_var_ids;
$elseif_context->referenced_var_ids = array_merge(
$referenced_var_ids,
$elseif_context->referenced_var_ids
);
$new_assigned_var_ids = array_diff_key($elseif_context->assigned_var_ids, $pre_assigned_var_ids);
$new_referenced_var_ids = array_diff_key($new_referenced_var_ids, $new_assigned_var_ids);
2017-02-13 00:06:18 +01:00
$elseif_context->inside_conditional = false;
$mixed_var_ids = [];
foreach ($elseif_context->vars_in_scope as $var_id => $type) {
if ($type->isMixed()) {
$mixed_var_ids[] = $var_id;
}
}
$elseif_clauses = AlgebraChecker::getFormula(
$elseif->cond,
$statements_checker->getFQCLN(),
2017-01-07 20:35:07 +01:00
$statements_checker
);
$elseif_clauses = array_values(
array_filter(
$elseif_clauses,
/** @return bool */
function (Clause $c) use ($mixed_var_ids) {
$keys = array_keys($c->possibilities);
foreach ($keys as $key) {
foreach ($mixed_var_ids as $mixed_var_id) {
if (preg_match('/^' . preg_quote($mixed_var_id) . '(\[|-)/', $key)) {
return false;
}
}
}
return true;
}
)
);
if (!$elseif_clauses) {
$elseif_clauses = [new Clause([], true)];
}
2017-04-02 21:26:10 +02:00
// this will see whether any of the clauses in set A conflict with the clauses in set B
AlgebraChecker::checkForParadox(
$entry_clauses,
$elseif_clauses,
$statements_checker,
$elseif->cond,
$new_assigned_var_ids
);
2017-04-02 21:26:10 +02:00
$elseif_context->clauses = AlgebraChecker::simplifyCNF(
array_merge(
2017-04-02 21:26:10 +02:00
$entry_clauses,
$elseif_clauses
)
);
$reconcilable_elseif_types = AlgebraChecker::getTruthsFromFormula($elseif_context->clauses);
$negated_elseif_types = AlgebraChecker::getTruthsFromFormula(AlgebraChecker::negateFormula($elseif_clauses));
$all_negated_vars = array_unique(
array_merge(
array_keys($negated_elseif_types),
array_keys($if_scope->negated_types)
)
);
foreach ($all_negated_vars as $var_id) {
if (isset($negated_elseif_types[$var_id])) {
if (isset($if_scope->negated_types[$var_id])) {
$if_scope->negated_types[$var_id] = $if_scope->negated_types[$var_id] . '&' .
$negated_elseif_types[$var_id];
} else {
$if_scope->negated_types[$var_id] = $negated_elseif_types[$var_id];
}
2016-10-22 19:23:18 +02:00
}
}
2016-10-22 19:23:18 +02:00
// if the elseif has an || in the conditional, we cannot easily reason about it
if ($reconcilable_elseif_types) {
$changed_var_ids = [];
$elseif_vars_reconciled = Reconciler::reconcileKeyedTypes(
$reconcilable_elseif_types,
$elseif_context->vars_in_scope,
$changed_var_ids,
$new_referenced_var_ids,
$statements_checker,
new CodeLocation($statements_checker->getSource(), $elseif->cond, $outer_context->include_location),
$statements_checker->getSuppressedIssues()
);
2016-10-22 19:23:18 +02:00
$elseif_context->vars_in_scope = $elseif_vars_reconciled;
if ($changed_var_ids) {
$elseif_context->removeReconciledClauses($changed_var_ids);
}
}
$old_elseif_context = clone $elseif_context;
2016-10-22 19:23:18 +02:00
2017-12-03 00:28:18 +01:00
if ($statements_checker->analyze(
2018-01-03 03:23:48 +01:00
$elseif->stmts,
$elseif_context,
$loop_scope
) === false
2017-12-03 00:28:18 +01:00
) {
return false;
}
2016-10-22 19:23:18 +02:00
if ($elseif_context->byref_constraints !== null) {
foreach ($elseif_context->byref_constraints as $var_id => $byref_constraint) {
if ($outer_context->byref_constraints !== null &&
isset($outer_context->byref_constraints[$var_id]) &&
!TypeChecker::isContainedBy(
$project_checker,
$byref_constraint->type,
$outer_context->byref_constraints[$var_id]->type
)
) {
if (IssueBuffer::accepts(
new ConflictingReferenceConstraint(
'There is more than one pass-by--reference constraint on ' . $var_id,
new CodeLocation($statements_checker, $elseif, $outer_context->include_location, true)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
} else {
$outer_context->byref_constraints[$var_id] = $byref_constraint;
}
}
}
2017-12-03 00:28:18 +01:00
$final_actions = ScopeChecker::getFinalControlActions($elseif->stmts);
// has a return/throw at end
$has_ending_statements = $final_actions === [ScopeChecker::ACTION_END];
$has_leaving_statements = $has_ending_statements
|| (count($final_actions) && !in_array(ScopeChecker::ACTION_NONE, $final_actions, true));
2017-12-03 00:28:18 +01:00
$has_break_statement = $final_actions === [ScopeChecker::ACTION_BREAK];
$has_continue_statement = $final_actions === [ScopeChecker::ACTION_CONTINUE];
$if_scope->final_actions = array_merge($final_actions, $if_scope->final_actions);
if (count($elseif->stmts)) {
// update the parent context as necessary
2017-12-03 00:28:18 +01:00
$elseif_redefined_vars = $elseif_context->getRedefinedVars($original_context->vars_in_scope);
if (!$has_leaving_statements) {
if ($if_scope->new_vars === null) {
$if_scope->new_vars = array_diff_key($elseif_context->vars_in_scope, $outer_context->vars_in_scope);
} else {
foreach ($if_scope->new_vars as $new_var => $type) {
if (!$elseif_context->hasVariable($new_var)) {
unset($if_scope->new_vars[$new_var]);
} else {
$if_scope->new_vars[$new_var] = Type::combineUnionTypes(
$type,
$elseif_context->vars_in_scope[$new_var]
);
2016-10-22 19:23:18 +02:00
}
}
}
2016-10-22 19:23:18 +02:00
if ($if_scope->redefined_vars === null) {
$if_scope->redefined_vars = $elseif_redefined_vars;
$if_scope->possibly_redefined_vars = $if_scope->redefined_vars;
} else {
foreach ($if_scope->redefined_vars as $redefined_var => $type) {
if (!isset($elseif_redefined_vars[$redefined_var])) {
unset($if_scope->redefined_vars[$redefined_var]);
} else {
$if_scope->redefined_vars[$redefined_var] = Type::combineUnionTypes(
$elseif_redefined_vars[$redefined_var],
$type
);
}
}
foreach ($elseif_redefined_vars as $var => $type) {
if ($type->isMixed()) {
$if_scope->possibly_redefined_vars[$var] = $type;
} elseif (isset($if_scope->possibly_redefined_vars[$var])) {
$if_scope->possibly_redefined_vars[$var] = Type::combineUnionTypes(
$type,
$if_scope->possibly_redefined_vars[$var]
);
} else {
$if_scope->possibly_redefined_vars[$var] = $type;
}
}
}
if ($if_scope->reasonable_clauses && $elseif_clauses) {
$if_scope->reasonable_clauses = AlgebraChecker::combineOredClauses(
$if_scope->reasonable_clauses,
$elseif_clauses
);
} else {
$if_scope->reasonable_clauses = [];
}
} else {
$if_scope->reasonable_clauses = [];
}
if ($project_checker->infer_types_from_usage) {
$elseif_possible_param_types = $elseif_context->possible_param_types;
if ($if_scope->possible_param_types) {
$vars_to_remove = [];
foreach ($if_scope->possible_param_types as $var => $type) {
if (isset($elseif_possible_param_types[$var])) {
$if_scope->possible_param_types[$var] = Type::combineUnionTypes(
$elseif_possible_param_types[$var],
$type
);
} else {
$vars_to_remove[] = $var;
}
}
foreach ($vars_to_remove as $var) {
unset($if_scope->possible_param_types[$var]);
}
}
}
if ($negated_elseif_types) {
if ($has_leaving_statements) {
$changed_var_ids = [];
$leaving_vars_reconciled = Reconciler::reconcileKeyedTypes(
$negated_elseif_types,
$pre_conditional_context->vars_in_scope,
$changed_var_ids,
[],
$statements_checker,
new CodeLocation($statements_checker->getSource(), $elseif, $outer_context->include_location),
$statements_checker->getSuppressedIssues()
);
$implied_outer_context = clone $elseif_context;
$implied_outer_context->vars_in_scope = $leaving_vars_reconciled;
$outer_context->update(
$elseif_context,
$implied_outer_context,
false,
array_keys($negated_elseif_types),
$if_scope->updated_vars
);
} else {
$outer_context->update(
$old_elseif_context,
$elseif_context,
false,
array_keys($negated_elseif_types),
$if_scope->updated_vars
);
}
}
if (!$has_ending_statements) {
$vars = array_diff_key($elseif_context->vars_possibly_in_scope, $outer_context->vars_possibly_in_scope);
2017-12-03 00:28:18 +01:00
if ($has_leaving_statements && $loop_scope) {
if (!$has_continue_statement && !$has_break_statement) {
$if_scope->new_vars_possibly_in_scope = array_merge(
$vars,
$if_scope->new_vars_possibly_in_scope
);
2016-10-22 19:23:18 +02:00
}
2017-12-03 00:28:18 +01:00
$loop_scope->vars_possibly_in_scope = array_merge(
$vars,
2017-12-03 00:28:18 +01:00
$loop_scope->vars_possibly_in_scope
2016-11-02 07:29:00 +01:00
);
} elseif (!$has_leaving_statements) {
$if_scope->new_vars_possibly_in_scope = array_merge($vars, $if_scope->new_vars_possibly_in_scope);
2016-10-22 19:23:18 +02:00
}
}
}
if ($outer_context->collect_references) {
$outer_context->referenced_var_ids = array_merge(
$outer_context->referenced_var_ids,
$elseif_context->referenced_var_ids
);
}
$if_scope->negated_clauses = array_merge(
$if_scope->negated_clauses,
AlgebraChecker::negateFormula($elseif_clauses)
);
}
2016-10-22 19:23:18 +02:00
/**
* @param StatementsChecker $statements_checker
* @param PhpParser\Node\Stmt\Else_ $else
* @param IfScope $if_scope
* @param Context $else_context
* @param Context $outer_context
2017-05-27 02:16:18 +02:00
*
* @return false|null
*/
protected static function analyzeElseBlock(
StatementsChecker $statements_checker,
PhpParser\Node\Stmt\Else_ $else,
IfScope $if_scope,
Context $else_context,
2017-12-03 00:28:18 +01:00
Context $outer_context,
LoopScope $loop_scope = null
) {
$project_checker = $statements_checker->getFileChecker()->project_checker;
$original_context = clone $else_context;
2016-10-22 19:23:18 +02:00
$else_context->clauses = AlgebraChecker::simplifyCNF(
array_merge(
$outer_context->clauses,
$if_scope->negated_clauses
)
);
$else_types = AlgebraChecker::getTruthsFromFormula($else_context->clauses);
if ($else_types) {
$changed_var_ids = [];
$else_vars_reconciled = Reconciler::reconcileKeyedTypes(
$else_types,
$else_context->vars_in_scope,
$changed_var_ids,
[],
$statements_checker,
new CodeLocation($statements_checker->getSource(), $else, $outer_context->include_location),
$statements_checker->getSuppressedIssues()
);
2016-10-22 19:23:18 +02:00
$else_context->vars_in_scope = $else_vars_reconciled;
$else_context->removeReconciledClauses($changed_var_ids);
}
2016-10-22 19:23:18 +02:00
$old_else_context = clone $else_context;
2016-11-02 07:29:00 +01:00
2017-12-03 00:28:18 +01:00
if ($statements_checker->analyze(
2018-01-03 03:23:48 +01:00
$else->stmts,
$else_context,
$loop_scope
) === false
2017-12-03 00:28:18 +01:00
) {
return false;
}
2016-10-22 19:23:18 +02:00
if ($else_context->byref_constraints !== null) {
$project_checker = $statements_checker->getFileChecker()->project_checker;
foreach ($else_context->byref_constraints as $var_id => $byref_constraint) {
if ($outer_context->byref_constraints !== null &&
isset($outer_context->byref_constraints[$var_id]) &&
!TypeChecker::isContainedBy(
$project_checker,
$byref_constraint->type,
$outer_context->byref_constraints[$var_id]->type
)
) {
if (IssueBuffer::accepts(
new ConflictingReferenceConstraint(
'There is more than one pass-by--reference constraint on ' . $var_id,
new CodeLocation($statements_checker, $else, $outer_context->include_location, true)
),
$statements_checker->getSuppressedIssues()
)) {
// fall through
}
} else {
$outer_context->byref_constraints[$var_id] = $byref_constraint;
}
}
}
if ($outer_context->collect_references) {
$outer_context->referenced_var_ids = array_merge(
$outer_context->referenced_var_ids,
$else_context->referenced_var_ids
2017-02-02 06:45:23 +01:00
);
}
2017-12-03 00:28:18 +01:00
$final_actions = ScopeChecker::getFinalControlActions($else->stmts);
// has a return/throw at end
$has_ending_statements = $final_actions === [ScopeChecker::ACTION_END];
$has_leaving_statements = $has_ending_statements
|| (count($final_actions) && !in_array(ScopeChecker::ACTION_NONE, $final_actions, true));
$has_break_statement = $final_actions === [ScopeChecker::ACTION_BREAK];
$has_continue_statement = $final_actions === [ScopeChecker::ACTION_CONTINUE];
2016-10-22 19:23:18 +02:00
2017-12-03 00:28:18 +01:00
$if_scope->final_actions = array_merge($final_actions, $if_scope->final_actions);
if (count($else->stmts)) {
$else_redefined_vars = $else_context->getRedefinedVars($original_context->vars_in_scope);
2016-10-22 19:23:18 +02:00
// if it doesn't end in a return
if (!$has_leaving_statements) {
if ($if_scope->new_vars === null) {
$if_scope->new_vars = array_diff_key($else_context->vars_in_scope, $outer_context->vars_in_scope);
} else {
foreach ($if_scope->new_vars as $new_var => $type) {
if (!$else_context->hasVariable($new_var)) {
unset($if_scope->new_vars[$new_var]);
} else {
$if_scope->new_vars[$new_var] = Type::combineUnionTypes(
$type,
$else_context->vars_in_scope[$new_var]
);
}
}
}
2016-10-22 19:23:18 +02:00
if ($if_scope->redefined_vars === null) {
$if_scope->redefined_vars = $else_redefined_vars;
$if_scope->possibly_redefined_vars = $if_scope->redefined_vars;
} else {
foreach ($if_scope->redefined_vars as $redefined_var => $type) {
if (!isset($else_redefined_vars[$redefined_var])) {
unset($if_scope->redefined_vars[$redefined_var]);
} else {
$if_scope->redefined_vars[$redefined_var] = Type::combineUnionTypes(
$else_redefined_vars[$redefined_var],
$type
);
}
}
2016-10-22 19:23:18 +02:00
foreach ($else_redefined_vars as $var => $type) {
if ($type->isMixed()) {
$if_scope->possibly_redefined_vars[$var] = $type;
} elseif (isset($if_scope->possibly_redefined_vars[$var])) {
$if_scope->possibly_redefined_vars[$var] = Type::combineUnionTypes(
$type,
$if_scope->possibly_redefined_vars[$var]
);
} else {
$if_scope->possibly_redefined_vars[$var] = $type;
2016-10-22 19:23:18 +02:00
}
}
}
} elseif ($if_scope->reasonable_clauses) {
$outer_context->clauses = AlgebraChecker::simplifyCNF(
array_merge(
$if_scope->reasonable_clauses,
$original_context->clauses
)
);
}
2016-10-22 19:23:18 +02:00
// update the parent context as necessary
if ($if_scope->negatable_if_types) {
$outer_context->update(
$old_else_context,
$else_context,
$has_leaving_statements,
array_keys($if_scope->negatable_if_types),
$if_scope->updated_vars
);
}
if (!$has_ending_statements) {
$vars = array_diff_key($else_context->vars_possibly_in_scope, $outer_context->vars_possibly_in_scope);
2017-12-03 00:28:18 +01:00
if ($has_leaving_statements && $loop_scope) {
if (!$has_continue_statement && !$has_break_statement) {
$if_scope->new_vars_possibly_in_scope = array_merge(
$vars,
$if_scope->new_vars_possibly_in_scope
);
2016-10-22 19:23:18 +02:00
}
2017-12-03 00:28:18 +01:00
$loop_scope->vars_possibly_in_scope = array_merge(
$vars,
2017-12-03 00:28:18 +01:00
$loop_scope->vars_possibly_in_scope
2016-11-02 07:29:00 +01:00
);
} elseif (!$has_leaving_statements) {
$if_scope->new_vars_possibly_in_scope = array_merge($vars, $if_scope->new_vars_possibly_in_scope);
2016-10-22 19:23:18 +02:00
}
}
}
if ($project_checker->infer_types_from_usage) {
$else_possible_param_types = $else_context->possible_param_types;
if ($if_scope->possible_param_types) {
$vars_to_remove = [];
foreach ($if_scope->possible_param_types as $var => $type) {
if (isset($else_possible_param_types[$var])) {
$if_scope->possible_param_types[$var] = Type::combineUnionTypes(
$else_possible_param_types[$var],
$type
);
} else {
$vars_to_remove[] = $var;
}
}
foreach ($vars_to_remove as $var) {
unset($if_scope->possible_param_types[$var]);
}
}
}
2016-10-22 19:23:18 +02:00
}
/**
* @param PhpParser\Node\Expr $stmt
2017-05-27 02:16:18 +02:00
*
2016-10-22 19:23:18 +02:00
* @return PhpParser\Node\Expr|null
*/
2016-12-28 23:04:03 +01:00
protected static function getDefinitelyEvaluatedExpression(PhpParser\Node\Expr $stmt)
2016-10-22 19:23:18 +02:00
{
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp) {
2016-12-28 23:04:03 +01:00
if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd ||
2017-02-17 03:00:45 +01:00
$stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalAnd ||
2016-12-28 23:04:03 +01:00
$stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalXor
) {
return self::getDefinitelyEvaluatedExpression($stmt->left);
}
return $stmt;
}
2016-10-22 19:23:18 +02:00
if ($stmt instanceof PhpParser\Node\Expr\BooleanNot) {
2016-12-28 23:04:03 +01:00
return self::getDefinitelyEvaluatedExpression($stmt->expr);
2016-10-22 19:23:18 +02:00
}
2017-01-07 20:35:07 +01:00
return $stmt;
2016-10-22 19:23:18 +02:00
}
}