config; if ($lhs_type_part instanceof Type\Atomic\TTemplateParam && !$lhs_type_part->as->isMixed() ) { $extra_types = $lhs_type_part->extra_types; $lhs_type_part = array_values( $lhs_type_part->as->getAtomicTypes() )[0]; $lhs_type_part->from_docblock = true; if ($lhs_type_part instanceof TNamedObject) { $lhs_type_part->extra_types = $extra_types; } elseif ($lhs_type_part instanceof Type\Atomic\TObject && $extra_types) { $lhs_type_part = array_shift($extra_types); if ($extra_types) { $lhs_type_part->extra_types = $extra_types; } } $result->has_mixed_method_call = true; } $source = $statements_analyzer->getSource(); if (!$lhs_type_part instanceof TNamedObject) { self::handleInvalidClass( $statements_analyzer, $codebase, $stmt, $lhs_type_part, $lhs_var_id, $context, $is_intersection, $result ); return; } if (!$context->collect_initializations && !$context->collect_mutations && $statements_analyzer->getFilePath() === $statements_analyzer->getRootFilePath() && (!(($parent_source = $statements_analyzer->getSource()) instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer) || !$parent_source->getSource() instanceof \Psalm\Internal\Analyzer\TraitAnalyzer) ) { $codebase->analyzer->incrementNonMixedCount($statements_analyzer->getFilePath()); } $result->has_valid_method_call_type = true; $fq_class_name = $lhs_type_part->value; $is_mock = ExpressionAnalyzer::isMock($fq_class_name); $result->has_mock = $result->has_mock || $is_mock; if ($fq_class_name === 'static') { $fq_class_name = (string) $context->self; } if ($is_mock || $context->isPhantomClass($fq_class_name) ) { $result->return_type = Type::getMixed(); self::checkFunctionArguments( $statements_analyzer, $stmt->args, null, null, $context ); return; } if ($lhs_var_id === '$this') { $does_class_exist = true; } else { $does_class_exist = ClassLikeAnalyzer::checkFullyQualifiedClassLikeName( $statements_analyzer, $fq_class_name, new CodeLocation($source, $stmt->var), $context->self, $statements_analyzer->getSuppressedIssues(), true, false, true, $lhs_type_part->from_docblock ); } if (!$does_class_exist) { return; } $class_storage = $codebase->classlike_storage_provider->get($fq_class_name); $result->check_visibility = $result->check_visibility && !$class_storage->override_method_visibility; $intersection_types = $lhs_type_part->getIntersectionTypes(); $all_intersection_return_type = null; $all_intersection_existent_method_ids = []; if ($intersection_types) { foreach ($intersection_types as $intersection_type) { $intersection_result = clone $result; /** @var ?Type\Union */ $intersection_result->return_type = null; self::analyze( $statements_analyzer, $stmt, $codebase, $context, $intersection_type, $lhs_type_part, true, $lhs_var_id, $intersection_result ); $result->returns_by_ref = $intersection_result->returns_by_ref; $result->has_mock = $intersection_result->has_mock; $result->has_valid_method_call_type = $intersection_result->has_valid_method_call_type; $result->has_mixed_method_call = $intersection_result->has_mixed_method_call; $result->invalid_method_call_types = $intersection_result->invalid_method_call_types; $result->check_visibility = $intersection_result->check_visibility; $all_intersection_existent_method_ids = array_merge( $all_intersection_existent_method_ids, $intersection_result->existent_method_ids ); if ($intersection_result->return_type) { if (!$all_intersection_return_type || $all_intersection_return_type->isMixed()) { $all_intersection_return_type = $intersection_result->return_type; } else { $all_intersection_return_type = Type::intersectUnionTypes( $all_intersection_return_type, $intersection_result->return_type ) ?: Type::getMixed(); } } } } if (!$stmt->name instanceof PhpParser\Node\Identifier) { if (!$context->ignore_variable_method) { $codebase->analyzer->addMixedMemberName( strtolower($fq_class_name) . '::', $context->calling_function_id ?: $statements_analyzer->getFileName() ); } $result->return_type = Type::getMixed(); return; } $method_name_lc = strtolower($stmt->name->name); $method_id = new MethodIdentifier($fq_class_name, $method_name_lc); $cased_method_id = $fq_class_name . '::' . $stmt->name->name; $intersection_method_id = $intersection_types ? '(' . $lhs_type_part . ')' . '::' . $stmt->name->name : null; $args = $stmt->args; $old_node_data = null; if (!$codebase->methods->methodExists( $method_id, $context->calling_function_id, $codebase->collect_references ? new CodeLocation($source, $stmt->name) : null, !$context->collect_initializations && !$context->collect_mutations ? $statements_analyzer : null, $statements_analyzer->getFilePath() ) || !MethodAnalyzer::isMethodVisible( $method_id, $context, $statements_analyzer->getSource() ) ) { $interface_has_method = false; if ($class_storage->abstract && $class_storage->class_implements) { foreach ($class_storage->class_implements as $interface_fqcln_lc => $_) { $interface_storage = $codebase->classlike_storage_provider->get($interface_fqcln_lc); if (isset($interface_storage->methods[$method_name_lc])) { $interface_has_method = true; $fq_class_name = $interface_storage->name; $method_id = new MethodIdentifier( $fq_class_name, $method_name_lc ); break; } } } if (!$interface_has_method && $codebase->methods->methodExists( new MethodIdentifier($fq_class_name, '__call'), $context->calling_function_id ) ) { $new_call_context = self::handleMagicMethod( $statements_analyzer, $codebase, $stmt, $method_id, $class_storage, $context, $all_intersection_return_type, $result ); if ($new_call_context) { if ($method_id === $new_call_context->method_id) { return; } $method_id = $new_call_context->method_id; $args = $new_call_context->args; $old_node_data = $statements_analyzer->node_data; } else { return; } } } $source_source = $statements_analyzer->getSource(); /** * @var \Psalm\Internal\Analyzer\ClassLikeAnalyzer|null */ $classlike_source = $source_source->getSource(); $classlike_source_fqcln = $classlike_source ? $classlike_source->getFQCLN() : null; if ($lhs_var_id === '$this' && $context->self && $classlike_source_fqcln && $fq_class_name !== $context->self && $codebase->methods->methodExists( new MethodIdentifier($context->self, $method_name_lc) ) ) { $method_id = new MethodIdentifier($context->self, $method_name_lc); $cased_method_id = $context->self . '::' . $stmt->name->name; $fq_class_name = $context->self; } $is_interface = false; if ($codebase->interfaceExists($fq_class_name)) { $is_interface = true; } $source_method_id = $source instanceof FunctionLikeAnalyzer ? $source->getId() : null; if (!$codebase->methods->methodExists( $method_id, $context->calling_function_id, $method_id !== $source_method_id ? new CodeLocation($source, $stmt->name) : null ) || ($config->use_phpdoc_method_without_magic_or_parent && isset($class_storage->pseudo_methods[$method_name_lc])) ) { self::handleMissingOrMagicMethod( $statements_analyzer, $codebase, $stmt, $method_id, $is_interface, $context, $config, $all_intersection_return_type, $result ); if ($all_intersection_return_type && $all_intersection_existent_method_ids) { $result->existent_method_ids = array_merge( $result->existent_method_ids, $all_intersection_existent_method_ids ); if (!$result->return_type) { $result->return_type = $all_intersection_return_type; } else { $result->return_type = Type::combineUnionTypes($all_intersection_return_type, $result->return_type); } return; } if ((!$is_interface && !$config->use_phpdoc_method_without_magic_or_parent) || !isset($class_storage->pseudo_methods[$method_name_lc]) ) { if ($is_interface) { $result->non_existent_interface_method_ids[] = $intersection_method_id ?: $method_id; } else { $result->non_existent_class_method_ids[] = $intersection_method_id ?: $method_id; } } return; } if ($codebase->store_node_types && !$context->collect_initializations && !$context->collect_mutations ) { $codebase->analyzer->addNodeReference( $statements_analyzer->getFilePath(), $stmt->name, $method_id . '()' ); } if ($context->collect_initializations && $context->calling_function_id) { list($calling_method_class) = explode('::', $context->calling_function_id); $codebase->file_reference_provider->addMethodReferenceToClassMember( $calling_method_class . '::__construct', strtolower((string) $method_id) ); } $result->existent_method_ids[] = $method_id; if ($stmt->var instanceof PhpParser\Node\Expr\Variable && ($context->collect_initializations || $context->collect_mutations) && $stmt->var->name === 'this' && $source instanceof FunctionLikeAnalyzer ) { self::collectSpecialInformation($source, $stmt->name->name, $context); } $fq_class_name = $codebase->classlikes->getUnAliasedName($fq_class_name); $parent_source = $statements_analyzer->getSource(); $class_template_params = MethodCallAnalyzer::getClassTemplateParams( $codebase, $codebase->methods->getClassLikeStorageForMethod($method_id), $fq_class_name, $method_name_lc, $lhs_type_part, $lhs_var_id ); if ($lhs_var_id === '$this' && $parent_source instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer) { $grandparent_source = $parent_source->getSource(); if ($grandparent_source instanceof \Psalm\Internal\Analyzer\TraitAnalyzer) { $fq_trait_name = $grandparent_source->getFQCLN(); $fq_trait_name_lc = strtolower($fq_trait_name); $trait_storage = $codebase->classlike_storage_provider->get($fq_trait_name_lc); if (isset($trait_storage->methods[$method_name_lc])) { $trait_method_id = new MethodIdentifier($trait_storage->name, $method_name_lc); $class_template_params = MethodCallAnalyzer::getClassTemplateParams( $codebase, $codebase->methods->getClassLikeStorageForMethod($trait_method_id), $fq_class_name, $method_name_lc, $lhs_type_part, $lhs_var_id ); } } } $template_result = new \Psalm\Internal\Type\TemplateResult([], $class_template_params ?: []); if ($codebase->store_node_types && !$context->collect_initializations && !$context->collect_mutations ) { ArgumentMapPopulator::recordArgumentPositions( $statements_analyzer, $stmt, $codebase, (string) $method_id ); } if (self::checkMethodArgs( $method_id, $args, $template_result, $context, new CodeLocation($source, $stmt->name), $statements_analyzer ) === false) { return; } $declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id); $can_memoize = false; $return_type_candidate = self::getAtomicReturnType( $statements_analyzer, $codebase, $stmt, $context, $method_id, $declaring_method_id, $cased_method_id, $lhs_type_part, $static_type, $args, $result, $template_result ); if (!CallMap::inCallMap((string) ($declaring_method_id ?: $method_id))) { $name_code_location = new CodeLocation($statements_analyzer, $stmt->name); if ($result->check_visibility) { if (MethodAnalyzer::checkMethodVisibility( $method_id, $context, $statements_analyzer->getSource(), $name_code_location, $statements_analyzer->getSuppressedIssues() ) === false) { return; } } MethodAnalyzer::checkMethodNotDeprecatedOrInternal( $codebase, $context, $method_id, $name_code_location, $statements_analyzer->getSuppressedIssues() ); $getter_return_type = self::getMagicGetterOrSetterProperty( $statements_analyzer, $stmt, $context, $fq_class_name ); if ($getter_return_type) { $return_type_candidate = $getter_return_type; } } try { $method_storage = $codebase->methods->getStorage($declaring_method_id ?: $method_id); } catch (\UnexpectedValueException $e) { $method_storage = null; } if ($method_storage) { if (!$context->collect_mutations && !$context->collect_initializations) { $can_memoize = self::checkCallPurity( $statements_analyzer, $codebase, $stmt, $lhs_var_id, $cased_method_id, $method_id, $method_storage, $class_storage, $context, $config ); } $class_template_params = $template_result->generic_params; if ($method_storage->assertions) { self::applyAssertionsToContext( $stmt->name, ExpressionAnalyzer::getArrayVarId($stmt->var, null, $statements_analyzer), $method_storage->assertions, $args, $class_template_params, $context, $statements_analyzer ); } if ($method_storage->if_true_assertions) { $statements_analyzer->node_data->setIfTrueAssertions( $stmt, array_map( function (Assertion $assertion) use ( $class_template_params, $lhs_var_id ) : Assertion { return $assertion->getUntemplatedCopy( $class_template_params ?: [], $lhs_var_id ); }, $method_storage->if_true_assertions ) ); } if ($method_storage->if_false_assertions) { $statements_analyzer->node_data->setIfFalseAssertions( $stmt, array_map( function (Assertion $assertion) use ( $class_template_params, $lhs_var_id ) : Assertion { return $assertion->getUntemplatedCopy( $class_template_params ?: [], $lhs_var_id ); }, $method_storage->if_false_assertions ) ); } } if ($old_node_data) { $statements_analyzer->node_data = $old_node_data; } if (!$args && $lhs_var_id) { if ($config->memoize_method_calls || $can_memoize) { $method_var_id = $lhs_var_id . '->' . $method_name_lc . '()'; if (isset($context->vars_in_scope[$method_var_id])) { $return_type_candidate = clone $context->vars_in_scope[$method_var_id]; if ($can_memoize) { /** @psalm-suppress UndefinedPropertyAssignment */ $stmt->pure = true; } } elseif ($return_type_candidate) { $context->vars_in_scope[$method_var_id] = $return_type_candidate; } } } if ($codebase->methods_to_rename) { $declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id); foreach ($codebase->methods_to_rename as $original_method_id => $new_method_name) { if ($declaring_method_id && (strtolower((string) $declaring_method_id)) === $original_method_id) { $file_manipulations = [ new \Psalm\FileManipulation( (int) $stmt->name->getAttribute('startFilePos'), (int) $stmt->name->getAttribute('endFilePos') + 1, $new_method_name ) ]; \Psalm\Internal\FileManipulation\FileManipulationBuffer::add( $statements_analyzer->getFilePath(), $file_manipulations ); } } } if ($config->after_method_checks) { $file_manipulations = []; $appearing_method_id = $codebase->methods->getAppearingMethodId($method_id); $declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id); if ($appearing_method_id !== null && $declaring_method_id !== null) { foreach ($config->after_method_checks as $plugin_fq_class_name) { $plugin_fq_class_name::afterMethodCallAnalysis( $stmt, (string) $method_id, (string) $appearing_method_id, (string) $declaring_method_id, $context, $source, $codebase, $file_manipulations, $return_type_candidate ); } } if ($file_manipulations) { FileManipulationBuffer::add($statements_analyzer->getFilePath(), $file_manipulations); } } if ($return_type_candidate) { if ($all_intersection_return_type) { $return_type_candidate = Type::intersectUnionTypes( $all_intersection_return_type, $return_type_candidate ) ?: Type::getMixed(); } if (!$result->return_type) { $result->return_type = $return_type_candidate; } else { $result->return_type = Type::combineUnionTypes($return_type_candidate, $result->return_type); } } elseif ($all_intersection_return_type) { if (!$result->return_type) { $result->return_type = $all_intersection_return_type; } else { $result->return_type = Type::combineUnionTypes($all_intersection_return_type, $result->return_type); } } elseif (strtolower($stmt->name->name) === '__tostring') { $result->return_type = Type::getString(); } else { $result->return_type = Type::getMixed(); } } private static function handleInvalidClass( StatementsAnalyzer $statements_analyzer, Codebase $codebase, PhpParser\Node\Expr\MethodCall $stmt, Type\Atomic $lhs_type_part, ?string $lhs_var_id, Context $context, bool $is_intersection, AtomicMethodCallAnalysisResult $result ) : void { switch (get_class($lhs_type_part)) { case Type\Atomic\TNull::class: case Type\Atomic\TFalse::class: // handled above return; case Type\Atomic\TTemplateParam::class: case Type\Atomic\TEmptyMixed::class: case Type\Atomic\TEmpty::class: case Type\Atomic\TMixed::class: case Type\Atomic\TNonEmptyMixed::class: case Type\Atomic\TObject::class: case Type\Atomic\TObjectWithProperties::class: if (!$context->collect_initializations && !$context->collect_mutations && $statements_analyzer->getFilePath() === $statements_analyzer->getRootFilePath() && (!(($parent_source = $statements_analyzer->getSource()) instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer) || !$parent_source->getSource() instanceof \Psalm\Internal\Analyzer\TraitAnalyzer) ) { $codebase->analyzer->incrementMixedCount($statements_analyzer->getFilePath()); } $result->has_mixed_method_call = true; if ($lhs_type_part instanceof Type\Atomic\TObjectWithProperties && $stmt->name instanceof PhpParser\Node\Identifier && isset($lhs_type_part->methods[$stmt->name->name]) ) { $result->existent_method_ids[] = $lhs_type_part->methods[$stmt->name->name]; } elseif (!$is_intersection) { if ($stmt->name instanceof PhpParser\Node\Identifier) { $codebase->analyzer->addMixedMemberName( strtolower($stmt->name->name), $context->calling_function_id ?: $statements_analyzer->getFileName() ); } if ($context->check_methods) { $message = 'Cannot determine the type of the object' . ' on the left hand side of this expression'; if ($lhs_var_id) { $message = 'Cannot determine the type of ' . $lhs_var_id; if ($stmt->name instanceof PhpParser\Node\Identifier) { $message .= ' when calling method ' . $stmt->name->name; } } if (IssueBuffer::accepts( new MixedMethodCall( $message, new CodeLocation($statements_analyzer, $stmt->name) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } } if (self::checkFunctionArguments( $statements_analyzer, $stmt->args, null, null, $context ) === false) { return; } $result->return_type = Type::getMixed(); return; default: $result->invalid_method_call_types[] = (string)$lhs_type_part; return; } } private static function handleMagicMethod( StatementsAnalyzer $statements_analyzer, Codebase $codebase, PhpParser\Node\Expr\MethodCall $stmt, MethodIdentifier $method_id, \Psalm\Storage\ClassLikeStorage $class_storage, Context $context, ?Type\Union $all_intersection_return_type, AtomicMethodCallAnalysisResult $result ) : ?AtomicCallContext { $fq_class_name = $method_id->fq_class_name; $method_name_lc = $method_id->method_name; if (isset($class_storage->pseudo_methods[$method_name_lc])) { $result->has_valid_method_call_type = true; $result->existent_method_ids[] = $method_id; $pseudo_method_storage = $class_storage->pseudo_methods[$method_name_lc]; self::checkFunctionArguments( $statements_analyzer, $stmt->args, $pseudo_method_storage->params, (string) $method_id, $context ); self::checkFunctionLikeArgumentsMatch( $statements_analyzer, $stmt->args, null, $pseudo_method_storage->params, $pseudo_method_storage, null, null, new CodeLocation($statements_analyzer, $stmt), $context ); if ($pseudo_method_storage->return_type) { $return_type_candidate = clone $pseudo_method_storage->return_type; $return_type_candidate = ExpressionAnalyzer::fleshOutType( $codebase, $return_type_candidate, $fq_class_name, $fq_class_name, $class_storage->parent_class ); if ($all_intersection_return_type) { $return_type_candidate = Type::intersectUnionTypes( $all_intersection_return_type, $return_type_candidate ) ?: Type::getMixed(); } if (!$result->return_type) { $result->return_type = $return_type_candidate; } else { $result->return_type = Type::combineUnionTypes( $return_type_candidate, $result->return_type ); } return null; } } else { self::checkFunctionArguments( $statements_analyzer, $stmt->args, null, null, $context ); if ($class_storage->sealed_methods) { $result->non_existent_magic_method_ids[] = $method_id; return null; } } $result->has_valid_method_call_type = true; $result->existent_method_ids[] = $method_id; $array_values = array_map( /** * @return PhpParser\Node\Expr\ArrayItem */ function (PhpParser\Node\Arg $arg) { return new PhpParser\Node\Expr\ArrayItem($arg->value); }, $stmt->args ); $old_node_data = $statements_analyzer->node_data; $statements_analyzer->node_data = clone $statements_analyzer->node_data; return new AtomicCallContext( new MethodIdentifier( $fq_class_name, '__call' ), [ new PhpParser\Node\Arg(new PhpParser\Node\Scalar\String_($method_name_lc)), new PhpParser\Node\Arg(new PhpParser\Node\Expr\Array_($array_values)), ], $old_node_data ); } private static function handleMissingOrMagicMethod( StatementsAnalyzer $statements_analyzer, Codebase $codebase, PhpParser\Node\Expr\MethodCall $stmt, MethodIdentifier $method_id, bool $is_interface, Context $context, \Psalm\Config $config, ?Type\Union $all_intersection_return_type, AtomicMethodCallAnalysisResult $result ) : void { $fq_class_name = $method_id->fq_class_name; $method_name_lc = $method_id->method_name; $class_storage = $codebase->classlike_storage_provider->get($fq_class_name); if (($is_interface || $config->use_phpdoc_method_without_magic_or_parent) && isset($class_storage->pseudo_methods[$method_name_lc]) ) { $result->has_valid_method_call_type = true; $result->existent_method_ids[] = $method_id; $pseudo_method_storage = $class_storage->pseudo_methods[$method_name_lc]; if (self::checkFunctionArguments( $statements_analyzer, $stmt->args, $pseudo_method_storage->params, (string) $method_id, $context ) === false) { return; } if (self::checkFunctionLikeArgumentsMatch( $statements_analyzer, $stmt->args, null, $pseudo_method_storage->params, $pseudo_method_storage, null, null, new CodeLocation($statements_analyzer, $stmt->name), $context ) === false) { return; } if ($pseudo_method_storage->return_type) { $return_type_candidate = clone $pseudo_method_storage->return_type; if ($all_intersection_return_type) { $return_type_candidate = Type::intersectUnionTypes( $all_intersection_return_type, $return_type_candidate ) ?: Type::getMixed(); } if (!$result->return_type) { $result->return_type = $return_type_candidate; } else { $result->return_type = Type::combineUnionTypes($return_type_candidate, $result->return_type); } return; } $result->return_type = Type::getMixed(); return; } if (self::checkFunctionArguments( $statements_analyzer, $stmt->args, null, null, $context ) === false) { return; } } private static function checkCallPurity( StatementsAnalyzer $statements_analyzer, Codebase $codebase, PhpParser\Node\Expr\MethodCall $stmt, ?string $lhs_var_id, ?string $cased_method_id, MethodIdentifier $method_id, \Psalm\Storage\MethodStorage $method_storage, \Psalm\Storage\ClassLikeStorage $class_storage, Context $context, \Psalm\Config $config ) : bool { $can_memoize = false; $method_pure_compatible = $method_storage->external_mutation_free && $statements_analyzer->node_data->isPureCompatible($stmt->var); if ($context->pure && !$method_storage->mutation_free && !$method_pure_compatible ) { if (IssueBuffer::accepts( new ImpureMethodCall( 'Cannot call an mutation-free method ' . $cased_method_id . ' from a pure context', new CodeLocation($statements_analyzer, $stmt->name) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($context->mutation_free && !$method_storage->mutation_free && !$method_pure_compatible ) { if (IssueBuffer::accepts( new ImpureMethodCall( 'Cannot call an possibly-mutating method ' . $cased_method_id . ' from a mutation-free context', new CodeLocation($statements_analyzer, $stmt->name) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($context->external_mutation_free && !$method_storage->mutation_free && $method_id->fq_class_name !== $context->self && !$method_pure_compatible ) { if (IssueBuffer::accepts( new ImpureMethodCall( 'Cannot call an possibly-mutating method ' . $cased_method_id . ' from a mutation-free context', new CodeLocation($statements_analyzer, $stmt->name) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif (($method_storage->mutation_free || ($method_storage->external_mutation_free && (isset($stmt->var->external_mutation_free) || isset($stmt->var->pure)))) && !$context->inside_unset ) { if ($method_storage->mutation_free && !$method_storage->mutation_free_inferred) { if ($context->inside_conditional) { /** @psalm-suppress UndefinedPropertyAssignment */ $stmt->pure = true; } $can_memoize = true; } if ($codebase->find_unused_variables && !$context->inside_conditional) { if (!$context->inside_assignment && !$context->inside_call) { if (IssueBuffer::accepts( new \Psalm\Issue\UnusedMethodCall( 'The call to ' . $cased_method_id . ' is not used', new CodeLocation($statements_analyzer, $stmt->name), (string) $method_id ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif (!$method_storage->mutation_free_inferred) { /** @psalm-suppress UndefinedPropertyAssignment */ $stmt->pure = true; } } } if (!$config->remember_property_assignments_after_call && !$method_storage->mutation_free && !$method_pure_compatible ) { $context->removeAllObjectVars(); } elseif ($method_storage->this_property_mutations) { foreach ($method_storage->this_property_mutations as $name => $_) { $mutation_var_id = $lhs_var_id . '->' . $name; $this_property_didnt_exist = $lhs_var_id === '$this' && isset($context->vars_in_scope[$mutation_var_id]) && !isset($class_storage->declaring_property_ids[$name]); $context->remove($mutation_var_id); if ($this_property_didnt_exist) { $context->vars_in_scope[$mutation_var_id] = Type::getMixed(); } } } return $can_memoize; } /** * @param Type\Atomic\TNamedObject|Type\Atomic\TTemplateParam $static_type * @param array $args */ private static function getAtomicReturnType( StatementsAnalyzer $statements_analyzer, Codebase $codebase, PhpParser\Node\Expr\MethodCall $stmt, Context $context, MethodIdentifier $method_id, ?MethodIdentifier $declaring_method_id, string $cased_method_id, Type\Atomic $lhs_type_part, ?Type\Atomic $static_type, array $args, AtomicMethodCallAnalysisResult $result, TemplateResult $template_result ) : ?Type\Union { $call_map_id = $declaring_method_id ?: $method_id; $fq_class_name = $method_id->fq_class_name; $method_name = $method_id->method_name; $return_type_candidate = null; if ($codebase->methods->return_type_provider->has($fq_class_name)) { $return_type_candidate = $codebase->methods->return_type_provider->getReturnType( $statements_analyzer, $fq_class_name, $method_name, $stmt->args, $context, new CodeLocation($statements_analyzer->getSource(), $stmt->name), $lhs_type_part instanceof TGenericObject ? $lhs_type_part->type_params : null ); } if (!$return_type_candidate && $declaring_method_id && $declaring_method_id !== $method_id) { $declaring_fq_class_name = $declaring_method_id->fq_class_name; $declaring_method_name = $declaring_method_id->method_name; if ($codebase->methods->return_type_provider->has($declaring_fq_class_name)) { $return_type_candidate = $codebase->methods->return_type_provider->getReturnType( $statements_analyzer, $declaring_fq_class_name, $declaring_method_name, $stmt->args, $context, new CodeLocation($statements_analyzer->getSource(), $stmt->name), $lhs_type_part instanceof TGenericObject ? $lhs_type_part->type_params : null, $fq_class_name, $method_name ); } } $class_storage = $codebase->methods->getClassLikeStorageForMethod($method_id); if (!$return_type_candidate) { if (CallMap::inCallMap((string) $call_map_id)) { if (($template_result->generic_params || $class_storage->stubbed) && isset($class_storage->methods[$method_id->method_name]) && ($method_storage = $class_storage->methods[$method_id->method_name]) && $method_storage->return_type ) { $return_type_candidate = clone $method_storage->return_type; if ($template_result->generic_params) { $return_type_candidate->replaceTemplateTypesWithArgTypes( $template_result->generic_params, $codebase ); } } else { $callmap_callables = CallMap::getCallablesFromCallMap((string) $call_map_id); if (!$callmap_callables || $callmap_callables[0]->return_type === null) { throw new \UnexpectedValueException('Shouldn’t get here'); } $return_type_candidate = $callmap_callables[0]->return_type; } if ($return_type_candidate->isFalsable()) { $return_type_candidate->ignore_falsable_issues = true; } $return_type_candidate = ExpressionAnalyzer::fleshOutType( $codebase, $return_type_candidate, $fq_class_name, $static_type, $class_storage->parent_class ); } else { $self_fq_class_name = $fq_class_name; $return_type_candidate = $codebase->methods->getMethodReturnType( $method_id, $self_fq_class_name, $statements_analyzer, $args ); if ($return_type_candidate) { $return_type_candidate = clone $return_type_candidate; if ($template_result->template_types) { $bindable_template_types = $return_type_candidate->getTemplateTypes(); foreach ($bindable_template_types as $template_type) { if ($template_type->defining_class !== $fq_class_name && !isset( $template_result->generic_params [$template_type->param_name] [$template_type->defining_class] ) ) { $template_result->generic_params[$template_type->param_name] = [ ($template_type->defining_class) => [Type::getEmpty(), 0] ]; } } } if ($template_result->generic_params) { $return_type_candidate->replaceTemplateTypesWithArgTypes( $template_result->generic_params, $codebase ); } $return_type_candidate = ExpressionAnalyzer::fleshOutType( $codebase, $return_type_candidate, $self_fq_class_name, $static_type, $class_storage->parent_class ); $return_type_candidate->sources = [ new Source( strtolower((string) $method_id), $cased_method_id, new CodeLocation($statements_analyzer, $stmt->name) ) ]; $return_type_location = $codebase->methods->getMethodReturnTypeLocation( $method_id, $secondary_return_type_location ); if ($secondary_return_type_location) { $return_type_location = $secondary_return_type_location; } $config = \Psalm\Config::getInstance(); // only check the type locally if it's defined externally if ($return_type_location && !$config->isInProjectDirs($return_type_location->file_path)) { $return_type_candidate->check( $statements_analyzer, new CodeLocation($statements_analyzer, $stmt), $statements_analyzer->getSuppressedIssues(), $context->phantom_classes ); } } else { $result->returns_by_ref = $result->returns_by_ref || $codebase->methods->getMethodReturnsByRef($method_id); } } } return $return_type_candidate; } /** * Check properties accessed with magic getters and setters. * If `@psalm-seal-properties` is set, they must be defined. * If an `@property` annotation is specified, the setter must set something with the correct * type. * * @param StatementsAnalyzer $statements_analyzer * @param PhpParser\Node\Expr\MethodCall $stmt * @param string $fq_class_name */ private static function getMagicGetterOrSetterProperty( StatementsAnalyzer $statements_analyzer, PhpParser\Node\Expr\MethodCall $stmt, Context $context, $fq_class_name ) : ?Type\Union { if (!$stmt->name instanceof PhpParser\Node\Identifier) { return null; } $method_name = strtolower($stmt->name->name); if (!in_array($method_name, ['__get', '__set'], true)) { return null; } $codebase = $statements_analyzer->getCodebase(); $first_arg_value = $stmt->args[0]->value; if (!$first_arg_value instanceof PhpParser\Node\Scalar\String_) { return null; } $prop_name = $first_arg_value->value; $property_id = $fq_class_name . '::$' . $prop_name; $class_storage = $codebase->classlike_storage_provider->get($fq_class_name); $codebase->properties->propertyExists( $property_id, $method_name === '__get', $statements_analyzer, $context, new CodeLocation($statements_analyzer->getSource(), $stmt) ); switch ($method_name) { case '__set': // If `@psalm-seal-properties` is set, the property must be defined with // a `@property` annotation if ($class_storage->sealed_properties && !isset($class_storage->pseudo_property_set_types['$' . $prop_name]) && IssueBuffer::accepts( new UndefinedThisPropertyAssignment( 'Instance property ' . $property_id . ' is not defined', new CodeLocation($statements_analyzer->getSource(), $stmt), $property_id ), $statements_analyzer->getSuppressedIssues() ) ) { // fall through } // If a `@property` annotation is set, the type of the value passed to the // magic setter must match the annotation. $second_arg_type = $statements_analyzer->node_data->getType($stmt->args[1]->value); if (isset($class_storage->pseudo_property_set_types['$' . $prop_name]) && $second_arg_type) { $pseudo_set_type = ExpressionAnalyzer::fleshOutType( $codebase, $class_storage->pseudo_property_set_types['$' . $prop_name], $fq_class_name, new Type\Atomic\TNamedObject($fq_class_name), $class_storage->parent_class ); $union_comparison_results = new \Psalm\Internal\Analyzer\TypeComparisonResult(); $type_match_found = TypeAnalyzer::isContainedBy( $codebase, $second_arg_type, $pseudo_set_type, $second_arg_type->ignore_nullable_issues, $second_arg_type->ignore_falsable_issues, $union_comparison_results ); if ($union_comparison_results->type_coerced) { if ($union_comparison_results->type_coerced_from_mixed) { if (IssueBuffer::accepts( new MixedPropertyTypeCoercion( $prop_name . ' expects \'' . $pseudo_set_type . '\', ' . ' parent type `' . $second_arg_type . '` provided', new CodeLocation($statements_analyzer->getSource(), $stmt), $property_id ), $statements_analyzer->getSuppressedIssues() )) { // keep soldiering on } } else { if (IssueBuffer::accepts( new PropertyTypeCoercion( $prop_name . ' expects \'' . $pseudo_set_type . '\', ' . ' parent type `' . $second_arg_type . '` provided', new CodeLocation($statements_analyzer->getSource(), $stmt), $property_id ), $statements_analyzer->getSuppressedIssues() )) { // keep soldiering on } } } if (!$type_match_found && !$union_comparison_results->type_coerced_from_mixed) { if (TypeAnalyzer::canBeContainedBy( $codebase, $second_arg_type, $pseudo_set_type )) { if (IssueBuffer::accepts( new PossiblyInvalidPropertyAssignmentValue( $prop_name . ' with declared type \'' . $pseudo_set_type . '\' cannot be assigned possibly different type \'' . $second_arg_type . '\'', new CodeLocation($statements_analyzer->getSource(), $stmt), $property_id ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } else { if (IssueBuffer::accepts( new InvalidPropertyAssignmentValue( $prop_name . ' with declared type \'' . $pseudo_set_type . '\' cannot be assigned type \'' . $second_arg_type . '\'', new CodeLocation($statements_analyzer->getSource(), $stmt), $property_id ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } } } break; case '__get': // If `@psalm-seal-properties` is set, the property must be defined with // a `@property` annotation if ($class_storage->sealed_properties && !isset($class_storage->pseudo_property_get_types['$' . $prop_name]) && IssueBuffer::accepts( new UndefinedThisPropertyFetch( 'Instance property ' . $property_id . ' is not defined', new CodeLocation($statements_analyzer->getSource(), $stmt), $property_id ), $statements_analyzer->getSuppressedIssues() ) ) { // fall through } if (isset($class_storage->pseudo_property_get_types['$' . $prop_name])) { return clone $class_storage->pseudo_property_get_types['$' . $prop_name]; } break; } return null; } }