$args */ public static function analyze( StatementsAnalyzer $statements_analyzer, PhpParser\Node\Expr\MethodCall $stmt, PhpParser\Node\Identifier $stmt_name, array $args, Codebase $codebase, Context $context, Type\Atomic\TNamedObject $lhs_type_part, ?Type\Atomic $static_type, ?string $lhs_var_id, MethodIdentifier $method_id, AtomicMethodCallAnalysisResult $result ) : Type\Union { $config = $codebase->config; $fq_class_name = $lhs_type_part->value; if ($fq_class_name === 'static') { $fq_class_name = (string) $context->self; } $method_name_lc = $method_id->method_name; $cased_method_id = $fq_class_name . '::' . $stmt_name->name; $result->existent_method_ids[] = $method_id; if ($context->collect_initializations && $context->calling_method_id) { [$calling_method_class] = explode('::', $context->calling_method_id); $codebase->file_reference_provider->addMethodReferenceToClassMember( $calling_method_class . '::__construct', strtolower((string) $method_id) ); } if ($codebase->store_node_types && !$context->collect_initializations && !$context->collect_mutations ) { ArgumentMapPopulator::recordArgumentPositions( $statements_analyzer, $stmt, $codebase, (string) $method_id ); } if ($fq_class_name === 'Closure' && $method_name_lc === '__invoke') { $statements_analyzer->node_data = clone $statements_analyzer->node_data; $fake_function_call = new PhpParser\Node\Expr\FuncCall( $stmt->var, $args, $stmt->getAttributes() ); FunctionCallAnalyzer::analyze( $statements_analyzer, $fake_function_call, $context ); $function_return = $statements_analyzer->node_data->getType($fake_function_call) ?: Type::getMixed(); return $function_return; } $source = $statements_analyzer->getSource(); 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_method_id) { [$calling_method_class] = explode('::', $context->calling_method_id); $codebase->file_reference_provider->addMethodReferenceToClassMember( $calling_method_class . '::__construct', strtolower((string) $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); $class_storage = $codebase->classlike_storage_provider->get($fq_class_name); $parent_source = $statements_analyzer->getSource(); $class_template_params = ClassTemplateParamCollector::collect( $codebase, $codebase->methods->getClassLikeStorageForMethod($method_id), $class_storage, $method_name_lc, $lhs_type_part, $lhs_var_id === '$this' ); 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), $class_storage, $method_name_lc, $lhs_type_part, true ); } } } $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 Type::getMixed(); } $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, $method_id, $cased_method_id, $lhs_type_part, $static_type, $args, $result, $template_result ); $in_call_map = InternalCallMapHandler::inCallMap((string) ($declaring_method_id ?: $method_id)); if (!$in_call_map) { $name_code_location = new CodeLocation($statements_analyzer, $stmt_name); MethodCallProhibitionAnalyzer::analyze( $codebase, $context, $method_id, $statements_analyzer->getNamespace(), $name_code_location, $statements_analyzer->getSuppressedIssues() ); $getter_return_type = self::getMagicGetterOrSetterProperty( $statements_analyzer, $stmt, $stmt_name, $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 = MethodCallPurityAnalyzer::analyze( $statements_analyzer, $codebase, $stmt, $lhs_var_id, $cased_method_id, $method_id, $method_storage, $class_storage, $context, $config ); } $has_packed_arg = false; foreach ($args as $arg) { $has_packed_arg = $has_packed_arg || $arg->unpack; } if (!$has_packed_arg) { $has_variadic_param = $method_storage->variadic; 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; } } $class_template_params = $template_result->upper_bounds; if ($method_storage->assertions) { self::applyAssertionsToContext( $stmt_name, ExpressionIdentifier::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 (!$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; } } else { $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, $statements_analyzer, $codebase, $file_manipulations, $return_type_candidate ); } } if ($file_manipulations) { FileManipulationBuffer::add($statements_analyzer->getFilePath(), $file_manipulations); } } return $return_type_candidate ?: Type::getMixed(); } /** * 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. */ private static function getMagicGetterOrSetterProperty( StatementsAnalyzer $statements_analyzer, PhpParser\Node\Expr\MethodCall $stmt, PhpParser\Node\Identifier $stmt_name, Context $context, string $fq_class_name ) : ?Type\Union { $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 = \Psalm\Internal\Type\TypeExpander::expandUnion( $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\Type\Comparator\TypeComparisonResult(); $type_match_found = UnionTypeComparator::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->getId() . '\', ' . ' 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->getId() . '\', ' . ' 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 (UnionTypeComparator::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; } }