getCodebase(); $if_scope = new IfScope(); // We need to clone the original context for later use if we're exiting in this if conditional if ($stmt->cond instanceof PhpParser\Node\Expr\BinaryOp || ($stmt->cond instanceof PhpParser\Node\Expr\BooleanNot && $stmt->cond->expr instanceof PhpParser\Node\Expr\BinaryOp) ) { $final_actions = ScopeAnalyzer::getControlActions( $stmt->stmts, null, $codebase->config->exit_functions, $context->break_types ); $has_leaving_statements = $final_actions === [ScopeAnalyzer::ACTION_END] || (count($final_actions) && !in_array(ScopeAnalyzer::ACTION_NONE, $final_actions, true)); if ($has_leaving_statements) { $if_scope->post_leaving_if_context = clone $context; } } try { $if_conditional_scope = IfConditionalAnalyzer::analyze( $statements_analyzer, $stmt->cond, $context, $codebase, $if_scope, $context->branch_point ?: (int) $stmt->getAttribute('startFilePos') ); $if_context = $if_conditional_scope->if_context; $post_if_context = $if_conditional_scope->post_if_context; $cond_referenced_var_ids = $if_conditional_scope->cond_referenced_var_ids; $assigned_in_conditional_var_ids = $if_conditional_scope->assigned_in_conditional_var_ids; } catch (\Psalm\Exception\ScopeAnalysisException $e) { return false; } $mixed_var_ids = []; foreach ($if_context->vars_in_scope as $var_id => $type) { if ($type->isMixed() && isset($context->vars_in_scope[$var_id])) { $mixed_var_ids[] = $var_id; } } $cond_object_id = \spl_object_id($stmt->cond); $if_clauses = FormulaGenerator::getFormula( $cond_object_id, $cond_object_id, $stmt->cond, $context->self, $statements_analyzer, $codebase ); if (count($if_clauses) > 200) { $if_clauses = []; } $if_clauses = array_values( array_map( /** * @return Clause */ function (Clause $c) use ($mixed_var_ids, $cond_object_id): Clause { $keys = array_keys($c->possibilities); $mixed_var_ids = \array_diff($mixed_var_ids, $keys); foreach ($keys as $key) { foreach ($mixed_var_ids as $mixed_var_id) { if (preg_match('/^' . preg_quote($mixed_var_id, '/') . '(\[|-)/', $key)) { return new Clause([], $cond_object_id, $cond_object_id, true); } } } return $c; }, $if_clauses ) ); $entry_clauses = $context->clauses; // this will see whether any of the clauses in set A conflict with the clauses in set B AlgebraAnalyzer::checkForParadox( $context->clauses, $if_clauses, $statements_analyzer, $stmt->cond, $assigned_in_conditional_var_ids ); // if we have assignments in the if, we may have duplicate clauses if ($assigned_in_conditional_var_ids) { $if_clauses = Algebra::simplifyCNF($if_clauses); } $if_context_clauses = array_merge($entry_clauses, $if_clauses); $if_context->clauses = Algebra::simplifyCNF($if_context_clauses); if ($if_context->reconciled_expression_clauses) { $reconciled_expression_clauses = $if_context->reconciled_expression_clauses; $if_context->clauses = array_values( array_filter( $if_context->clauses, function ($c) use ($reconciled_expression_clauses): bool { return !in_array($c->hash, $reconciled_expression_clauses); } ) ); if (count($if_context->clauses) === 1 && $if_context->clauses[0]->wedge && !$if_context->clauses[0]->possibilities ) { $if_context->clauses = []; $if_context->reconciled_expression_clauses = []; } } // define this before we alter local claues after reconciliation $if_scope->reasonable_clauses = $if_context->clauses; try { $if_scope->negated_clauses = Algebra::negateFormula($if_clauses); } catch (\Psalm\Exception\ComplicatedExpressionException $e) { try { $if_scope->negated_clauses = FormulaGenerator::getFormula( $cond_object_id, $cond_object_id, new VirtualBooleanNot($stmt->cond), $context->self, $statements_analyzer, $codebase, false ); } catch (\Psalm\Exception\ComplicatedExpressionException $e) { $if_scope->negated_clauses = []; } } $if_scope->negated_types = Algebra::getTruthsFromFormula( Algebra::simplifyCNF( array_merge($context->clauses, $if_scope->negated_clauses) ) ); $active_if_types = []; $reconcilable_if_types = Algebra::getTruthsFromFormula( $if_context->clauses, \spl_object_id($stmt->cond), $cond_referenced_var_ids, $active_if_types ); if (array_filter( $context->clauses, function ($clause): bool { return !!$clause->possibilities; } )) { $omit_keys = array_reduce( $context->clauses, /** * @param array $carry * @return array */ function (array $carry, Clause $clause): array { return array_merge($carry, array_keys($clause->possibilities)); }, [] ); $omit_keys = array_combine($omit_keys, $omit_keys); $omit_keys = array_diff_key($omit_keys, Algebra::getTruthsFromFormula($context->clauses)); $cond_referenced_var_ids = array_diff_key( $cond_referenced_var_ids, $omit_keys ); } // 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 = Reconciler::reconcileKeyedTypes( $reconcilable_if_types, $active_if_types, $if_context->vars_in_scope, $changed_var_ids, $cond_referenced_var_ids, $statements_analyzer, $statements_analyzer->getTemplateTypeMap() ?: [], $if_context->inside_loop, $context->check_variables ? new CodeLocation( $statements_analyzer->getSource(), $stmt->cond instanceof PhpParser\Node\Expr\BooleanNot ? $stmt->cond->expr : $stmt->cond, $context->include_location ) : null ); $if_context->vars_in_scope = $if_vars_in_scope_reconciled; foreach ($reconcilable_if_types as $var_id => $_) { $if_context->vars_possibly_in_scope[$var_id] = true; } if ($changed_var_ids) { $if_context->clauses = Context::removeReconciledClauses($if_context->clauses, $changed_var_ids)[0]; foreach ($changed_var_ids as $changed_var_id => $_) { foreach ($if_context->vars_in_scope as $var_id => $_) { if (preg_match('/' . preg_quote($changed_var_id, '/') . '[\]\[\-]/', $var_id) && !\array_key_exists($var_id, $changed_var_ids) && !\array_key_exists($var_id, $cond_referenced_var_ids) ) { unset($if_context->vars_in_scope[$var_id]); } } } } $if_scope->if_cond_changed_var_ids = $changed_var_ids; } $if_context->reconciled_expression_clauses = []; $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 $post_if_context; $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, [], $statements_analyzer, $statements_analyzer->getTemplateTypeMap() ?: [], $context->inside_loop, $context->check_variables ? new CodeLocation( $statements_analyzer->getSource(), $stmt->cond instanceof PhpParser\Node\Expr\BooleanNot ? $stmt->cond->expr : $stmt->cond, $context->include_location ) : null ); $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 = array_intersect_key( $temp_else_context->getRedefinedVars($context->vars_in_scope, true), $changed_var_ids ); // check the if if (IfElse\IfAnalyzer::analyze( $statements_analyzer, $stmt, $if_scope, $if_conditional_scope, $if_context, $old_if_context, $context, $pre_assignment_else_redefined_vars ) === false) { return false; } // this has to go on a separate line because the phar compactor messes with precedence $scope_to_clone = $if_scope->post_leaving_if_context ?? $post_if_context; $else_context = clone $scope_to_clone; // check the elseifs foreach ($stmt->elseifs as $elseif) { if (IfElse\ElseIfAnalyzer::analyze( $statements_analyzer, $elseif, $if_scope, $else_context, $context, $codebase, $else_context->branch_point ?: (int) $stmt->getAttribute('startFilePos') ) === false) { return false; } } if ($stmt->else) { if ($codebase->alter_code) { $else_context->branch_point = $else_context->branch_point ?: (int) $stmt->getAttribute('startFilePos'); } } if (IfElse\ElseAnalyzer::analyze( $statements_analyzer, $stmt->else, $if_scope, $else_context, $context ) === false) { return false; } if ($context->loop_scope) { $context->loop_scope->final_actions = array_unique( array_merge( $context->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->possibly_assigned_var_ids = array_merge( $context->possibly_assigned_var_ids, $if_scope->possibly_assigned_var_ids ?: [] ); // vars can only be defined/redefined if there was an else (defined in every block) $context->assigned_var_ids = array_merge( $context->assigned_var_ids, $if_scope->assigned_var_ids ?: [] ); if ($if_scope->new_vars) { foreach ($if_scope->new_vars as $var_id => $type) { if (isset($context->vars_possibly_in_scope[$var_id]) && $statements_analyzer->data_flow_graph ) { $type->parent_nodes += $statements_analyzer->getParentNodesForPossiblyUndefinedVariable($var_id); } $context->vars_in_scope[$var_id] = $type; } } if ($if_scope->redefined_vars) { foreach ($if_scope->redefined_vars as $var_id => $type) { $context->vars_in_scope[$var_id] = $type; $if_scope->updated_vars[$var_id] = true; if ($if_scope->reasonable_clauses) { $if_scope->reasonable_clauses = Context::filterClauses( $var_id, $if_scope->reasonable_clauses, isset($context->vars_in_scope[$var_id]) ? $context->vars_in_scope[$var_id] : null, $statements_analyzer ); } } } if ($if_scope->possible_param_types) { foreach ($if_scope->possible_param_types as $var => $type) { $context->possible_param_types[$var] = $type; } } if ($if_scope->reasonable_clauses && (count($if_scope->reasonable_clauses) > 1 || !$if_scope->reasonable_clauses[0]->wedge) ) { $context->clauses = Algebra::simplifyCNF( array_merge( $if_scope->reasonable_clauses, $context->clauses ) ); } if ($if_scope->possibly_redefined_vars) { foreach ($if_scope->possibly_redefined_vars as $var_id => $type) { if (isset($context->vars_in_scope[$var_id])) { if (!$type->failed_reconciliation && !isset($if_scope->updated_vars[$var_id]) ) { $combined_type = Type::combineUnionTypes( $context->vars_in_scope[$var_id], $type, $codebase ); if (!$combined_type->equals($context->vars_in_scope[$var_id])) { $context->removeDescendents($var_id, $combined_type); } $context->vars_in_scope[$var_id] = $combined_type; } else { $context->vars_in_scope[$var_id]->parent_nodes += $type->parent_nodes; } } } } $context->possibly_assigned_var_ids += $if_scope->possibly_assigned_var_ids; if (!in_array(ScopeAnalyzer::ACTION_NONE, $if_scope->final_actions, true)) { $context->has_returned = true; } return null; } }