topotru / psalm-conditional-final
Smart final/abstract class enforcement with attributes-based exclusions for Psalm. Perfect for Doctrine entities.
Package info
github.com/topotru/psalm-conditional-final
Type:psalm-plugin
pkg:composer/topotru/psalm-conditional-final
Requires
- php: ^8.3
- ext-simplexml: *
- vimeo/psalm: ^5.15 || ^6.0
Requires (Dev)
- phpunit/phpunit: ^10.5 || ^11.0
README
Smart final/abstract class enforcement with attributes-based exclusions for Psalm.
Enforce final or abstract on your PHP classes without breaking your Doctrine Entities or other proxy-reliant classes.
This plugin replaces dumb token-based linters (like PHPCS) with smart, attributes-aware architectural control on top of the Psalm static analysis engine.
The Problem
Standard linters (e.g., SlevomatCodingStandard.Classes.RequireAbstractOrFinal) force you to make every class final. However, Doctrine Entities (or MappedSuperclasses) must not be final because Doctrine needs to extend them to generate lazy-loading proxy classes at runtime.
If you accidentally make an Entity final, it usually works fine in dev environment but crashes with a Fatal Error on production. To avoid this, you are forced to litter your codebase with ugly comments:
#[ORM\Entity] // phpcs:ignore SlevomatCodingStandard.Classes.RequireAbstractOrFinal class User {} // Annoying and error-prone!
The Solution
Conditional Final reverses the logic:
- Every class must be
finalorabstractby default. - If a class has a forbidden attribute (like
#[ORM\Entity]), it must not befinal(protects your production). - Completely config-driven. No more inline ignore comments!
- Zero-dependency core (does not require
doctrine/ormto be installed).
Installation
composer require --dev topotru/psalm-conditional-final
Enable the plugin in your psalm.xml:
vendor/bin/psalm-plugin enable topotru/psalm-conditional-final
Configuration
By default, the plugin requires all classes to be final or abstract and has an empty exclusion list.
Integration with Doctrine ORM
To enable the built-in preset for Doctrine (#[Entity] and #[MappedSuperclass]), simply add the <useDoctrinePreset /> tag inside the plugin configuration in your psalm.xml:
<plugins> <pluginClass class="Topotru\Psalm\ConditionalFinal\Plugin"> <useDoctrinePreset /> </pluginClass> </plugins>
Custom Configurations
You can add any custom proxy or framework attributes (like API Platform or custom annotations) to the exclusion list manually inside the forbiddenFinalAttributes section:
<plugins> <pluginClass class="Topotru\Psalm\ConditionalFinal\Plugin"> <useDoctrinePreset /> <forbiddenFinalAttributes> <attribute>App\Attributes\CustomProxy</attribute> <attribute>ApiPlatform\Metadata\ApiResource</attribute> </forbiddenFinalAttributes> </pluginClass> </plugins>
Errors Handled
ClassShouldNotBeFinal— Triggers when an entity/proxy class is accidentally marked asfinal(prevents production crashes).ClassShouldBeFinal— Triggers when a standard class (service, repository, etc.) misses thefinalkeyword.
License
The MIT License (MIT). Please see License File for more information.