====== PHP RFC: Attributes on Constants ====== * Version: 0.2 * Date: 2024-11-30 * Author: Daniel Scherzer, daniel.e.scherzer@gmail.com * Status: Pending Implementation * First Published at: http://wiki.php.net/rfc/attributes-on-constants * Implementation: https://github.com/php/php-src/pull/16952 ===== Introduction ===== Attributes were first introduced in [[rfc:attributes_v2]], with the current syntax adopted [[rfc:shorter_attribute_syntax_change]]. This RFC would add support for attributes on * Compile-time non-class constants To the list of current valid targets (see [[rfc:attributes_v2#attribute_syntax]]): * functions (including closures and short closures) * classes (including anonymous classes), interfaces, traits * class constants * class properties * class methods * function/method parameters Additionally, taking advantage of this new feature, the #[\Deprecated] attribute is updated to also target compile-time non-class constants. "Compile-time non-class constants" refers to constants defined via const MY_CONST = 5; (for example) rather than those via define( 'MY_CONST', 5 );. Unless clear from context, in this RFC "constant" refers to a compile-time non-class constant. ===== Proposal ===== Attributes are allowed to target constants, with the introduction of a Attribute::TARGET_CONSTANT. Additionally, Attribute::TARGET_ALL will now include constants. To prevent confusing code, attributes are only allowed to be applied to constants that are declared in their own statement: // This is okay #[\MyAttribute] const Example1 = 1; // This is an error #[\MyAttribute] const Example2 = 2, Example3 = 3; To expose information about these attributes to runtime, ReflectionConstant::getAttributes() is introduced. Additionally, the #[\Deprecated] attribute is updated to allow targeting constants; when applied, the constant is marked with ''CONST_DEPRECATED''. ===== Backward Incompatible Changes ===== Given that Attribute is final, the added constant shouldn't cause any conflicts. Likewise, ReflectionConstant is final, so the added method shouldn't cause any conflicts. Existing attributes that use Attribute::TARGET_ALL may need to be updated if they do not apply to constants. Furthermore, as with all RFCs that make code that previously had a compile-time error become valid, this RFC requires updates to static analyzers and IDEs to understand the semantics of adding attributes to constants and not erroneously report errors. ===== Proposed PHP Version(s) ===== Next version of PHP (PHP 8.5 or PHP 9.0) ===== RFC Impact ===== ==== To SAPIs ==== Just noting for completion: the current implementation passes the attribute information to runtime via a new opcode, ''ZEND_DECLARE_ATTRIBUTED_CONST'', that phpdbg will ignore the same way it ignores ''ZEND_DECLARE_CONST''. ==== To Opcache ==== The JIT currently has logic to emit deprecation messages for constants that are deprecated; to avoid duplicating the new logic to check a constant's attributes for a #[\Deprecated] attribute, that logic is centralized in a new zend_deprecated_constant() function. ==== To Reflection ==== New method ReflectionConstant::getAttributes() is added, with a signature to match the other such methods for retrieving attributes: final class ReflectionConstant implements Reflector { // Everything that currently exists // ... public function getAttributes(?string $name = null, int $flags = 0): array {} } Additionally, code that expects ReflectionConstant::isDeprecated() to always return false for userland constants will need to be updated, since it can now be true. ==== New Constants ==== Attribute::TARGET_CONSTANT indicates that an attribute can be applied to compile-time non-class constants. ===== Unaffected PHP Functionality ===== Existing uses of attributes or constants - previously, trying to add an attribute to a constant would have generated an error, so any existing working current code should continue to work as expected. ===== Future Scope ===== Adding support to non-compile-time non-class constants, via define(). ===== Proposed Voting Choices ===== Voting started on 2024-12-16 and will end on 2025-01-06 00:00:00 UTC. This vote requires a 2/3 majority to adopt. * Yes * No ===== Patches and Tests ===== https://github.com/php/php-src/pull/16952 ===== Implementation ===== After the project is implemented, this section should contain - the version(s) it was merged into - a link to the git commit(s) - a link to the PHP manual entry for the feature - a link to the language specification section (if any) ===== References ===== * [[rfc:attributes_v2]] * [[rfc:shorter_attribute_syntax_change]] * https://github.com/php/php-src/pull/16952 ===== Changelog ===== 0.2: * Improved discussion of backward incompatible changes 0.1: * Initial proposal