cond); $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 = []; $project_checker = $statements_checker->getFileChecker()->project_checker; if ($first_if_cond_expr && ExpressionChecker::analyze($statements_checker, $first_if_cond_expr, $context) === false ) { 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 ); $context->inside_conditional = false; $if_scope = new IfScope(); $if_scope->has_elseifs = count($stmt->elseifs) > 0; $if_context = clone $context; $if_context->parent_context = $context; // we need to clone the current context so our ongoing updates to $context don't mess with elseif/else blocks $original_context = clone $context; $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 = []; if ($first_if_cond_expr !== $stmt->cond && ExpressionChecker::analyze($statements_checker, $stmt->cond, $if_context) === false ) { return false; } $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 */ $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); $if_context->inside_conditional = false; $if_clauses = AlgebraChecker::getFormula( $stmt->cond, $context->self, $statements_checker ); // 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); } $if_context->clauses = AlgebraChecker::simplifyCNF(array_merge($context->clauses, $if_clauses)); // 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); $reconcilable_if_types = AlgebraChecker::getTruthsFromFormula($if_context->clauses); // if the if has an || in the conditional, we cannot easily reason about it if ($reconcilable_if_types) { $changed_var_ids = []; $if_vars_in_scope_reconciled = TypeChecker::reconcileKeyedTypes( $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), $statements_checker->getSuppressedIssues() ); $if_context->vars_in_scope = $if_vars_in_scope_reconciled; 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); } } $old_if_context = clone $if_context; $context->vars_possibly_in_scope = array_merge( $if_context->vars_possibly_in_scope, $context->vars_possibly_in_scope ); $context->referenced_var_ids = array_merge( $if_context->referenced_var_ids, $context->referenced_var_ids ); $temp_else_context = clone $original_context; $changed_var_ids = []; if ($if_scope->negated_types) { $else_vars_reconciled = TypeChecker::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), $statements_checker->getSuppressedIssues() ); $temp_else_context->vars_in_scope = $else_vars_reconciled; } // we calculate the vars redefined in a hypothetical else statement to determine // which vars of the if we can safely change $pre_assignment_else_redefined_vars = $temp_else_context->getRedefinedVars($context->vars_in_scope); // check the if self::analyzeIfBlock( $statements_checker, $stmt, $if_scope, $if_context, $old_if_context, $context, $pre_assignment_else_redefined_vars, $loop_scope ); // check the elseifs foreach ($stmt->elseifs as $elseif) { $elseif_context = clone $original_context; self::analyzeElseIfBlock( $statements_checker, $elseif, $if_scope, $elseif_context, $context, $loop_scope ); } // check the else if ($stmt->else) { $else_context = clone $original_context; self::analyzeElseBlock( $statements_checker, $stmt->else, $if_scope, $else_context, $context, $loop_scope ); } 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 ) ); } $context->vars_possibly_in_scope = array_merge( $context->vars_possibly_in_scope, $if_scope->new_vars_possibly_in_scope ); $context->assigned_var_ids = array_merge( $context->assigned_var_ids, $if_context->assigned_var_ids ); $updated_loop_vars = []; // 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); } 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; } } 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); } if ($loop_scope && !in_array(ScopeChecker::ACTION_NONE, $if_scope->final_actions, true)) { $loop_scope->redefined_loop_vars = null; } } if ($if_scope->possibly_redefined_vars) { foreach ($if_scope->possibly_redefined_vars as $var => $type) { 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); } } } return null; } /** * @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 $pre_assignment_else_redefined_vars * * @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, array $pre_assignment_else_redefined_vars, LoopScope $loop_scope = null ) { $final_actions = ScopeChecker::getFinalControlActions($stmt->stmts); $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]; $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 = []; if ($statements_checker->analyze( $stmt->stmts, $if_context, $loop_scope ) === false ) { return false; } /** @var array */ $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; 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 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->forced_new_vars[$var_id] = Type::getMixed(); } } $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 { if (!$has_break_statement) { $if_scope->reasonable_clauses = []; } } if ($has_leaving_statements && !$has_break_statement && !$stmt->else && !$stmt->elseifs) { if ($if_scope->negated_types) { $changed_var_ids = []; $outer_context_vars_reconciled = TypeChecker::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() ); 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 ); } $outer_context->vars_in_scope = $outer_context_vars_reconciled; $mic_drop = true; } $outer_context->clauses = AlgebraChecker::simplifyCNF( array_merge($outer_context->clauses, $if_scope->negated_clauses) ); } // 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) { $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)); } $outer_context->update( $old_if_context, $if_context, $has_leaving_statements, $vars_to_update, $if_scope->updated_vars ); } if (!$has_ending_statements) { $vars = array_diff_key($if_context->vars_possibly_in_scope, $outer_context->vars_possibly_in_scope); if ($loop_scope) { if (!$has_continue_statement && !$has_break_statement) { $if_scope->new_vars_possibly_in_scope = $vars; } $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; } } } /** * @param StatementsChecker $statements_checker * @param PhpParser\Node\Stmt\ElseIf_ $elseif * @param IfScope $if_scope * @param Context $elseif_context * @param Context $outer_context * * @return false|null */ protected static function analyzeElseIfBlock( StatementsChecker $statements_checker, PhpParser\Node\Stmt\ElseIf_ $elseif, IfScope $if_scope, Context $elseif_context, Context $outer_context, LoopScope $loop_scope = null ) { $project_checker = $statements_checker->getFileChecker()->project_checker; $original_context = clone $elseif_context; $entry_clauses = array_merge($original_context->clauses, $if_scope->negated_clauses); if ($if_scope->negated_types) { $changed_var_ids = []; $elseif_vars_reconciled = TypeChecker::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() ); $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; $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 = []; // check the elseif if (ExpressionChecker::analyze($statements_checker, $elseif->cond, $elseif_context) === false) { 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); $elseif_context->inside_conditional = false; $elseif_clauses = AlgebraChecker::getFormula( $elseif->cond, $statements_checker->getFQCLN(), $statements_checker ); // 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 ); $elseif_context->clauses = AlgebraChecker::simplifyCNF( array_merge( $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]; } } } // if the elseif has an || in the conditional, we cannot easily reason about it if ($reconcilable_elseif_types) { $changed_var_ids = []; $elseif_vars_reconciled = TypeChecker::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() ); $elseif_context->vars_in_scope = $elseif_vars_reconciled; if ($changed_var_ids) { $elseif_context->removeReconciledClauses($changed_var_ids); } } $old_elseif_context = clone $elseif_context; if ($statements_checker->analyze( $elseif->stmts, $elseif_context, $loop_scope ) === false ) { return false; } 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; } } } $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)); $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 $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] ); } } } 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 = TypeChecker::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); 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 ); } $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 = array_merge($vars, $if_scope->new_vars_possibly_in_scope); } } } 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) ); } /** * @param StatementsChecker $statements_checker * @param PhpParser\Node\Stmt\Else_ $else * @param IfScope $if_scope * @param Context $else_context * @param Context $outer_context * * @return false|null */ protected static function analyzeElseBlock( StatementsChecker $statements_checker, PhpParser\Node\Stmt\Else_ $else, IfScope $if_scope, Context $else_context, Context $outer_context, LoopScope $loop_scope = null ) { $project_checker = $statements_checker->getFileChecker()->project_checker; $original_context = clone $else_context; $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 = TypeChecker::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() ); $else_context->vars_in_scope = $else_vars_reconciled; $else_context->removeReconciledClauses($changed_var_ids); } $old_else_context = clone $else_context; if ($statements_checker->analyze( $else->stmts, $else_context, $loop_scope ) === false ) { return false; } 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 ); } $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]; $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); // 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] ); } } } 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 ); } } 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; } } } } elseif ($if_scope->reasonable_clauses) { $outer_context->clauses = AlgebraChecker::simplifyCNF( array_merge( $if_scope->reasonable_clauses, $original_context->clauses ) ); } // 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); 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 ); } $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 = array_merge($vars, $if_scope->new_vars_possibly_in_scope); } } } 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]); } } } } /** * @param PhpParser\Node\Expr $stmt * * @return PhpParser\Node\Expr|null */ protected static function getDefinitelyEvaluatedExpression(PhpParser\Node\Expr $stmt) { if ($stmt instanceof PhpParser\Node\Expr\BinaryOp) { if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd || $stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalAnd || $stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalXor ) { return self::getDefinitelyEvaluatedExpression($stmt->left); } return $stmt; } if ($stmt instanceof PhpParser\Node\Expr\BooleanNot) { return self::getDefinitelyEvaluatedExpression($stmt->expr); } return $stmt; } }