diff --git a/src/Psalm/Internal/Analyzer/Statements/Expression/Fetch/AtomicPropertyFetchAnalyzer.php b/src/Psalm/Internal/Analyzer/Statements/Expression/Fetch/AtomicPropertyFetchAnalyzer.php index 0011896ea..afa84a3d7 100644 --- a/src/Psalm/Internal/Analyzer/Statements/Expression/Fetch/AtomicPropertyFetchAnalyzer.php +++ b/src/Psalm/Internal/Analyzer/Statements/Expression/Fetch/AtomicPropertyFetchAnalyzer.php @@ -63,7 +63,7 @@ use Psalm\Type\Union; use function array_keys; use function array_search; -use function array_values; +use function count; use function in_array; use function is_int; use function is_string; @@ -570,15 +570,9 @@ class AtomicPropertyFetchAnalyzer $class_storage->parent_class ); - if ($class_storage->template_types) { + if (count($template_types = $class_storage->getClassTemplateTypes()) !== 0) { if (!$lhs_type_part instanceof TGenericObject) { - $type_params = []; - - foreach ($class_storage->template_types as $type_map) { - $type_params[] = clone array_values($type_map)[0]; - } - - $lhs_type_part = new TGenericObject($lhs_type_part->value, $type_params); + $lhs_type_part = new TGenericObject($lhs_type_part->value, $template_types); } $stmt_type = self::localizePropertyType( @@ -1100,15 +1094,9 @@ class AtomicPropertyFetchAnalyzer ) { $stmt_type = clone $class_storage->pseudo_property_get_types['$' . $prop_name]; - if ($class_storage->template_types) { + if (count($template_types = $class_storage->getClassTemplateTypes()) !== 0) { if (!$lhs_type_part instanceof TGenericObject) { - $type_params = []; - - foreach ($class_storage->template_types as $type_map) { - $type_params[] = clone array_values($type_map)[0]; - } - - $lhs_type_part = new TGenericObject($lhs_type_part->value, $type_params); + $lhs_type_part = new TGenericObject($lhs_type_part->value, $template_types); } $stmt_type = self::localizePropertyType( @@ -1200,15 +1188,9 @@ class AtomicPropertyFetchAnalyzer $declaring_class_storage->parent_class ); - if ($declaring_class_storage->template_types) { + if (count($template_types = $declaring_class_storage->getClassTemplateTypes()) !== 0) { if (!$lhs_type_part instanceof TGenericObject) { - $type_params = []; - - foreach ($declaring_class_storage->template_types as $type_map) { - $type_params[] = clone array_values($type_map)[0]; - } - - $lhs_type_part = new TGenericObject($lhs_type_part->value, $type_params); + $lhs_type_part = new TGenericObject($lhs_type_part->value, $template_types); } $class_property_type = self::localizePropertyType( diff --git a/src/Psalm/Internal/Type/Comparator/GenericTypeComparator.php b/src/Psalm/Internal/Type/Comparator/GenericTypeComparator.php index dd7e648c3..8cd529c66 100644 --- a/src/Psalm/Internal/Type/Comparator/GenericTypeComparator.php +++ b/src/Psalm/Internal/Type/Comparator/GenericTypeComparator.php @@ -4,16 +4,11 @@ namespace Psalm\Internal\Type\Comparator; use Psalm\Codebase; use Psalm\Internal\Type\TemplateStandinTypeReplacer; -use Psalm\Type; use Psalm\Type\Atomic; use Psalm\Type\Atomic\TGenericObject; use Psalm\Type\Atomic\TIterable; use Psalm\Type\Atomic\TNamedObject; -use function array_fill; -use function array_values; -use function count; - /** * @internal */ @@ -44,38 +39,13 @@ class GenericTypeComparator $container_was_iterable = true; } - if (!$input_type_part instanceof TGenericObject && !$input_type_part instanceof TIterable) { - if ($input_type_part instanceof TNamedObject - && $codebase->classExists($input_type_part->value) - ) { - $class_storage = $codebase->classlike_storage_provider->get($input_type_part->value); - - $container_class = $container_type_part->value; - - // attempt to transform it - if (!empty($class_storage->template_extended_params[$container_class])) { - $input_type_part = new TGenericObject( - $input_type_part->value, - array_values($class_storage->template_extended_params[$container_class]) - ); - } + if (!$input_type_part instanceof TNamedObject && !$input_type_part instanceof TIterable) { + if ($atomic_comparison_result) { + $atomic_comparison_result->type_coerced = true; + $atomic_comparison_result->type_coerced_from_mixed = true; } - if (!$input_type_part instanceof TGenericObject) { - if ($input_type_part instanceof TNamedObject) { - $input_type_part = new TGenericObject( - $input_type_part->value, - array_fill(0, count($container_type_part->type_params), Type::getMixed()) - ); - } else { - if ($atomic_comparison_result) { - $atomic_comparison_result->type_coerced = true; - $atomic_comparison_result->type_coerced_from_mixed = true; - } - - return false; - } - } + return false; } $container_type_params_covariant = []; diff --git a/src/Psalm/Internal/Type/TemplateStandinTypeReplacer.php b/src/Psalm/Internal/Type/TemplateStandinTypeReplacer.php index 3791dc03e..7d6dbf6ce 100644 --- a/src/Psalm/Internal/Type/TemplateStandinTypeReplacer.php +++ b/src/Psalm/Internal/Type/TemplateStandinTypeReplacer.php @@ -33,6 +33,7 @@ use Psalm\Type\Atomic\TTemplateParamClass; use Psalm\Type\Union; use Throwable; +use function array_fill; use function array_keys; use function array_merge; use function array_search; @@ -42,6 +43,7 @@ use function count; use function in_array; use function reset; use function strpos; +use function strtolower; use function substr; use function usort; @@ -1124,7 +1126,7 @@ class TemplateStandinTypeReplacer } /** - * @param TGenericObject|TIterable $input_type_part + * @param TGenericObject|TNamedObject|TIterable $input_type_part * @param TGenericObject|TIterable $container_type_part * @return list */ @@ -1134,7 +1136,21 @@ class TemplateStandinTypeReplacer Atomic $container_type_part, ?array &$container_type_params_covariant = null ): array { - $input_type_params = $input_type_part->type_params; + if ($input_type_part instanceof TGenericObject || $input_type_part instanceof TIterable) { + $input_type_params = $input_type_part->type_params; + } else { + $class_storage = $codebase->classlike_storage_provider->get($input_type_part->value); + + $container_class = $container_type_part->value; + + if (strtolower($input_type_part->value) === strtolower($container_type_part->value)) { + $input_type_params = $class_storage->getClassTemplateTypes(); + } elseif (!empty($class_storage->template_extended_params[$container_class])) { + $input_type_params = array_values($class_storage->template_extended_params[$container_class]); + } else { + $input_type_params = array_fill(0, count($class_storage->template_types ?? []), Type::getMixed()); + } + } try { $input_class_storage = $codebase->classlike_storage_provider->get($input_type_part->value); diff --git a/src/Psalm/Storage/ClassLikeStorage.php b/src/Psalm/Storage/ClassLikeStorage.php index 1b33c69ac..01b369689 100644 --- a/src/Psalm/Storage/ClassLikeStorage.php +++ b/src/Psalm/Storage/ClassLikeStorage.php @@ -11,6 +11,8 @@ use Psalm\Type\Atomic\TNamedObject; use Psalm\Type\Atomic\TTemplateParam; use Psalm\Type\Union; +use function array_values; + class ClassLikeStorage implements HasAttributesInterface { use CustomMetadataTrait; @@ -470,4 +472,20 @@ class ClassLikeStorage implements HasAttributesInterface { return $this->attributes; } + + /** + * Get the template constraint types for the class. + * + * @return list + */ + public function getClassTemplateTypes(): array + { + $type_params = []; + + foreach ($this->template_types ?? [] as $type_map) { + $type_params[] = clone array_values($type_map)[0]; + } + + return $type_params; + } } diff --git a/tests/Template/ClassTemplateExtendsTest.php b/tests/Template/ClassTemplateExtendsTest.php index 8fcba0c57..c8c0fa4e5 100644 --- a/tests/Template/ClassTemplateExtendsTest.php +++ b/tests/Template/ClassTemplateExtendsTest.php @@ -5342,7 +5342,7 @@ class ClassTemplateExtendsTest extends TestCase } new Foo(new DoStuffX());', - 'error_message' => 'MixedArgumentTypeCoercion' + 'error_message' => 'TooManyTemplateParams' ], 'concreteDefinesSignatureTypesDifferent' => [ ' [ + 'code' => ' $_fooClass */ + function bar(string $_fooClass): void {} + + bar(Foo::class); + ', + ], + 'classStringWithGenericChildSatisfiesGenericParentWithDifferentConstraint' => [ + 'code' => ' + */ + class Bar extends Foo {} + + /** @param class-string $_fooClass */ + function bar(string $_fooClass): void {} + + bar(Bar::class); + ', + ], ]; }