From 6effd1bd2d2c1492189cd839f6d157f484180518 Mon Sep 17 00:00:00 2001 From: Bruce Weirdan Date: Thu, 10 Nov 2022 01:59:34 -0400 Subject: [PATCH] Flag invalid enum case value types Fixes vimeo/psalm#8267 --- .../issues/InvalidEnumCaseValue.md | 25 ++++++++++++ .../Reflector/ClassLikeNodeScanner.php | 40 ++++++++++++------- tests/EnumTest.php | 10 +++++ 3 files changed, 60 insertions(+), 15 deletions(-) diff --git a/docs/running_psalm/issues/InvalidEnumCaseValue.md b/docs/running_psalm/issues/InvalidEnumCaseValue.md index 86c7f621c..414f78942 100644 --- a/docs/running_psalm/issues/InvalidEnumCaseValue.md +++ b/docs/running_psalm/issues/InvalidEnumCaseValue.md @@ -75,3 +75,28 @@ enum Status: string case Open = "open"; } ``` + +## Case with a type that back an enum + +Case type should be either `int` or `string`. + +```php +scalarType) { if ($node->scalarType->name === 'string' || $node->scalarType->name === 'int') { $storage->enum_type = $node->scalarType->name; + $storage->class_implements['backedenum'] = 'BackedEnum'; + $storage->direct_class_interfaces['backedenum'] = 'BackedEnum'; + $this->file_storage->required_interfaces['backedenum'] = 'BackedEnum'; + $this->codebase->scanner->queueClassLikeForScanning('BackedEnum'); + $storage->declaring_method_ids['from'] = new MethodIdentifier('BackedEnum', 'from'); + $storage->appearing_method_ids['from'] = $storage->declaring_method_ids['from']; + $storage->declaring_method_ids['tryfrom'] = new MethodIdentifier( + 'BackedEnum', + 'tryfrom' + ); + $storage->appearing_method_ids['tryfrom'] = $storage->declaring_method_ids['tryfrom']; } else { IssueBuffer::maybeAdd( new InvalidEnumBackingType( @@ -325,17 +337,6 @@ class ClassLikeNodeScanner $this->file_storage->has_visitor_issues = true; $storage->has_visitor_issues = true; } - $storage->class_implements['backedenum'] = 'BackedEnum'; - $storage->direct_class_interfaces['backedenum'] = 'BackedEnum'; - $this->file_storage->required_interfaces['backedenum'] = 'BackedEnum'; - $this->codebase->scanner->queueClassLikeForScanning('BackedEnum'); - $storage->declaring_method_ids['from'] = new MethodIdentifier('BackedEnum', 'from'); - $storage->appearing_method_ids['from'] = $storage->declaring_method_ids['from']; - $storage->declaring_method_ids['tryfrom'] = new MethodIdentifier( - 'BackedEnum', - 'tryfrom' - ); - $storage->appearing_method_ids['tryfrom'] = $storage->declaring_method_ids['tryfrom']; } $this->codebase->scanner->queueClassLikeForScanning('UnitEnum'); @@ -710,7 +711,11 @@ class ClassLikeNodeScanner } elseif ($node_stmt instanceof PhpParser\Node\Stmt\EnumCase && $node instanceof PhpParser\Node\Stmt\Enum_ ) { - $this->visitEnumDeclaration($node_stmt, $storage, $fq_classlike_name); + $this->visitEnumDeclaration( + $node_stmt, + $storage, + $fq_classlike_name + ); } } @@ -1364,6 +1369,8 @@ class ClassLikeNodeScanner $enum_value = null; + $case_location = new CodeLocation($this->file_scanner, $stmt); + if ($stmt->expr !== null) { $case_type = SimpleTypeInferer::infer( $this->codebase, @@ -1381,8 +1388,12 @@ class ClassLikeNodeScanner } elseif ($case_type->isSingleStringLiteral()) { $enum_value = $case_type->getSingleStringLiteral()->value; } else { - throw new RuntimeException( - 'Unexpected: case value for ' . $stmt->name->name . ' is ' . $case_type->getId() + IssueBuffer::maybeAdd( + new InvalidEnumCaseValue( + 'Case of a backed enum should have either string or int value', + $case_location, + $fq_classlike_name + ) ); } } else { @@ -1390,7 +1401,6 @@ class ClassLikeNodeScanner } } - $case_location = new CodeLocation($this->file_scanner, $stmt); if (!isset($storage->enum_cases[$stmt->name->name])) { $case = new EnumCaseStorage( diff --git a/tests/EnumTest.php b/tests/EnumTest.php index b65d840ff..19e0e401b 100644 --- a/tests/EnumTest.php +++ b/tests/EnumTest.php @@ -559,6 +559,16 @@ class EnumTest extends TestCase 'ignored_issues' => [], 'php_version' => '8.1', ], + 'invalidCaseTypeForBackedEnum' => [ + 'code' => ' 'InvalidEnumCaseValue', + 'ignored_issues' => [], + 'php_version' => '8.1', + ], 'duplicateValues' => [ 'code' => '