rector/rules/TypeDeclaration/Rector/ClassMethod/AddArrayReturnDocTypeRector.php
Tomas Votruba 28c719d1fb Updated Rector to commit 8961d20f56fa4b39bb21feec1f24422398f13dba
8961d20f56 [DX] Localize multi instance checker and privates accessor from Symplify + bump to PHPStan 1.8.3 (#2883)
2022-09-01 19:30:48 +00:00

303 lines
12 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\TypeDeclaration\Rector\ClassMethod;
use PhpParser\Node;
use PhpParser\Node\Name\FullyQualified;
use PhpParser\Node\Stmt\Class_;
use PhpParser\Node\Stmt\ClassMethod;
use PHPStan\Analyser\Scope;
use PHPStan\PhpDocParser\Ast\PhpDoc\ReturnTagValueNode;
use PHPStan\PhpDocParser\Ast\Type\ArrayShapeNode;
use PHPStan\PhpDocParser\Ast\Type\ArrayTypeNode;
use PHPStan\PhpDocParser\Ast\Type\GenericTypeNode;
use PHPStan\Reflection\ClassReflection;
use PHPStan\Type\ArrayType;
use PHPStan\Type\Generic\GenericObjectType;
use PHPStan\Type\IterableType;
use PHPStan\Type\MixedType;
use PHPStan\Type\Type;
use PHPStan\Type\VoidType;
use PHPUnit\Framework\TestCase;
use Rector\BetterPhpDocParser\PhpDocInfo\PhpDocInfo;
use Rector\BetterPhpDocParser\PhpDocManipulator\PhpDocTypeChanger;
use Rector\Core\Rector\AbstractScopeAwareRector;
use Rector\DeadCode\PhpDoc\TagRemover\ReturnTagRemover;
use Rector\Privatization\TypeManipulator\NormalizeTypeToRespectArrayScalarType;
use Rector\Privatization\TypeManipulator\TypeNormalizer;
use Rector\TypeDeclaration\NodeAnalyzer\PHPUnitDataProviderResolver;
use Rector\TypeDeclaration\NodeTypeAnalyzer\DetailedTypeAnalyzer;
use Rector\TypeDeclaration\TypeAnalyzer\AdvancedArrayAnalyzer;
use Rector\TypeDeclaration\TypeAnalyzer\IterableTypeAnalyzer;
use Rector\TypeDeclaration\TypeInferer\ReturnTypeInferer;
use Rector\TypeDeclaration\TypeInferer\ReturnTypeInferer\ReturnTypeDeclarationReturnTypeInfererTypeInferer;
use Rector\VendorLocker\NodeVendorLocker\ClassMethodReturnTypeOverrideGuard;
use Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample;
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
/**
* @see \Rector\Tests\TypeDeclaration\Rector\ClassMethod\AddArrayReturnDocTypeRector\AddArrayReturnDocTypeRectorTest
*/
final class AddArrayReturnDocTypeRector extends AbstractScopeAwareRector
{
/**
* @readonly
* @var \Rector\TypeDeclaration\TypeInferer\ReturnTypeInferer
*/
private $returnTypeInferer;
/**
* @readonly
* @var \Rector\VendorLocker\NodeVendorLocker\ClassMethodReturnTypeOverrideGuard
*/
private $classMethodReturnTypeOverrideGuard;
/**
* @readonly
* @var \Rector\TypeDeclaration\TypeAnalyzer\AdvancedArrayAnalyzer
*/
private $advancedArrayAnalyzer;
/**
* @readonly
* @var \Rector\BetterPhpDocParser\PhpDocManipulator\PhpDocTypeChanger
*/
private $phpDocTypeChanger;
/**
* @readonly
* @var \Rector\Privatization\TypeManipulator\NormalizeTypeToRespectArrayScalarType
*/
private $normalizeTypeToRespectArrayScalarType;
/**
* @readonly
* @var \Rector\DeadCode\PhpDoc\TagRemover\ReturnTagRemover
*/
private $returnTagRemover;
/**
* @readonly
* @var \Rector\TypeDeclaration\NodeTypeAnalyzer\DetailedTypeAnalyzer
*/
private $detailedTypeAnalyzer;
/**
* @readonly
* @var \Rector\Privatization\TypeManipulator\TypeNormalizer
*/
private $typeNormalizer;
/**
* @readonly
* @var \Rector\TypeDeclaration\TypeAnalyzer\IterableTypeAnalyzer
*/
private $iterableTypeAnalyzer;
/**
* @readonly
* @var \Rector\TypeDeclaration\NodeAnalyzer\PHPUnitDataProviderResolver
*/
private $phpUnitDataProviderResolver;
public function __construct(ReturnTypeInferer $returnTypeInferer, ClassMethodReturnTypeOverrideGuard $classMethodReturnTypeOverrideGuard, AdvancedArrayAnalyzer $advancedArrayAnalyzer, PhpDocTypeChanger $phpDocTypeChanger, NormalizeTypeToRespectArrayScalarType $normalizeTypeToRespectArrayScalarType, ReturnTagRemover $returnTagRemover, DetailedTypeAnalyzer $detailedTypeAnalyzer, TypeNormalizer $typeNormalizer, IterableTypeAnalyzer $iterableTypeAnalyzer, PHPUnitDataProviderResolver $phpUnitDataProviderResolver)
{
$this->returnTypeInferer = $returnTypeInferer;
$this->classMethodReturnTypeOverrideGuard = $classMethodReturnTypeOverrideGuard;
$this->advancedArrayAnalyzer = $advancedArrayAnalyzer;
$this->phpDocTypeChanger = $phpDocTypeChanger;
$this->normalizeTypeToRespectArrayScalarType = $normalizeTypeToRespectArrayScalarType;
$this->returnTagRemover = $returnTagRemover;
$this->detailedTypeAnalyzer = $detailedTypeAnalyzer;
$this->typeNormalizer = $typeNormalizer;
$this->iterableTypeAnalyzer = $iterableTypeAnalyzer;
$this->phpUnitDataProviderResolver = $phpUnitDataProviderResolver;
}
public function getRuleDefinition() : RuleDefinition
{
return new RuleDefinition('Adds @return annotation to array parameters inferred from the rest of the code', [new CodeSample(<<<'CODE_SAMPLE'
class SomeClass
{
/**
* @var int[]
*/
private $values;
public function getValues(): array
{
return $this->values;
}
}
CODE_SAMPLE
, <<<'CODE_SAMPLE'
class SomeClass
{
/**
* @var int[]
*/
private $values;
/**
* @return int[]
*/
public function getValues(): array
{
return $this->values;
}
}
CODE_SAMPLE
)]);
}
/**
* @return array<class-string<Node>>
*/
public function getNodeTypes() : array
{
return [ClassMethod::class];
}
/**
* @param ClassMethod $node
*/
public function refactorWithScope(Node $node, Scope $scope) : ?Node
{
if ($this->isDataProviderClassMethod($node, $scope)) {
return null;
}
$phpDocInfo = $this->phpDocInfoFactory->createFromNodeOrEmpty($node);
if ($this->shouldSkip($node, $phpDocInfo)) {
return null;
}
$inferredReturnType = $this->returnTypeInferer->inferFunctionLikeWithExcludedInferers($node, [ReturnTypeDeclarationReturnTypeInfererTypeInferer::class]);
$inferredReturnType = $this->normalizeTypeToRespectArrayScalarType->normalizeToArray($inferredReturnType, $node->returnType);
// generalize false/true type to bool, as mostly default value but accepts both
$inferredReturnType = $this->typeNormalizer->generalizeConstantBoolTypes($inferredReturnType);
$currentReturnType = $phpDocInfo->getReturnType();
if ($this->classMethodReturnTypeOverrideGuard->shouldSkipClassMethodOldTypeWithNewType($currentReturnType, $inferredReturnType, $node)) {
return null;
}
if ($this->shouldSkipType($inferredReturnType, $currentReturnType, $node, $phpDocInfo)) {
return null;
}
$hasChanged = $this->phpDocTypeChanger->changeReturnType($phpDocInfo, $inferredReturnType);
if (!$hasChanged) {
return null;
}
$hasChanged = $this->returnTagRemover->removeReturnTagIfUseless($phpDocInfo, $node);
if ($hasChanged) {
return $node;
}
return null;
}
private function shouldSkip(ClassMethod $classMethod, PhpDocInfo $phpDocInfo) : bool
{
if ($this->shouldSkipClassMethod($classMethod)) {
return \true;
}
if ($this->hasArrayShapeNode($classMethod)) {
return \true;
}
$currentPhpDocReturnType = $phpDocInfo->getReturnType();
if ($currentPhpDocReturnType instanceof ArrayType && $currentPhpDocReturnType->getItemType() instanceof MixedType) {
return \true;
}
if ($this->hasInheritDoc($classMethod)) {
return \true;
}
return $currentPhpDocReturnType instanceof IterableType;
}
private function shouldSkipType(Type $newType, Type $currentType, ClassMethod $classMethod, PhpDocInfo $phpDocInfo) : bool
{
if (!$this->iterableTypeAnalyzer->isIterableType($newType)) {
return \true;
}
if ($newType instanceof ArrayType && $this->shouldSkipArrayType($newType, $classMethod, $phpDocInfo)) {
return \true;
}
// not an array type
if ($newType instanceof VoidType) {
return \true;
}
if ($this->advancedArrayAnalyzer->isMoreSpecificArrayTypeOverride($newType, $phpDocInfo)) {
return \true;
}
if ($this->isGenericTypeToMixedTypeOverride($newType, $currentType)) {
return \true;
}
return $this->detailedTypeAnalyzer->isTooDetailed($newType);
}
private function shouldSkipClassMethod(ClassMethod $classMethod) : bool
{
if ($this->classMethodReturnTypeOverrideGuard->shouldSkipClassMethod($classMethod)) {
return \true;
}
if ($classMethod->returnType === null) {
return \false;
}
return !$this->isNames($classMethod->returnType, ['array', 'iterable', 'Iterator', 'Generator']);
}
private function hasArrayShapeNode(ClassMethod $classMethod) : bool
{
$phpDocInfo = $this->phpDocInfoFactory->createFromNodeOrEmpty($classMethod);
$returnTagValueNode = $phpDocInfo->getReturnTagValue();
if (!$returnTagValueNode instanceof ReturnTagValueNode) {
return \false;
}
if ($returnTagValueNode->type instanceof GenericTypeNode) {
return \true;
}
if ($returnTagValueNode->type instanceof ArrayShapeNode) {
return \true;
}
if (!$returnTagValueNode->type instanceof ArrayTypeNode) {
return \false;
}
return $returnTagValueNode->type->type instanceof ArrayShapeNode;
}
private function hasInheritDoc(ClassMethod $classMethod) : bool
{
$phpDocInfo = $this->phpDocInfoFactory->createFromNodeOrEmpty($classMethod);
return $phpDocInfo->hasInheritDoc();
}
private function shouldSkipArrayType(ArrayType $arrayType, ClassMethod $classMethod, PhpDocInfo $phpDocInfo) : bool
{
if ($this->advancedArrayAnalyzer->isNewAndCurrentTypeBothCallable($arrayType, $phpDocInfo)) {
return \true;
}
if ($this->advancedArrayAnalyzer->isClassStringArrayByStringArrayOverride($arrayType, $classMethod)) {
return \true;
}
return $this->advancedArrayAnalyzer->isMixedOfSpecificOverride($arrayType, $phpDocInfo);
}
private function isGenericTypeToMixedTypeOverride(Type $newType, Type $currentType) : bool
{
if ($newType instanceof GenericObjectType && $currentType instanceof MixedType) {
$types = $newType->getTypes();
if ($types[0] instanceof MixedType && $types[1] instanceof ArrayType) {
return \true;
}
}
return \false;
}
private function isDataProviderClassMethod(ClassMethod $classMethod, Scope $scope) : bool
{
if ($this->isPublicMethodInTestCaseWithReturnIterator($scope, $classMethod)) {
return \true;
}
// should skip data provider, because complex structures
$class = $this->betterNodeFinder->findParentType($classMethod, Class_::class);
if (!$class instanceof Class_) {
return \false;
}
$dataProviderMethodNames = $this->phpUnitDataProviderResolver->resolveDataProviderMethodNames($class);
$classMethodName = $classMethod->name->toString();
return \in_array($classMethodName, $dataProviderMethodNames, \true);
}
private function isPublicMethodInTestCaseWithReturnIterator(Scope $scope, ClassMethod $classMethod) : bool
{
$classReflection = $scope->getClassReflection();
if (!$classReflection instanceof ClassReflection) {
return \false;
}
if (!$classReflection->isSubclassOf(TestCase::class)) {
return \false;
}
if (!$classMethod->returnType instanceof FullyQualified) {
return \false;
}
if (!$this->isName($classMethod->returnType, 'Iterator')) {
return \false;
}
return $classMethod->isPublic();
}
}