From 6e141a10590832adaafff0e6cc7a491b0a0678cf Mon Sep 17 00:00:00 2001 From: robchett Date: Thu, 31 Aug 2023 18:21:47 +0100 Subject: [PATCH] InheritorViolation was only being triggered on grand-childen classes Fixes #10167 --- src/Psalm/Internal/Analyzer/ClassAnalyzer.php | 17 +++++++++++++++++ .../Internal/Analyzer/ClassLikeAnalyzer.php | 19 ------------------- 2 files changed, 17 insertions(+), 19 deletions(-) diff --git a/src/Psalm/Internal/Analyzer/ClassAnalyzer.php b/src/Psalm/Internal/Analyzer/ClassAnalyzer.php index bdb3573260c..c8dd5a03e63 100644 --- a/src/Psalm/Internal/Analyzer/ClassAnalyzer.php +++ b/src/Psalm/Internal/Analyzer/ClassAnalyzer.php @@ -36,6 +36,7 @@ use Psalm\Issue\ExtensionRequirementViolation; use Psalm\Issue\ImplementationRequirementViolation; use Psalm\Issue\InaccessibleMethod; +use Psalm\Issue\InheritorViolation; use Psalm\Issue\InternalClass; use Psalm\Issue\InvalidEnumCaseValue; use Psalm\Issue\InvalidExtendClass; @@ -271,6 +272,22 @@ public function analyze( ); } + $classUnion = new Union([new TNamedObject($fq_class_name)]); + foreach ($storage->parent_classes + $storage->direct_class_interfaces as $parent_class) { + $parent_storage = $codebase->classlikes->getStorageFor($parent_class); + if ($parent_storage && $parent_storage->inheritors) { + if (!UnionTypeComparator::isContainedBy($codebase, $classUnion, $parent_storage->inheritors)) { + IssueBuffer::maybeAdd( + new InheritorViolation( + 'Class ' . $fq_class_name . ' is not an allowed inheritor of parent class ' . $parent_class, + new CodeLocation($this, $this->class), + ), + $this->getSuppressedIssues(), + ); + } + } + } + if ($storage->template_types) { foreach ($storage->template_types as $param_name => $_) { diff --git a/src/Psalm/Internal/Analyzer/ClassLikeAnalyzer.php b/src/Psalm/Internal/Analyzer/ClassLikeAnalyzer.php index a6c909191b8..2c32967f805 100644 --- a/src/Psalm/Internal/Analyzer/ClassLikeAnalyzer.php +++ b/src/Psalm/Internal/Analyzer/ClassLikeAnalyzer.php @@ -14,7 +14,6 @@ use Psalm\Internal\Type\TemplateResult; use Psalm\Internal\Type\TemplateStandinTypeReplacer; use Psalm\Issue\InaccessibleProperty; -use Psalm\Issue\InheritorViolation; use Psalm\Issue\InvalidClass; use Psalm\Issue\InvalidTemplateParam; use Psalm\Issue\MissingDependency; @@ -29,7 +28,6 @@ use Psalm\StatementsSource; use Psalm\Storage\ClassLikeStorage; use Psalm\Type; -use Psalm\Type\Atomic\TNamedObject; use Psalm\Type\Atomic\TTemplateParam; use Psalm\Type\Union; use UnexpectedValueException; @@ -332,23 +330,6 @@ public static function checkFullyQualifiedClassLikeName( return null; } - - $classUnion = new Union([new TNamedObject($fq_class_name)]); - foreach ($class_storage->parent_classes + $class_storage->direct_class_interfaces as $parent_class) { - $parent_storage = $codebase->classlikes->getStorageFor($parent_class); - if ($parent_storage && $parent_storage->inheritors) { - if (!UnionTypeComparator::isContainedBy($codebase, $classUnion, $parent_storage->inheritors)) { - IssueBuffer::maybeAdd( - new InheritorViolation( - 'Class ' . $fq_class_name . ' is not an allowed inheritor of parent class ' . $parent_class, - $code_location, - ), - $suppressed_issues, - ); - } - } - } - foreach ($class_storage->invalid_dependencies as $dependency_class_name => $_) { // if the implemented/extended class is stubbed, it may not yet have // been hydrated