getCodebase(); foreach ($stmt->exprs as $i => $expr) { $context->inside_call = true; ExpressionAnalyzer::analyze($statements_analyzer, $expr, $context); $context->inside_call = false; $expr_type = $statements_analyzer->node_data->getType($expr); if ($statements_analyzer->control_flow_graph && $expr_type) { $expr_type = CastAnalyzer::castStringAttempt( $statements_analyzer, $context, $expr_type, $expr, false ); } if ($statements_analyzer->control_flow_graph) { $call_location = new CodeLocation($statements_analyzer->getSource(), $stmt); $echo_param_sink = TaintSink::getForMethodArgument( 'echo', 'echo', (int) $i, null, $call_location ); $echo_param_sink->taints = [ Type\TaintKind::INPUT_HTML, Type\TaintKind::USER_SECRET, Type\TaintKind::SYSTEM_SECRET ]; $statements_analyzer->control_flow_graph->addSink($echo_param_sink); } if ($expr_type) { if (ArgumentAnalyzer::verifyType( $statements_analyzer, $expr_type, Type::getString(), null, 'echo', (int)$i, new CodeLocation($statements_analyzer->getSource(), $expr), $expr, $context, $echo_param, false, null, true, true, new CodeLocation($statements_analyzer, $stmt) ) === false) { return false; } } } if ($codebase->config->forbid_echo) { if (IssueBuffer::accepts( new ForbiddenEcho( 'Use of echo', new CodeLocation($statements_analyzer, $stmt) ), $statements_analyzer->getSource()->getSuppressedIssues() )) { return false; } } elseif (isset($codebase->config->forbidden_functions['echo'])) { if (IssueBuffer::accepts( new ForbiddenCode( 'Use of echo', new CodeLocation($statements_analyzer, $stmt) ), $statements_analyzer->getSource()->getSuppressedIssues() )) { // continue } } if (!$context->collect_initializations && !$context->collect_mutations) { if ($context->mutation_free || $context->external_mutation_free) { if (IssueBuffer::accepts( new ImpureFunctionCall( 'Cannot call echo from a mutation-free context', new CodeLocation($statements_analyzer, $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($statements_analyzer->getSource() instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer && $statements_analyzer->getSource()->track_mutations ) { $statements_analyzer->getSource()->inferred_has_mutation = true; $statements_analyzer->getSource()->inferred_impure = true; } } return true; } }