getCodebase(); $config = $codebase->config; $can_extend = false; $from_static = false; if ($stmt->class instanceof PhpParser\Node\Name) { if (!in_array(strtolower($stmt->class->parts[0]), ['self', 'static', 'parent'], true)) { $aliases = $statements_analyzer->getAliases(); if ($context->calling_method_id && !$stmt->class instanceof PhpParser\Node\Name\FullyQualified ) { $codebase->file_reference_provider->addMethodReferenceToClassMember( $context->calling_method_id, 'use:' . $stmt->class->parts[0] . ':' . \md5($statements_analyzer->getFilePath()), false ); } $fq_class_name = ClassLikeAnalyzer::getFQCLNFromNameObject( $stmt->class, $aliases ); $fq_class_name = $codebase->classlikes->getUnAliasedName($fq_class_name); } elseif ($context->self !== null) { switch ($stmt->class->parts[0]) { case 'self': $class_storage = $codebase->classlike_storage_provider->get($context->self); $fq_class_name = $class_storage->name; break; case 'parent': $fq_class_name = $context->parent; break; case 'static': // @todo maybe we can do better here $class_storage = $codebase->classlike_storage_provider->get($context->self); $fq_class_name = $class_storage->name; if (!$class_storage->final) { $can_extend = true; $from_static = true; } break; } } if ($codebase->store_node_types && $fq_class_name && !$context->collect_initializations && !$context->collect_mutations ) { $codebase->analyzer->addNodeReference( $statements_analyzer->getFilePath(), $stmt->class, $codebase->classlikes->classExists($fq_class_name) ? $fq_class_name : '*' . ($stmt->class instanceof PhpParser\Node\Name\FullyQualified ? '\\' : $statements_analyzer->getNamespace() . '-') . implode('\\', $stmt->class->parts) ); } } elseif ($stmt->class instanceof PhpParser\Node\Stmt\Class_) { $statements_analyzer->analyze([$stmt->class], $context); $fq_class_name = ClassAnalyzer::getAnonymousClassName($stmt->class, $statements_analyzer->getFilePath()); } else { self::analyzeConstructorExpression( $statements_analyzer, $codebase, $context, $stmt, $stmt->class, $config, $fq_class_name, $can_extend ); } if ($fq_class_name) { if ($codebase->alter_code && $stmt->class instanceof PhpParser\Node\Name && !in_array($stmt->class->parts[0], ['parent', 'static']) ) { $codebase->classlikes->handleClassLikeReferenceInMigration( $codebase, $statements_analyzer, $stmt->class, $fq_class_name, $context->calling_method_id ); } if ($context->check_classes) { if ($context->isPhantomClass($fq_class_name)) { ArgumentsAnalyzer::analyze( $statements_analyzer, $stmt->args, null, null, true, $context ); return true; } if (ClassLikeAnalyzer::checkFullyQualifiedClassLikeName( $statements_analyzer, $fq_class_name, new CodeLocation($statements_analyzer->getSource(), $stmt->class), $context->self, $context->calling_method_id, $statements_analyzer->getSuppressedIssues() ) === false) { ArgumentsAnalyzer::analyze( $statements_analyzer, $stmt->args, null, null, true, $context ); return true; } if ($codebase->interfaceExists($fq_class_name)) { if (IssueBuffer::accepts( new InterfaceInstantiation( 'Interface ' . $fq_class_name . ' cannot be instantiated', new CodeLocation($statements_analyzer->getSource(), $stmt->class) ), $statements_analyzer->getSuppressedIssues() )) { } return true; } } if ($stmt->class instanceof PhpParser\Node\Stmt\Class_) { $extends = $stmt->class->extends ? (string) $stmt->class->extends : null; $result_atomic_type = new Type\Atomic\TAnonymousClassInstance($fq_class_name, false, $extends); } else { //if the class is a Name, it can't represent a child $definite_class = $stmt->class instanceof PhpParser\Node\Name; $result_atomic_type = new TNamedObject($fq_class_name, $from_static, $definite_class); } $statements_analyzer->node_data->setType( $stmt, new Type\Union([$result_atomic_type]) ); if (strtolower($fq_class_name) !== 'stdclass' && $codebase->classlikes->classExists($fq_class_name) ) { self::analyzeNamedConstructor( $statements_analyzer, $codebase, $stmt, $context, $fq_class_name, $from_static, $can_extend ); } else { ArgumentsAnalyzer::analyze( $statements_analyzer, $stmt->args, null, null, true, $context ); } } if (!$config->remember_property_assignments_after_call && !$context->collect_initializations) { $context->removeMutableObjectVars(); } return true; } private static function analyzeNamedConstructor( StatementsAnalyzer $statements_analyzer, \Psalm\Codebase $codebase, PhpParser\Node\Expr\New_ $stmt, Context $context, string $fq_class_name, bool $from_static, bool $can_extend ): void { $storage = $codebase->classlike_storage_provider->get($fq_class_name); if ($from_static) { if (!$storage->preserve_constructor_signature) { if (IssueBuffer::accepts( new UnsafeInstantiation( 'Cannot safely instantiate class ' . $fq_class_name . ' with "new static" as' . ' its constructor might change in child classes', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($storage->template_types && !$storage->enforce_template_inheritance ) { $source = $statements_analyzer->getSource(); if ($source instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer) { $function_storage = $source->getFunctionLikeStorage($statements_analyzer); if ($function_storage->return_type && preg_match('/\bstatic\b/', $function_storage->return_type->getId()) ) { if (IssueBuffer::accepts( new UnsafeGenericInstantiation( 'Cannot safely instantiate generic class ' . $fq_class_name . ' with "new static" as' . ' its generic parameters may be constrained in child classes.', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } } } } // if we're not calling this constructor via new static() if ($storage->abstract && !$can_extend) { if (IssueBuffer::accepts( new AbstractInstantiation( 'Unable to instantiate a abstract class ' . $fq_class_name, new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { return; } } if ($storage->deprecated && strtolower($fq_class_name) !== strtolower((string)$context->self)) { if (IssueBuffer::accepts( new DeprecatedClass( $fq_class_name . ' is marked deprecated', new CodeLocation($statements_analyzer->getSource(), $stmt), $fq_class_name ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } if ($context->self && !$context->collect_initializations && !$context->collect_mutations && !NamespaceAnalyzer::isWithin($context->self, $storage->internal) ) { if (IssueBuffer::accepts( new InternalClass( $fq_class_name . ' is internal to ' . $storage->internal . ' but called from ' . $context->self, new CodeLocation($statements_analyzer->getSource(), $stmt), $fq_class_name ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } $method_id = new \Psalm\Internal\MethodIdentifier($fq_class_name, '__construct'); if ($codebase->methods->methodExists( $method_id, $context->calling_method_id, $codebase->collect_locations ? new CodeLocation($statements_analyzer->getSource(), $stmt) : null, $statements_analyzer, $statements_analyzer->getFilePath() )) { if ($codebase->store_node_types && !$context->collect_initializations && !$context->collect_mutations ) { ArgumentMapPopulator::recordArgumentPositions( $statements_analyzer, $stmt, $codebase, (string)$method_id ); } $template_result = new \Psalm\Internal\Type\TemplateResult([], []); if (self::checkMethodArgs( $method_id, $stmt->args, $template_result, $context, new CodeLocation($statements_analyzer->getSource(), $stmt), $statements_analyzer ) === false) { return; } if (Method\MethodVisibilityAnalyzer::analyze( $method_id, $context, $statements_analyzer->getSource(), new CodeLocation($statements_analyzer->getSource(), $stmt), $statements_analyzer->getSuppressedIssues() ) === false) { return; } $declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id); if ($declaring_method_id) { $method_storage = $codebase->methods->getStorage($declaring_method_id); $namespace = $statements_analyzer->getNamespace() ?: ''; if (!NamespaceAnalyzer::isWithin( $namespace, $method_storage->internal )) { if (IssueBuffer::accepts( new InternalMethod( 'Constructor ' . $codebase->methods->getCasedMethodId($declaring_method_id) . ' is internal to ' . $method_storage->internal . ' but called from ' . ($namespace ?: 'root namespace'), new CodeLocation($statements_analyzer, $stmt), (string) $method_id ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } if (!$method_storage->external_mutation_free && !$context->inside_throw) { if ($context->pure) { if (IssueBuffer::accepts( new ImpureMethodCall( 'Cannot call an impure constructor from a pure context', new CodeLocation($statements_analyzer, $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($statements_analyzer->getSource() instanceof \Psalm\Internal\Analyzer\FunctionLikeAnalyzer && $statements_analyzer->getSource()->track_mutations ) { $statements_analyzer->getSource()->inferred_has_mutation = true; $statements_analyzer->getSource()->inferred_impure = true; } } $generic_params = $template_result->lower_bounds; if ($method_storage->assertions && $stmt->class instanceof PhpParser\Node\Name) { self::applyAssertionsToContext( $stmt->class, null, $method_storage->assertions, $stmt->args, $generic_params, $context, $statements_analyzer ); } if ($method_storage->if_true_assertions) { $statements_analyzer->node_data->setIfTrueAssertions( $stmt, \array_map( function ($assertion) use ($generic_params, $codebase) { return $assertion->getUntemplatedCopy($generic_params, null, $codebase); }, $method_storage->if_true_assertions ) ); } if ($method_storage->if_false_assertions) { $statements_analyzer->node_data->setIfFalseAssertions( $stmt, \array_map( function ($assertion) use ($generic_params, $codebase) { return $assertion->getUntemplatedCopy($generic_params, null, $codebase); }, $method_storage->if_false_assertions ) ); } } $generic_param_types = null; if ($storage->template_types) { foreach ($storage->template_types as $template_name => $base_type) { if (isset($template_result->lower_bounds[$template_name][$fq_class_name])) { $generic_param_type = TemplateStandinTypeReplacer::getMostSpecificTypeFromBounds( $template_result->lower_bounds[$template_name][$fq_class_name], $codebase ); } elseif ($storage->template_extended_params && $template_result->lower_bounds) { $generic_param_type = self::getGenericParamForOffset( $fq_class_name, $template_name, $storage->template_extended_params, array_map( function ($type_map) use ($codebase) { return array_map( function ($bounds) use ($codebase) { return TemplateStandinTypeReplacer::getMostSpecificTypeFromBounds( $bounds, $codebase ); }, $type_map ); }, $template_result->lower_bounds ) ); } else { if ($fq_class_name === 'SplObjectStorage') { $generic_param_type = Type::getEmpty(); } else { $generic_param_type = clone array_values($base_type)[0]; } } $generic_param_type->had_template = true; $generic_param_types[] = $generic_param_type; } } if ($generic_param_types) { $result_atomic_type = new Type\Atomic\TGenericObject( $fq_class_name, $generic_param_types ); $result_atomic_type->was_static = $from_static; $statements_analyzer->node_data->setType( $stmt, new Type\Union([$result_atomic_type]) ); } } elseif ($stmt->args) { if (IssueBuffer::accepts( new TooManyArguments( 'Class ' . $fq_class_name . ' has no __construct, but arguments were passed', new CodeLocation($statements_analyzer->getSource(), $stmt), $fq_class_name . '::__construct' ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($storage->template_types) { $result_atomic_type = new Type\Atomic\TGenericObject( $fq_class_name, array_values( array_map( function ($map) { return clone reset($map); }, $storage->template_types ) ) ); $result_atomic_type->was_static = $from_static; $statements_analyzer->node_data->setType( $stmt, new Type\Union([$result_atomic_type]) ); } if ($storage->external_mutation_free) { /** @psalm-suppress UndefinedPropertyAssignment */ $stmt->external_mutation_free = true; $stmt_type = $statements_analyzer->node_data->getType($stmt); if ($stmt_type) { $stmt_type->reference_free = true; } } if ($statements_analyzer->data_flow_graph instanceof TaintFlowGraph && !\in_array('TaintedInput', $statements_analyzer->getSuppressedIssues()) && ($stmt_type = $statements_analyzer->node_data->getType($stmt)) ) { $code_location = new CodeLocation($statements_analyzer->getSource(), $stmt); $method_storage = null; $declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id); if ($declaring_method_id) { $method_storage = $codebase->methods->getStorage($declaring_method_id); } if ($storage->external_mutation_free || ($method_storage && $method_storage->specialize_call) ) { $method_source = DataFlowNode::getForMethodReturn( (string)$method_id, $fq_class_name . '::__construct', $storage->location, $code_location ); } else { $method_source = DataFlowNode::getForMethodReturn( (string)$method_id, $fq_class_name . '::__construct', $storage->location ); } $statements_analyzer->data_flow_graph->addNode($method_source); $stmt_type->parent_nodes = [$method_source->id => $method_source]; } } private static function analyzeConstructorExpression( StatementsAnalyzer $statements_analyzer, \Psalm\Codebase $codebase, Context $context, PhpParser\Node\Expr\New_ $stmt, PhpParser\Node\Expr $stmt_class, \Psalm\Config $config, ?string &$fq_class_name, bool &$can_extend ): void { $was_inside_general_use = $context->inside_general_use; $context->inside_general_use = true; ExpressionAnalyzer::analyze($statements_analyzer, $stmt_class, $context); $context->inside_general_use = $was_inside_general_use; $stmt_class_type = $statements_analyzer->node_data->getType($stmt_class); if (!$stmt_class_type) { ArgumentsAnalyzer::analyze( $statements_analyzer, $stmt->args, null, null, true, $context ); return; } $has_single_class = $stmt_class_type->isSingleStringLiteral(); if ($has_single_class) { $fq_class_name = $stmt_class_type->getSingleStringLiteral()->value; } else { if (self::checkMethodArgs( null, $stmt->args, null, $context, new CodeLocation($statements_analyzer->getSource(), $stmt), $statements_analyzer ) === false) { return; } } $new_type = null; $stmt_class_types = $stmt_class_type->getAtomicTypes(); while ($stmt_class_types) { $lhs_type_part = \array_shift($stmt_class_types); if ($lhs_type_part instanceof Type\Atomic\TTemplateParam) { $stmt_class_types = \array_merge($stmt_class_types, $lhs_type_part->as->getAtomicTypes()); continue; } if ($lhs_type_part instanceof Type\Atomic\TTemplateParamClass) { if (!$statements_analyzer->node_data->getType($stmt)) { $new_type_part = new Type\Atomic\TTemplateParam( $lhs_type_part->param_name, $lhs_type_part->as_type ? new Type\Union([$lhs_type_part->as_type]) : Type::getObject(), $lhs_type_part->defining_class ); if (!$lhs_type_part->as_type) { if (IssueBuffer::accepts( new MixedMethodCall( 'Cannot call constructor on an unknown class', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } $new_type = Type::combineUnionTypes($new_type, new Type\Union([$new_type_part])); if ($lhs_type_part->as_type && $codebase->classlikes->classExists($lhs_type_part->as_type->value) ) { $as_storage = $codebase->classlike_storage_provider->get( $lhs_type_part->as_type->value ); if (!$as_storage->preserve_constructor_signature) { if (IssueBuffer::accepts( new UnsafeInstantiation( 'Cannot safely instantiate class ' . $lhs_type_part->as_type->value . ' with "new $class_name" as' . ' its constructor might change in child classes', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } } } if ($lhs_type_part->as_type) { $codebase->methods->methodExists( new \Psalm\Internal\MethodIdentifier( $lhs_type_part->as_type->value, '__construct' ), $context->calling_method_id, $codebase->collect_locations ? new CodeLocation($statements_analyzer->getSource(), $stmt) : null, $statements_analyzer, $statements_analyzer->getFilePath() ); } continue; } if ($lhs_type_part instanceof Type\Atomic\TLiteralClassString || $lhs_type_part instanceof Type\Atomic\TClassString || $lhs_type_part instanceof Type\Atomic\TDependentGetClass ) { if (!$statements_analyzer->node_data->getType($stmt)) { if ($lhs_type_part instanceof Type\Atomic\TClassString) { $generated_type = $lhs_type_part->as_type ? clone $lhs_type_part->as_type : new Type\Atomic\TObject(); if ($lhs_type_part->as_type && $codebase->classlikes->classExists($lhs_type_part->as_type->value) ) { $as_storage = $codebase->classlike_storage_provider->get( $lhs_type_part->as_type->value ); if (!$as_storage->preserve_constructor_signature) { if (IssueBuffer::accepts( new UnsafeInstantiation( 'Cannot safely instantiate class ' . $lhs_type_part->as_type->value . ' with "new $class_name" as' . ' its constructor might change in child classes', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } } } elseif ($lhs_type_part instanceof Type\Atomic\TDependentGetClass) { $generated_type = new Type\Atomic\TObject(); if ($lhs_type_part->as_type->hasObjectType() && $lhs_type_part->as_type->isSingle() ) { foreach ($lhs_type_part->as_type->getAtomicTypes() as $typeof_type_atomic) { if ($typeof_type_atomic instanceof Type\Atomic\TNamedObject) { $generated_type = new Type\Atomic\TNamedObject( $typeof_type_atomic->value ); } } } } else { $generated_type = new Type\Atomic\TNamedObject( $lhs_type_part->value ); } if ($lhs_type_part instanceof Type\Atomic\TClassString) { $can_extend = true; } if ($generated_type instanceof Type\Atomic\TObject) { if (IssueBuffer::accepts( new MixedMethodCall( 'Cannot call constructor on an unknown class', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } $new_type = Type::combineUnionTypes($new_type, new Type\Union([$generated_type])); } continue; } if ($lhs_type_part instanceof Type\Atomic\TString) { if ($config->allow_string_standin_for_class && !$lhs_type_part instanceof Type\Atomic\TNumericString ) { // do nothing } elseif (IssueBuffer::accepts( new InvalidStringClass( 'String cannot be used as a class', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($lhs_type_part instanceof Type\Atomic\TMixed) { if (IssueBuffer::accepts( new MixedMethodCall( 'Cannot call constructor on an unknown class', new CodeLocation($statements_analyzer->getSource(), $stmt) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($lhs_type_part instanceof Type\Atomic\TFalse && $stmt_class_type->ignore_falsable_issues ) { // do nothing } elseif ($lhs_type_part instanceof Type\Atomic\TNull && $stmt_class_type->ignore_nullable_issues ) { // do nothing } elseif (IssueBuffer::accepts( new UndefinedClass( 'Type ' . $lhs_type_part . ' cannot be called as a class', new CodeLocation($statements_analyzer->getSource(), $stmt), (string)$lhs_type_part ), $statements_analyzer->getSuppressedIssues() )) { // fall through } $new_type = Type::combineUnionTypes($new_type, Type::getObject()); } if (!$has_single_class) { if ($new_type) { $statements_analyzer->node_data->setType($stmt, $new_type); } ArgumentsAnalyzer::analyze( $statements_analyzer, $stmt->args, null, null, true, $context ); return; } } }