mirror of
https://github.com/rectorphp/rector.git
synced 2024-06-04 02:10:51 +00:00
e9e85a1300
a8b01f217b
[automated] Re-Generate Nodes/Rectors Documentation (#619)
223 lines
7.7 KiB
PHP
223 lines
7.7 KiB
PHP
<?php
|
|
|
|
declare (strict_types=1);
|
|
namespace Rector\DowngradePhp72\Rector\ClassMethod;
|
|
|
|
use PhpParser\Node;
|
|
use PhpParser\Node\Param;
|
|
use PhpParser\Node\Stmt\ClassMethod;
|
|
use PHPStan\Reflection\ClassReflection;
|
|
use PHPStan\Reflection\ReflectionProvider;
|
|
use Rector\Core\Contract\Rector\ConfigurableRectorInterface;
|
|
use Rector\Core\Rector\AbstractRector;
|
|
use Rector\DowngradePhp72\PhpDoc\NativeParamToPhpDocDecorator;
|
|
use Rector\NodeTypeResolver\Node\AttributeKey;
|
|
use Rector\TypeDeclaration\NodeAnalyzer\AutowiredClassMethodOrPropertyAnalyzer;
|
|
use Symplify\RuleDocGenerator\ValueObject\CodeSample\ConfiguredCodeSample;
|
|
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
|
|
use RectorPrefix20210808\Webmozart\Assert\Assert;
|
|
/**
|
|
* @changelog https://www.php.net/manual/en/migration72.new-features.php#migration72.new-features.param-type-widening
|
|
* @see https://3v4l.org/fOgSE
|
|
*
|
|
* @see \Rector\Tests\DowngradePhp72\Rector\ClassMethod\DowngradeParameterTypeWideningRector\DowngradeParameterTypeWideningRectorTest
|
|
*/
|
|
final class DowngradeParameterTypeWideningRector extends \Rector\Core\Rector\AbstractRector implements \Rector\Core\Contract\Rector\ConfigurableRectorInterface
|
|
{
|
|
/**
|
|
* @var string
|
|
*/
|
|
public const SAFE_TYPES = 'safe_types';
|
|
/**
|
|
* @var string
|
|
*/
|
|
public const SAFE_TYPES_TO_METHODS = 'safe_types_to_methods';
|
|
/**
|
|
* @var class-string[]
|
|
*/
|
|
private $safeTypes = [];
|
|
/**
|
|
* @var array<class-string, string[]>
|
|
*/
|
|
private $safeTypesToMethods = [];
|
|
/**
|
|
* @var \Rector\DowngradePhp72\PhpDoc\NativeParamToPhpDocDecorator
|
|
*/
|
|
private $nativeParamToPhpDocDecorator;
|
|
/**
|
|
* @var \PHPStan\Reflection\ReflectionProvider
|
|
*/
|
|
private $reflectionProvider;
|
|
/**
|
|
* @var \Rector\TypeDeclaration\NodeAnalyzer\AutowiredClassMethodOrPropertyAnalyzer
|
|
*/
|
|
private $autowiredClassMethodOrPropertyAnalyzer;
|
|
public function __construct(\Rector\DowngradePhp72\PhpDoc\NativeParamToPhpDocDecorator $nativeParamToPhpDocDecorator, \PHPStan\Reflection\ReflectionProvider $reflectionProvider, \Rector\TypeDeclaration\NodeAnalyzer\AutowiredClassMethodOrPropertyAnalyzer $autowiredClassMethodOrPropertyAnalyzer)
|
|
{
|
|
$this->nativeParamToPhpDocDecorator = $nativeParamToPhpDocDecorator;
|
|
$this->reflectionProvider = $reflectionProvider;
|
|
$this->autowiredClassMethodOrPropertyAnalyzer = $autowiredClassMethodOrPropertyAnalyzer;
|
|
}
|
|
public function getRuleDefinition() : \Symplify\RuleDocGenerator\ValueObject\RuleDefinition
|
|
{
|
|
return new \Symplify\RuleDocGenerator\ValueObject\RuleDefinition('Change param type to match the lowest type in whole family tree', [new \Symplify\RuleDocGenerator\ValueObject\CodeSample\ConfiguredCodeSample(<<<'CODE_SAMPLE'
|
|
interface SomeInterface
|
|
{
|
|
public function test(array $input);
|
|
}
|
|
|
|
final class SomeClass implements SomeInterface
|
|
{
|
|
public function test($input)
|
|
{
|
|
}
|
|
}
|
|
CODE_SAMPLE
|
|
, <<<'CODE_SAMPLE'
|
|
interface SomeInterface
|
|
{
|
|
/**
|
|
* @param mixed[] $input
|
|
*/
|
|
public function test($input);
|
|
}
|
|
|
|
final class SomeClass implements SomeInterface
|
|
{
|
|
public function test($input)
|
|
{
|
|
}
|
|
}
|
|
CODE_SAMPLE
|
|
, [self::SAFE_TYPES => [], self::SAFE_TYPES_TO_METHODS => []])]);
|
|
}
|
|
/**
|
|
* @return array<class-string<Node>>
|
|
*/
|
|
public function getNodeTypes() : array
|
|
{
|
|
return [\PhpParser\Node\Stmt\ClassMethod::class];
|
|
}
|
|
/**
|
|
* @param ClassMethod $node
|
|
*/
|
|
public function refactor(\PhpParser\Node $node) : ?\PhpParser\Node
|
|
{
|
|
$classLike = $node->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::CLASS_NODE);
|
|
if ($classLike === null) {
|
|
return null;
|
|
}
|
|
$className = $this->nodeNameResolver->getName($classLike);
|
|
if ($className === null) {
|
|
return null;
|
|
}
|
|
if (!$this->reflectionProvider->hasClass($className)) {
|
|
return null;
|
|
}
|
|
$classReflection = $this->reflectionProvider->getClass($className);
|
|
if ($this->isSealedClass($classReflection)) {
|
|
return null;
|
|
}
|
|
if ($this->isSafeType($classReflection, $node)) {
|
|
return null;
|
|
}
|
|
if ($node->isPrivate()) {
|
|
return null;
|
|
}
|
|
if ($this->shouldSkipClassMethod($node)) {
|
|
return null;
|
|
}
|
|
// Downgrade every scalar parameter, just to be sure
|
|
foreach (\array_keys($node->params) as $paramPosition) {
|
|
$this->removeParamTypeFromMethod($node, $paramPosition);
|
|
}
|
|
return $node;
|
|
}
|
|
/**
|
|
* @param array<string, mixed> $configuration
|
|
*/
|
|
public function configure(array $configuration) : void
|
|
{
|
|
$safeTypes = $configuration[self::SAFE_TYPES] ?? [];
|
|
\RectorPrefix20210808\Webmozart\Assert\Assert::allString($safeTypes);
|
|
$this->safeTypes = $safeTypes;
|
|
$safeTypesToMethods = $configuration[self::SAFE_TYPES_TO_METHODS] ?? [];
|
|
\RectorPrefix20210808\Webmozart\Assert\Assert::isArray($safeTypesToMethods);
|
|
foreach ($safeTypesToMethods as $key => $value) {
|
|
\RectorPrefix20210808\Webmozart\Assert\Assert::string($key);
|
|
\RectorPrefix20210808\Webmozart\Assert\Assert::allString($value);
|
|
}
|
|
$this->safeTypesToMethods = $safeTypesToMethods;
|
|
}
|
|
private function removeParamTypeFromMethod(\PhpParser\Node\Stmt\ClassMethod $classMethod, int $paramPosition) : void
|
|
{
|
|
$param = $classMethod->params[$paramPosition] ?? null;
|
|
if (!$param instanceof \PhpParser\Node\Param) {
|
|
return;
|
|
}
|
|
// It already has no type => nothing to do - check original param, as it could have been removed by this rule
|
|
if ($param->type === null) {
|
|
return;
|
|
}
|
|
// Add the current type in the PHPDoc
|
|
$this->nativeParamToPhpDocDecorator->decorate($classMethod, $param);
|
|
$param->type = null;
|
|
}
|
|
private function shouldSkipClassMethod(\PhpParser\Node\Stmt\ClassMethod $classMethod) : bool
|
|
{
|
|
if ($classMethod->isMagic()) {
|
|
return \true;
|
|
}
|
|
if ($classMethod->params === []) {
|
|
return \true;
|
|
}
|
|
if ($this->autowiredClassMethodOrPropertyAnalyzer->detect($classMethod)) {
|
|
return \true;
|
|
}
|
|
foreach ($classMethod->params as $param) {
|
|
if ($param->type !== null) {
|
|
return \false;
|
|
}
|
|
}
|
|
return \true;
|
|
}
|
|
/**
|
|
* This method is perfectly sealed, nothing to downgrade here
|
|
*/
|
|
private function isSealedClass(\PHPStan\Reflection\ClassReflection $classReflection) : bool
|
|
{
|
|
if (!$classReflection->isClass()) {
|
|
return \false;
|
|
}
|
|
if (!$classReflection->isFinal()) {
|
|
return \false;
|
|
}
|
|
return \count($classReflection->getAncestors()) === 1;
|
|
}
|
|
private function isSafeType(\PHPStan\Reflection\ClassReflection $classReflection, \PhpParser\Node\Stmt\ClassMethod $classMethod) : bool
|
|
{
|
|
foreach ($this->safeTypes as $safeType) {
|
|
if ($classReflection->isSubclassOf($safeType)) {
|
|
return \true;
|
|
}
|
|
// skip self too
|
|
if ($classReflection->getName() === $safeType) {
|
|
return \true;
|
|
}
|
|
}
|
|
foreach ($this->safeTypesToMethods as $safeType => $safeMethods) {
|
|
if (!$this->isNames($classMethod, $safeMethods)) {
|
|
continue;
|
|
}
|
|
if ($classReflection->isSubclassOf($safeType)) {
|
|
return \true;
|
|
}
|
|
// skip self too
|
|
if ($classReflection->getName() === $safeType) {
|
|
return \true;
|
|
}
|
|
}
|
|
return \false;
|
|
}
|
|
}
|