*/ private $all_vars = []; /** * @var array */ public static $stub_constants = []; /** * @var array */ private $function_checkers = []; /** * @var Type\Union|null */ public $local_return_type; /** * @param StatementsSource $source */ public function __construct(StatementsSource $source) { $this->source = $source; $this->file_checker = $source->getFileChecker(); } /** * Checks an array of statements for validity * * @param array $stmts * @param Context $context * @param Context|null $global_context * * @return null|false */ public function analyze( array $stmts, Context $context, LoopScope $loop_scope = null, Context $global_context = null ) { $has_returned = false; // hoist functions to the top foreach ($stmts as $stmt) { if ($stmt instanceof PhpParser\Node\Stmt\Function_) { $function_checker = new FunctionChecker($stmt, $this->source); $this->function_checkers[strtolower($stmt->name)] = $function_checker; } } $project_checker = $this->getFileChecker()->project_checker; $original_context = null; if ($loop_scope) { $original_context = clone $context; } $plugins = Config::getInstance()->getPlugins(); foreach ($stmts as $stmt) { if ($has_returned && !($stmt instanceof PhpParser\Node\Stmt\Nop) && !($stmt instanceof PhpParser\Node\Stmt\InlineHTML) ) { if ($context->collect_references) { if (IssueBuffer::accepts( new UnevaluatedCode( 'Expressions after return/throw/continue', new CodeLocation($this->source, $stmt) ), $this->source->getSuppressedIssues() )) { return false; } } break; } /* if (isset($context->vars_in_scope['$tag']) && !$stmt instanceof PhpParser\Node\Stmt\Nop) { var_dump($stmt->getLine() . ' ' . $context->vars_in_scope['$tag']); } */ $new_issues = null; if ($docblock = $stmt->getDocComment()) { $comments = CommentChecker::parseDocComment((string)$docblock); if (isset($comments['specials']['psalm-suppress'])) { $suppressed = array_filter( array_map( /** * @param string $line * * @return string */ function ($line) { return explode(' ', trim($line))[0]; }, $comments['specials']['psalm-suppress'] ) ); if ($suppressed) { $new_issues = array_diff($suppressed, $this->source->getSuppressedIssues()); /** @psalm-suppress MixedTypeCoercion */ $this->addSuppressedIssues($new_issues); } } } if ($stmt instanceof PhpParser\Node\Stmt\If_) { IfChecker::analyze($this, $stmt, $context, $loop_scope); } elseif ($stmt instanceof PhpParser\Node\Stmt\TryCatch) { TryChecker::analyze($this, $stmt, $context, $loop_scope); } elseif ($stmt instanceof PhpParser\Node\Stmt\For_) { ForChecker::analyze($this, $stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\Foreach_) { ForeachChecker::analyze($this, $stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\While_) { WhileChecker::analyze($this, $stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\Do_) { $this->analyzeDo($stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\Const_) { $this->analyzeConstAssignment($stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\Unset_) { $suppressed_issues = $this->getSuppressedIssues(); if (!in_array('PossiblyUndefinedGlobalVariable', $suppressed_issues, true)) { $this->addSuppressedIssues(['PossiblyUndefinedGlobalVariable']); } if (!in_array('PossiblyUndefinedVariable', $suppressed_issues, true)) { $this->addSuppressedIssues(['PossiblyUndefinedVariable']); } foreach ($stmt->vars as $var) { ExpressionChecker::analyze($this, $var, $context); $var_id = ExpressionChecker::getArrayVarId( $var, $this->getFQCLN(), $this ); if ($var_id) { $context->remove($var_id); } } if (!in_array('PossiblyUndefinedGlobalVariable', $suppressed_issues, true)) { $this->removeSuppressedIssues(['PossiblyUndefinedGlobalVariable']); } if (!in_array('PossiblyUndefinedVariable', $suppressed_issues, true)) { $this->removeSuppressedIssues(['PossiblyUndefinedVariable']); } } elseif ($stmt instanceof PhpParser\Node\Stmt\Return_) { $has_returned = true; $this->analyzeReturn($project_checker, $stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\Throw_) { $has_returned = true; $this->analyzeThrow($project_checker, $stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\Switch_) { SwitchChecker::analyze($this, $stmt, $context, $loop_scope); } elseif ($stmt instanceof PhpParser\Node\Stmt\Break_) { if ($loop_scope && $original_context) { $loop_scope->final_actions[] = ScopeChecker::ACTION_BREAK; $redefined_vars = $context->getRedefinedVars($loop_scope->loop_parent_context->vars_in_scope); if ($loop_scope->possibly_redefined_loop_parent_vars === null) { $loop_scope->possibly_redefined_loop_parent_vars = $redefined_vars; } else { foreach ($redefined_vars as $var => $type) { if ($type->isMixed()) { $loop_scope->possibly_redefined_loop_parent_vars[$var] = $type; } elseif (isset($loop_scope->possibly_redefined_loop_parent_vars[$var])) { $loop_scope->possibly_redefined_loop_parent_vars[$var] = Type::combineUnionTypes( $type, $loop_scope->possibly_redefined_loop_parent_vars[$var] ); } else { $loop_scope->possibly_redefined_loop_parent_vars[$var] = $type; } } } } $has_returned = true; } elseif ($stmt instanceof PhpParser\Node\Stmt\Continue_) { if ($loop_scope === null) { if (IssueBuffer::accepts( new ContinueOutsideLoop( 'Continue call outside loop context', new CodeLocation($this->source, $stmt) ), $this->source->getSuppressedIssues() )) { return false; } } elseif ($original_context) { $loop_scope->final_actions[] = ScopeChecker::ACTION_CONTINUE; $redefined_vars = $context->getRedefinedVars($original_context->vars_in_scope); if ($loop_scope->redefined_loop_vars === null) { $loop_scope->redefined_loop_vars = $redefined_vars; } else { foreach ($loop_scope->redefined_loop_vars as $redefined_var => $type) { if (!isset($redefined_vars[$redefined_var])) { unset($loop_scope->redefined_loop_vars[$redefined_var]); } else { $loop_scope->redefined_loop_vars[$redefined_var] = Type::combineUnionTypes( $redefined_vars[$redefined_var], $type ); } } } if ($loop_scope->possibly_redefined_loop_vars === null) { $loop_scope->possibly_redefined_loop_vars = $redefined_vars; } else { foreach ($redefined_vars as $var => $type) { if ($type->isMixed()) { $loop_scope->possibly_redefined_loop_vars[$var] = $type; } elseif (isset($loop_scope->possibly_redefined_loop_vars[$var])) { $loop_scope->possibly_redefined_loop_vars[$var] = Type::combineUnionTypes( $type, $loop_scope->possibly_redefined_loop_vars[$var] ); } else { $loop_scope->possibly_redefined_loop_vars[$var] = $type; } } } } $has_returned = true; } elseif ($stmt instanceof PhpParser\Node\Stmt\Static_) { $this->analyzeStatic($stmt, $context); } elseif ($stmt instanceof PhpParser\Node\Stmt\Echo_) { foreach ($stmt->exprs as $i => $expr) { ExpressionChecker::analyze($this, $expr, $context); if (isset($expr->inferredType)) { if (CallChecker::checkFunctionArgumentType( $this, $expr->inferredType, Type::getString(), 'echo', (int)$i, new CodeLocation($this->getSource(), $expr), $expr, $context ) === false) { return false; } } } } elseif ($stmt instanceof PhpParser\Node\Stmt\Function_) { if (!$project_checker->register_global_functions) { $function_id = strtolower($stmt->name); $function_context = new Context($context->self); $function_context->collect_references = $project_checker->collect_references; $this->function_checkers[$function_id]->analyze($function_context, $context); $config = Config::getInstance(); if ($config->reportIssueInFile('InvalidReturnType', $this->getFilePath())) { /** @var string */ $method_id = $this->function_checkers[$function_id]->getMethodId(); $function_storage = FunctionChecker::getStorage( $this, $method_id ); $return_type = $function_storage->return_type; $return_type_location = $function_storage->return_type_location; $this->function_checkers[$function_id]->verifyReturnType( $project_checker, $return_type, $this->getFQCLN(), $return_type_location ); } } } elseif ($stmt instanceof PhpParser\Node\Expr) { if (ExpressionChecker::analyze($this, $stmt, $context) === false) { return false; } } elseif ($stmt instanceof PhpParser\Node\Stmt\InlineHTML) { // do nothing } elseif ($stmt instanceof PhpParser\Node\Stmt\Global_) { if (!$context->collect_initializations && !$global_context) { if (IssueBuffer::accepts( new InvalidGlobal( 'Cannot use global scope here', new CodeLocation($this->source, $stmt) ), $this->source->getSuppressedIssues() )) { // fall through } } foreach ($stmt->vars as $var) { if ($var instanceof PhpParser\Node\Expr\Variable) { if (is_string($var->name)) { $var_id = '$' . $var->name; $context->vars_in_scope[$var_id] = $global_context && $global_context->hasVariable($var_id) ? clone $global_context->vars_in_scope[$var_id] : Type::getMixed(); $context->vars_possibly_in_scope[$var_id] = true; } else { ExpressionChecker::analyze($this, $var, $context); } } } } elseif ($stmt instanceof PhpParser\Node\Stmt\Property) { foreach ($stmt->props as $prop) { if ($prop->default) { ExpressionChecker::analyze($this, $prop->default, $context); if (isset($prop->default->inferredType)) { if (!$stmt->isStatic()) { if (AssignmentChecker::analyzePropertyAssignment( $this, $prop, $prop->name, $prop->default, $prop->default->inferredType, $context ) === false) { // fall through } } } } } } elseif ($stmt instanceof PhpParser\Node\Stmt\ClassConst) { $const_visibility = \ReflectionProperty::IS_PUBLIC; if ($stmt->isProtected()) { $const_visibility = \ReflectionProperty::IS_PROTECTED; } if ($stmt->isPrivate()) { $const_visibility = \ReflectionProperty::IS_PRIVATE; } foreach ($stmt->consts as $const) { ExpressionChecker::analyze($this, $const->value, $context); if (isset($const->value->inferredType) && !$const->value->inferredType->isMixed()) { ClassLikeChecker::setConstantType( $project_checker, (string)$this->getFQCLN(), $const->name, $const->value->inferredType, $const_visibility ); } } } elseif ($stmt instanceof PhpParser\Node\Stmt\Class_) { try { $class_checker = (new ClassChecker($stmt, $this->source, $stmt->name)); $class_checker->analyze(null, $global_context); } catch (\InvalidArgumentException $e) { // disregard this exception, we'll likely see it elsewhere in the form // of an issue } } elseif ($stmt instanceof PhpParser\Node\Stmt\Nop) { if ((string)$stmt->getDocComment()) { try { $var_comment = CommentChecker::getTypeFromComment( (string)$stmt->getDocComment(), $this->getSource(), $this->getSource()->getAliases() ); } catch (DocblockParseException $e) { if (IssueBuffer::accepts( new InvalidDocblock( (string)$e->getMessage(), new CodeLocation($this->getSource(), $stmt, null, true) ) )) { // fall through } } if ($var_comment && $var_comment->var_id) { $comment_type = ExpressionChecker::fleshOutType( $project_checker, $var_comment->type, $context->self ); $context->vars_in_scope[$var_comment->var_id] = $comment_type; } } } elseif ($stmt instanceof PhpParser\Node\Stmt\Goto_) { // do nothing } elseif ($stmt instanceof PhpParser\Node\Stmt\Label) { // do nothing } elseif ($stmt instanceof PhpParser\Node\Stmt\Declare_) { // do nothing } else { if (IssueBuffer::accepts( new UnrecognizedStatement( 'Psalm does not understand ' . get_class($stmt), new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } } if ($loop_scope && $loop_scope->final_actions && !in_array(ScopeChecker::ACTION_NONE, $loop_scope->final_actions, true) ) { //$has_returned = true; } if ($plugins) { $file_manipulations = []; $code_location = new CodeLocation($this->source, $stmt); foreach ($plugins as $plugin) { if ($plugin->afterStatementCheck( $this, $stmt, $context, $code_location, $this->getSuppressedIssues(), $file_manipulations ) === false) { return false; } } if ($file_manipulations) { FileManipulationBuffer::add($this->getFilePath(), $file_manipulations); } } if ($new_issues) { /** @psalm-suppress MixedTypeCoercion */ $this->removeSuppressedIssues($new_issues); } } return null; } /** * @param PhpParser\Node\Stmt\Static_ $stmt * @param Context $context * * @return false|null */ private function analyzeStatic(PhpParser\Node\Stmt\Static_ $stmt, Context $context) { foreach ($stmt->vars as $var) { if ($var->default) { if (ExpressionChecker::analyze($this, $var->default, $context) === false) { return false; } } if ($context->check_variables) { $context->vars_in_scope['$' . $var->name] = Type::getMixed(); $context->vars_possibly_in_scope['$' . $var->name] = true; $this->registerVariable('$' . $var->name, new CodeLocation($this, $stmt)); } } return null; } /** * @param PhpParser\Node\Expr $stmt * @param array $existing_class_constants * * @return Type\Union|null */ public static function getSimpleType( PhpParser\Node\Expr $stmt, StatementsSource $statements_source = null, array $existing_class_constants = [] ) { if ($stmt instanceof PhpParser\Node\Expr\BinaryOp) { if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Concat) { return Type::getString(); } if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd || $stmt instanceof PhpParser\Node\Expr\BinaryOp\BooleanOr || $stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalAnd || $stmt instanceof PhpParser\Node\Expr\BinaryOp\LogicalOr || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Equal || $stmt instanceof PhpParser\Node\Expr\BinaryOp\NotEqual || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Identical || $stmt instanceof PhpParser\Node\Expr\BinaryOp\NotIdentical || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Greater || $stmt instanceof PhpParser\Node\Expr\BinaryOp\GreaterOrEqual || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Smaller || $stmt instanceof PhpParser\Node\Expr\BinaryOp\SmallerOrEqual ) { return Type::getBool(); } if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Coalesce) { return null; } if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Spaceship) { return Type::getInt(); } $stmt->left->inferredType = self::getSimpleType( $stmt->left, $statements_source, $existing_class_constants ); $stmt->right->inferredType = self::getSimpleType( $stmt->right, $statements_source, $existing_class_constants ); if (!$stmt->left->inferredType || !$stmt->right->inferredType) { return null; } if (!$statements_source) { return null; } if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Plus || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Minus || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Mod || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Mul || $stmt instanceof PhpParser\Node\Expr\BinaryOp\Pow ) { ExpressionChecker::analyzeNonDivArithmenticOp( $statements_source, $stmt->left, $stmt->right, $stmt, $result_type ); if ($result_type) { return $result_type; } return null; } if ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Div && ($stmt->left->inferredType->hasInt() || $stmt->left->inferredType->hasFloat()) && ($stmt->right->inferredType->hasInt() || $stmt->right->inferredType->hasFloat()) ) { return Type::combineUnionTypes(Type::getFloat(), Type::getInt()); } } if ($stmt instanceof PhpParser\Node\Expr\ConstFetch) { if (strtolower($stmt->name->parts[0]) === 'false') { return Type::getFalse(); } elseif (strtolower($stmt->name->parts[0]) === 'true') { return Type::getBool(); } elseif (strtolower($stmt->name->parts[0]) === 'null') { return Type::getNull(); } return null; } if ($stmt instanceof PhpParser\Node\Expr\ClassConstFetch) { if ($stmt->class instanceof PhpParser\Node\Name && $stmt->class->parts !== ['static'] && is_string($stmt->name) && isset($existing_class_constants[$stmt->name]) ) { return $existing_class_constants[$stmt->name]; } return null; } if ($stmt instanceof PhpParser\Node\Scalar\String_) { return Type::getString(); } if ($stmt instanceof PhpParser\Node\Scalar\LNumber) { return Type::getInt(); } if ($stmt instanceof PhpParser\Node\Scalar\DNumber) { return Type::getFloat(); } if ($stmt instanceof PhpParser\Node\Expr\Array_) { if (count($stmt->items) === 0) { return Type::getEmptyArray(); } $item_key_type = null; $item_value_type = null; $property_types = []; $can_create_objectlike = true; foreach ($stmt->items as $int_offset => $item) { if ($item === null) { continue; } if ($item->key) { $single_item_key_type = self::getSimpleType( $item->key, $statements_source, $existing_class_constants ); if ($single_item_key_type) { if ($item_key_type) { $item_key_type = Type::combineUnionTypes($single_item_key_type, $item_key_type); } else { $item_key_type = $single_item_key_type; } } } else { $item_key_type = Type::getInt(); } if ($item_value_type && $item_value_type->isMixed() && !$can_create_objectlike) { continue; } $single_item_value_type = self::getSimpleType( $item->value, $statements_source, $existing_class_constants ); if ($single_item_value_type) { if ($item->key instanceof PhpParser\Node\Scalar\String_ || $item->key instanceof PhpParser\Node\Scalar\LNumber || !$item->key ) { $property_types[$item->key ? $item->key->value : $int_offset] = $single_item_value_type; } else { $can_create_objectlike = false; } if ($item_value_type) { $item_value_type = Type::combineUnionTypes($single_item_value_type, $item_value_type); } else { $item_value_type = $single_item_value_type; } } else { $item_value_type = Type::getMixed(); if ($item->key instanceof PhpParser\Node\Scalar\String_ || $item->key instanceof PhpParser\Node\Scalar\LNumber || !$item->key ) { $property_types[$item->key ? $item->key->value : $int_offset] = $item_value_type; } else { $can_create_objectlike = false; } } } // if this array looks like an object-like array, let's return that instead if ($item_value_type && $item_key_type && ($item_key_type->hasString() || $item_key_type->hasInt()) && $can_create_objectlike ) { return new Type\Union([new Type\Atomic\ObjectLike($property_types)]); } return new Type\Union([ new Type\Atomic\TArray([ $item_key_type ?: Type::getMixed(), $item_value_type ?: Type::getMixed(), ]), ]); } if ($stmt instanceof PhpParser\Node\Expr\Cast\Int_) { return Type::getInt(); } if ($stmt instanceof PhpParser\Node\Expr\Cast\Double) { return Type::getFloat(); } if ($stmt instanceof PhpParser\Node\Expr\Cast\Bool_) { return Type::getBool(); } if ($stmt instanceof PhpParser\Node\Expr\Cast\String_) { return Type::getString(); } if ($stmt instanceof PhpParser\Node\Expr\Cast\Object_) { return Type::getObject(); } if ($stmt instanceof PhpParser\Node\Expr\Cast\Array_) { return Type::getArray(); } if ($stmt instanceof PhpParser\Node\Expr\UnaryMinus || $stmt instanceof PhpParser\Node\Expr\UnaryPlus) { return self::getSimpleType($stmt->expr, $statements_source, $existing_class_constants); } return null; } /** * @param PhpParser\Node\Stmt\Do_ $stmt * @param Context $context * * @return false|null */ private function analyzeDo(PhpParser\Node\Stmt\Do_ $stmt, Context $context) { $do_context = clone $context; $loop_scope = new LoopScope($do_context, $context); $loop_scope->protected_var_ids = $context->protected_var_ids; LoopChecker::analyze($this, $stmt->stmts, [], [], $loop_scope, $inner_loop_context); foreach ($context->vars_in_scope as $var => $type) { if ($type->isMixed()) { continue; } if ($do_context->hasVariable($var)) { if ($do_context->vars_in_scope[$var]->isMixed()) { $context->vars_in_scope[$var] = $do_context->vars_in_scope[$var]; } if ($do_context->vars_in_scope[$var]->getId() !== $type->getId()) { $context->vars_in_scope[$var] = Type::combineUnionTypes($do_context->vars_in_scope[$var], $type); } } } foreach ($do_context->vars_in_scope as $var_id => $type) { if (!isset($context->vars_in_scope[$var_id])) { $context->vars_in_scope[$var_id] = $type; } } // because it's a do {} while, inner loop vars belong to the main context if ($inner_loop_context) { foreach ($inner_loop_context->vars_in_scope as $var_id => $type) { if (!isset($context->vars_in_scope[$var_id])) { $context->vars_in_scope[$var_id] = $type; } } } $context->vars_possibly_in_scope = array_merge( $context->vars_possibly_in_scope, $do_context->vars_possibly_in_scope ); $context->referenced_var_ids = array_merge( $context->referenced_var_ids, $do_context->referenced_var_ids ); return ExpressionChecker::analyze($this, $stmt->cond, $context); } /** * @param PhpParser\Node\Stmt\Const_ $stmt * @param Context $context * * @return void */ private function analyzeConstAssignment(PhpParser\Node\Stmt\Const_ $stmt, Context $context) { foreach ($stmt->consts as $const) { ExpressionChecker::analyze($this, $const->value, $context); $this->setConstType( $const->name, isset($const->value->inferredType) ? $const->value->inferredType : Type::getMixed(), $context ); } } /** * @param string $const_name * @param bool $is_fully_qualified * @param Context $context * * @return Type\Union|null */ public function getConstType( StatementsChecker $statements_checker, $const_name, $is_fully_qualified, Context $context ) { $fq_const_name = null; $aliased_constants = $this->getAliases()->constants; if (isset($aliased_constants[$const_name])) { $fq_const_name = $aliased_constants[$const_name]; } elseif ($is_fully_qualified) { $fq_const_name = $const_name; } elseif (strpos($const_name, '\\')) { $fq_const_name = ClassLikeChecker::getFQCLNFromString($const_name, $this->getAliases()); } if ($fq_const_name) { $const_name_parts = explode('\\', $fq_const_name); /** @var string */ $const_name = array_pop($const_name_parts); $namespace_name = implode('\\', $const_name_parts); $namespace_constants = NamespaceChecker::getConstantsForNamespace( $namespace_name, \ReflectionProperty::IS_PUBLIC ); if (isset($namespace_constants[$const_name])) { return $namespace_constants[$const_name]; } } if ($context->hasVariable($const_name)) { return $context->vars_in_scope[$const_name]; } $file_path = $statements_checker->getFilePath(); $file_storage_provider = $statements_checker->getFileChecker()->project_checker->file_storage_provider; $file_storage = $file_storage_provider->get($file_path); if (isset($file_storage->declaring_constants[$const_name])) { $constant_file_path = $file_storage->declaring_constants[$const_name]; return $file_storage_provider->get($constant_file_path)->constants[$const_name]; } $predefined_constants = Config::getInstance()->getPredefinedConstants(); if (isset($predefined_constants[$fq_const_name ?: $const_name])) { return ClassLikeChecker::getTypeFromValue($predefined_constants[$fq_const_name ?: $const_name]); } if (isset(self::$stub_constants[$fq_const_name ?: $const_name])) { return self::$stub_constants[$fq_const_name ?: $const_name]; } return null; } /** * @param string $const_name * @param Type\Union $const_type * @param Context $context * * @return void */ public function setConstType($const_name, Type\Union $const_type, Context $context) { $context->vars_in_scope[$const_name] = $const_type; $context->constants[$const_name] = $const_type; if ($this->source instanceof NamespaceChecker) { $this->source->setConstType($const_name, $const_type); } } /** * @param PhpParser\Node\Stmt\Return_ $stmt * @param Context $context * * @return false|null */ private function analyzeReturn( ProjectChecker $project_checker, PhpParser\Node\Stmt\Return_ $stmt, Context $context ) { $doc_comment_text = (string)$stmt->getDocComment(); $var_comment = null; if ($doc_comment_text) { try { $var_comment = CommentChecker::getTypeFromComment( $doc_comment_text, $this->source, $this->source->getAliases() ); } catch (DocblockParseException $e) { if (IssueBuffer::accepts( new InvalidDocblock( (string)$e->getMessage(), new CodeLocation($this->source, $stmt) ) )) { // fall through } } if ($var_comment && $var_comment->var_id) { $comment_type = ExpressionChecker::fleshOutType( $project_checker, $var_comment->type, $context->self ); $context->vars_in_scope[$var_comment->var_id] = $comment_type; } } if ($stmt->expr) { if (ExpressionChecker::analyze($this, $stmt->expr, $context) === false) { return false; } if ($var_comment && !$var_comment->var_id) { $stmt->inferredType = $var_comment->type; } elseif (isset($stmt->expr->inferredType)) { $stmt->inferredType = $stmt->expr->inferredType; if ($stmt->inferredType->isVoid()) { $stmt->inferredType = Type::getNull(); } } else { $stmt->inferredType = Type::getMixed(); } } else { $stmt->inferredType = Type::getVoid(); } if ($this->source instanceof FunctionLikeChecker && !($this->source->getSource() instanceof TraitChecker) ) { $this->source->addReturnTypes($stmt->expr ? (string) $stmt->inferredType : '', $context); if ($stmt->expr) { $storage = $this->source->getFunctionLikeStorage($this); $cased_method_id = $this->source->getCorrectlyCasedMethodId(); if ($storage->return_type && !$storage->return_type->isMixed() ) { $inferred_type = ExpressionChecker::fleshOutType( $project_checker, $stmt->inferredType, $this->source->getFQCLN(), '' ); if (!$this->local_return_type) { $this->local_return_type = ExpressionChecker::fleshOutType( $project_checker, $storage->return_type, $this->source->getFQCLN(), '' ); } if ($this->local_return_type->isGenerator()) { return null; } if ($stmt->inferredType->isMixed()) { if ($this->local_return_type->isVoid()) { if (IssueBuffer::accepts( new InvalidReturnStatement( 'No return values are expected for ' . $cased_method_id, new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } } if (IssueBuffer::accepts( new MixedReturnStatement( 'Could not infer a return type', new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } return null; } if ($this->local_return_type->isVoid()) { if (IssueBuffer::accepts( new InvalidReturnStatement( 'No return values are expected for ' . $cased_method_id, new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } return null; } if (!$stmt->inferredType->ignore_nullable_issues && $inferred_type->isNullable() && !$this->local_return_type->isNullable() ) { if (IssueBuffer::accepts( new NullableReturnStatement( 'The declared return type \'' . $this->local_return_type . '\' for ' . $cased_method_id . ' is not nullable, but \'' . $inferred_type . '\' contains null', new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } } if ($inferred_type->isFalsable() && !$this->local_return_type->isFalsable() && !$this->local_return_type->hasBool() ) { if (IssueBuffer::accepts( new FalsableReturnStatement( 'The declared return type \'' . $this->local_return_type . '\' for ' . $cased_method_id . ' does not allow false, but \'' . $inferred_type . '\' contains false', new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } } if (!TypeChecker::isContainedBy( $this->source->getFileChecker()->project_checker, $inferred_type, $this->local_return_type, true, true, $has_scalar_match, $type_coerced, $type_coerced_from_mixed ) ) { // is the declared return type more specific than the inferred one? if ($type_coerced) { if (IssueBuffer::accepts( new LessSpecificReturnStatement( 'The type \'' . $stmt->inferredType . '\' is more general than the declared ' . 'return type \'' . $this->local_return_type . '\' for ' . $cased_method_id, new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } } else { if (IssueBuffer::accepts( new InvalidReturnStatement( 'The type \'' . $stmt->inferredType . '\' does not match the declared return ' . 'type \'' . $this->local_return_type . '\' for ' . $cased_method_id, new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } } } } } } return null; } /** * @param PhpParser\Node\Stmt\Throw_ $stmt * @param Context $context * * @return false|null */ private function analyzeThrow(ProjectChecker $project_checker, PhpParser\Node\Stmt\Throw_ $stmt, Context $context) { if (ExpressionChecker::analyze($this, $stmt->expr, $context) === false) { return false; } if ($context->check_classes && isset($stmt->expr->inferredType) && !$stmt->expr->inferredType->isMixed()) { $throw_type = $stmt->expr->inferredType; $exception_type = new Union([new TNamedObject('Exception'), new TNamedObject('Throwable')]); if (!TypeChecker::isContainedBy($project_checker, $throw_type, $exception_type)) { if (IssueBuffer::accepts( new InvalidThrow( 'Cannot throw ' . $throw_type . ' as it does not extend Exception or implement Throwable', new CodeLocation($this->source, $stmt) ), $this->getSuppressedIssues() )) { return false; } } } } /** * @param string $var_name * * @return bool */ public function hasVariable($var_name) { return isset($this->all_vars[$var_name]); } /** * @param string $var_name * @param CodeLocation $location * * @return void */ public function registerVariable($var_name, CodeLocation $location) { $this->all_vars[$var_name] = $location; } /** * @param PhpParser\Node\Expr\Include_ $stmt * @param Context $context * * @return false|null */ public function analyzeInclude(PhpParser\Node\Expr\Include_ $stmt, Context $context) { $config = Config::getInstance(); if (!$config->allow_includes) { throw new FileIncludeException( 'File includes are not allowed per your Psalm config - check the allowFileIncludes flag.' ); } if (ExpressionChecker::analyze($this, $stmt->expr, $context) === false) { return false; } $path_to_file = null; if ($stmt->expr instanceof PhpParser\Node\Scalar\String_) { $path_to_file = $stmt->expr->value; // attempts to resolve using get_include_path dirs $include_path = self::resolveIncludePath($path_to_file, dirname($this->getCheckedFileName())); $path_to_file = $include_path ? $include_path : $path_to_file; if ($path_to_file[0] !== DIRECTORY_SEPARATOR) { $path_to_file = getcwd() . DIRECTORY_SEPARATOR . $path_to_file; } } else { $path_to_file = self::getPathTo($stmt->expr, $this->getCheckedFileName()); } if ($path_to_file) { $reduce_pattern = '/\/[^\/]+\/\.\.\//'; while (preg_match($reduce_pattern, $path_to_file)) { $path_to_file = preg_replace($reduce_pattern, DIRECTORY_SEPARATOR, $path_to_file); } // if the file is already included, we can't check much more if (in_array($path_to_file, get_included_files(), true)) { return null; } if ($this->getFilePath() === $path_to_file) { return null; } $current_file_checker = $this->getFileChecker(); if ($current_file_checker->project_checker->fileExists($path_to_file)) { if (is_subclass_of($current_file_checker, 'Psalm\\Checker\\FileChecker')) { $include_file_checker = new FileChecker( $path_to_file, $current_file_checker->project_checker, false ); $this->analyze($include_file_checker->getStatements(), $context); } return null; } if (IssueBuffer::accepts( new MissingFile( 'Cannot find file ' . $path_to_file . ' to include', new CodeLocation($this->source, $stmt) ), $this->source->getSuppressedIssues() )) { // fall through } } else { if (IssueBuffer::accepts( new UnresolvableInclude( 'Cannot resolve the given expression to a file path', new CodeLocation($this->source, $stmt) ), $this->source->getSuppressedIssues() )) { // fall through } } $context->check_classes = false; $context->check_variables = false; $context->check_functions = false; return null; } /** * @param PhpParser\Node\Expr $stmt * @param string $file_name * * @return string|null * @psalm-suppress MixedAssignment */ public static function getPathTo(PhpParser\Node\Expr $stmt, $file_name) { if ($file_name[0] !== DIRECTORY_SEPARATOR) { $file_name = getcwd() . DIRECTORY_SEPARATOR . $file_name; } if ($stmt instanceof PhpParser\Node\Scalar\String_) { return $stmt->value; } elseif ($stmt instanceof PhpParser\Node\Expr\BinaryOp\Concat) { $left_string = self::getPathTo($stmt->left, $file_name); $right_string = self::getPathTo($stmt->right, $file_name); if ($left_string && $right_string) { return $left_string . $right_string; } } elseif ($stmt instanceof PhpParser\Node\Expr\FuncCall && $stmt->name instanceof PhpParser\Node\Name && $stmt->name->parts === ['dirname'] ) { if ($stmt->args) { $evaled_path = self::getPathTo($stmt->args[0]->value, $file_name); if (!$evaled_path) { return null; } return dirname($evaled_path); } } elseif ($stmt instanceof PhpParser\Node\Expr\ConstFetch && $stmt->name instanceof PhpParser\Node\Name) { $const_name = implode('', $stmt->name->parts); if (defined($const_name)) { $constant_value = constant($const_name); if (is_string($constant_value)) { return $constant_value; } } } elseif ($stmt instanceof PhpParser\Node\Scalar\MagicConst\Dir) { return dirname($file_name); } elseif ($stmt instanceof PhpParser\Node\Scalar\MagicConst\File) { return $file_name; } return null; } /** * @param string $file_name * @param string $current_directory * * @return string|null */ public static function resolveIncludePath($file_name, $current_directory) { if (!$current_directory) { return $file_name; } $paths = PATH_SEPARATOR == ':' ? preg_split('#(?all_vars[$var_name]) ? $this->all_vars[$var_name] : null; } /** * @return FileChecker */ public function getFileChecker() { return $this->file_checker; } /** * @return array */ public function getFunctionCheckers() { return $this->function_checkers; } /** * @return void */ public static function clearCache() { self::$stub_constants = []; ExpressionChecker::clearCache(); } }