Since the release of PHP 8.0 more and more libraries, frameworks and tools have been updated to use attributes instead of annotations in PHPDocs.
However, static analysis tools like PHPStan have not made this transition to attributes and they still rely on annotations in PHPDocs for a lot of their functionality.
This is a set of RectorPHP rules that allows us to convert standard PHP static analysis annotations into a new set of attributes that replace these annotations. These attributes are defined in this repository
In order to show how code would look with these attributes, we can look at the following example. This is how a class looks like with the current annotations:
<?php
class ArrayAdder
{
/** @var array<string> */
private array $result;
/**
* @param array<string> $array1
* @param array<string> $array2
* @return array<string>
*/
public function addArrays(array $array1, array $array2): array
{
$this->result = $array1 + $array2;
return $this->result;
}
}
And this is how it would look like using the new attributes:
<?php
use PhpStaticAnalysis\Attributes\Type;
use PhpStaticAnalysis\Attributes\Param;
use PhpStaticAnalysis\Attributes\Returns;
class ArrayAdder
{
#[Type('array<string>')]
private array $result;
#[Param(array1: 'array<string>')]
#[Param(array2: 'array<string>')]
#[Returns('array<string>')]
public function addArrays(array $array1, array $array2): array
{
$this->array = $array1 + $array2;
return $this->array;
}
}
First of all, to make the attributes available for your codebase use:
composer require php-static-analysis/attributes
To use these rules, install this package:
composer require --dev php-static-analysis/rector-rule
To replace all the annotations that this package covers, use the set provided by it:
use Rector\Config\RectorConfig;
use PhpStaticAnalysis\RectorRule\Set\PhpStaticAnalysisAnnotationsToAttributesSetList;
return RectorConfig::configure()
->withSets([
PhpStaticAnalysisAnnotationsToAttributesSetList::ANNOTATIONS_TO_ATTRIBUTES
])
->withImportNames();
(We recommend that you add the withImportNames()
option so that attributes are not added with their fully qualified name)
If you only want to replace some annotations and leave the others as they are, use the rule configured with the annotations that you need. For example, if you only want to replace the @return
and @param
annotations, use this configuration:
use Rector\Config\RectorConfig;
use Rector\Php80\ValueObject\AnnotationToAttribute;
use PhpStaticAnalysis\Attributes\Param;
use PhpStaticAnalysis\Attributes\Returns;
use PhpStaticAnalysis\RectorRule\AnnotationsToAttributesRector;
return RectorConfig::configure()
->withConfiguredRule(
AnnotationsToAttributesRector::class,
[
new AnnotationToAttribute('param', Param::class),
new AnnotationToAttribute('return', Returns::class),
]
);
If you want to replace most annotations but exclude a few, you can use the excludeAnnotations
config parameter like this:
use Rector\Config\RectorConfig;
use PhpStaticAnalysis\RectorRule\AnnotationsToAttributesRector;
return RectorConfig::configure()
->withConfiguredRule(
AnnotationsToAttributesRector::class,
[
'excludeAnnotations' => ['throws', 'deprecated'],
]
);
That would convert all annotations except @throws
and @deprecated
These are the available attributes and their corresponding PHPDoc annotations:
Attribute | PHPDoc Annotations |
---|---|
Assert | @assert |
AssertIfFalse | @assert-if-false |
AssertIfTrue | @assert-if-true |
DefineType | @type |
Deprecated | @deprecated |
Immmutable | @immmutable |
ImportType | @import-type |
Impure | @impure |
Internal | @internal |
IsReadOnly | @readonly |
Method | @method |
Mixin | @mixin |
Param | @param |
ParamOut | @param-out |
Property | @property @var |
PropertyRead | @property-read |
PropertyWrite | @property-write |
Pure | @pure |
RequireExtends | @require-extends |
RequireImplements | @require-implements |
Returns | @return |
SelfOut | @self-out @this-out |
Template | @template |
TemplateContravariant | @template-contravariant |
TemplateCovariant | @template-covariant |
TemplateExtends | @extends @template-extends |
TemplateImplements | @implements @template-implements |
TemplateUse | @use @template-use |
Throws | @throws |
Type | @var @return |
By default Param
and ParamOut
attributes are added on the method/function where the @param
or @param-out
annotation was located. It is possible to instead add them on the corresponding parameter in the function. To activate this option, add this code to your configuration:
use PhpStaticAnalysis\RectorRule\AnnotationsToAttributesRector;
use Rector\Config\RectorConfig;
...
return RectorConfig::configure()
...
->withConfiguredRule(
AnnotationsToAttributesRector::class,
[
'addParamAttributeOnParameters' => true,
]
);
By default Assert
, AssertIfFalse
and AssertIfTrue
attributes are added on the method/function where the @assert
, @assert-if-false
or @assert-if-true
annotation was located. It is possible to instead add them on the corresponding parameter in the function. To activate this option, add this code to your configuration:
use PhpStaticAnalysis\RectorRule\AnnotationsToAttributesRector;
use Rector\Config\RectorConfig;
...
return RectorConfig::configure()
...
->withConfiguredRule(
AnnotationsToAttributesRector::class,
[
'addAssertAttributeOnParameters' => true,
]
);
By default Returns
attributes are added to define the return type of methods/functions. It is possible to use the Type
attribute instead. To activate this option, add this code to your configuration:
use PhpStaticAnalysis\RectorRule\AnnotationsToAttributesRector;
use Rector\Config\RectorConfig;
...
return RectorConfig::configure()
...
->withConfiguredRule(
AnnotationsToAttributesRector::class,
[
'useTypeAttributeForReturnAnnotation' => true,
]
);
By default Type
attributes are added to define the type of class properties. It is possible to use the Property
attribute instead. To activate this option, add this code to your configuration:
use PhpStaticAnalysis\RectorRule\AnnotationsToAttributesRector;
use Rector\Config\RectorConfig;
...
return RectorConfig::configure()
...
->withConfiguredRule(
AnnotationsToAttributesRector::class,
[
'usePropertyAttributeForVarAnnotation' => true,
]
);
By default DefineType
attributes are added to define a type for a class. It is possible to use the Type
attribute instead. To activate this option, add this code to your configuration:
use PhpStaticAnalysis\RectorRule\AnnotationsToAttributesRector;
use Rector\Config\RectorConfig;
...
return RectorConfig::configure()
...
->withConfiguredRule(
AnnotationsToAttributesRector::class,
[
'useTypeAttributeForTypeClassAnnotation' => true,
]
);
Once you have converted your static analysis annotations to attributes, you may want to use them with Rector, so that Rector can understand them and use them to apply its rules. To do this, when running rector we first need to temporarily convert these attributes back to annotations, then run your Rector rules and finally convert the annotations back to attributes. To do this, use this in your configuration:
use PhpStaticAnalysis\RectorRule\Set\PhpStaticAnalysisAnnotationsToAttributesSetList;
use PhpStaticAnalysis\RectorRule\Set\PhpStaticAnalysisAttributesToAnnotationsSetList;
use Rector\Config\RectorConfig;
...
return RectorConfig::configure()
->withSets([
PhpStaticAnalysisAttributesToAnnotationsSetList::ATTRIBUTES_TO_ANNOTATIONS
])
...
//any other Rector rules or sets
...
->withSets([
PhpStaticAnalysisAnnotationsToAttributesSetList::ANNOTATIONS_TO_ATTRIBUTES
]);
If you use any special configuration for the Annotations to Attributes process, for example only converting some of the annotations or setting some flags, use it in this last part of the block of code instead of the sample.
If you would like to support the development of this project, please consider sponsoring me