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 = MissingMethodCallHandler::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])) ) { MissingMethodCallHandler::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 = ClassTemplateParamCollector::collect( $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 = ClassTemplateParamCollector::collect( $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 = MethodCallReturnTypeFetcher::fetch( $statements_analyzer, $codebase, $stmt, $context, $method_id, $declaring_method_id, $cased_method_id, $lhs_type_part, $static_type, $args, $result, $template_result ); $in_call_map = CallMap::inCallMap((string) ($declaring_method_id ?: $method_id)); if (!$in_call_map) { $name_code_location = new CodeLocation($statements_analyzer, $stmt->name); if ($result->check_visibility) { if (MethodVisibilityAnalyzer::analyze( $method_id, $context, $statements_analyzer->getSource(), $name_code_location, $statements_analyzer->getSuppressedIssues() ) === false) { return; } } MethodCallProhibitionAnalyzer::analyze( $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) { $has_packed_arg = false; foreach ($args as $arg) { $has_packed_arg = $has_packed_arg || $arg->unpack; } if (!$has_packed_arg) { $has_variadic_param = false; foreach ($method_storage->params as $param) { $has_variadic_param = $has_variadic_param || $param->is_variadic; } for ($i = count($args), $j = count($method_storage->params); $i < $j; ++$i) { $param = $method_storage->params[$i]; if (!$param->is_optional && !$param->is_variadic && !$in_call_map ) { $result->too_few_arguments = true; $result->too_few_arguments_method_ids[] = $declaring_method_id ?: $method_id; } } if ($has_variadic_param || count($method_storage->params) >= count($args) || $in_call_map) { $result->too_many_arguments = false; } else { $result->too_many_arguments_method_ids[] = $declaring_method_id ?: $method_id; } } if (!$context->collect_mutations && !$context->collect_initializations) { $can_memoize = MethodCallPurityAnalyzer::analyze( $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; } } /** * 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; } }