rector/rules/DowngradePhp72/NodeAnalyzer/OverrideFromAnonymousClassMethodAnalyzer.php
Tomas Votruba 2f48356b1f Updated Rector to commit c732f61317d64898b7b31ab171d7f1e52caca9f9
c732f61317 [DowngradePhp72] Handle DowngradeParameterTypeWideningRector on anonymous class implements interface (#811)
2021-09-02 06:26:12 +00:00

80 lines
3.2 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\DowngradePhp72\NodeAnalyzer;
use PhpParser\Node\Name\FullyQualified;
use PhpParser\Node\Stmt\Class_;
use PhpParser\Node\Stmt\ClassLike;
use PhpParser\Node\Stmt\ClassMethod;
use PHPStan\Reflection\ClassReflection;
use PHPStan\Reflection\Php\PhpMethodReflection;
use PHPStan\Reflection\ReflectionProvider;
use Rector\Core\NodeAnalyzer\ClassAnalyzer;
use Rector\NodeNameResolver\NodeNameResolver;
use Rector\NodeTypeResolver\Node\AttributeKey;
final class OverrideFromAnonymousClassMethodAnalyzer
{
/**
* @var \Rector\Core\NodeAnalyzer\ClassAnalyzer
*/
private $classAnalyzer;
/**
* @var \Rector\NodeNameResolver\NodeNameResolver
*/
private $nodeNameResolver;
/**
* @var \PHPStan\Reflection\ReflectionProvider
*/
private $reflectionProvider;
public function __construct(\Rector\Core\NodeAnalyzer\ClassAnalyzer $classAnalyzer, \Rector\NodeNameResolver\NodeNameResolver $nodeNameResolver, \PHPStan\Reflection\ReflectionProvider $reflectionProvider)
{
$this->classAnalyzer = $classAnalyzer;
$this->nodeNameResolver = $nodeNameResolver;
$this->reflectionProvider = $reflectionProvider;
}
public function matchAncestorClassReflectionOverrideable(\PhpParser\Node\Stmt\ClassLike $classLike, \PhpParser\Node\Stmt\ClassMethod $classMethod) : ?\PHPStan\Reflection\ClassReflection
{
if (!$this->classAnalyzer->isAnonymousClass($classLike)) {
return null;
}
/** @var Class_ $classLike */
$interfaces = $classLike->implements;
foreach ($interfaces as $interface) {
if (!$interface instanceof \PhpParser\Node\Name\FullyQualified) {
continue;
}
$resolve = $this->resolveClassReflectionWithNotPrivateMethod($interface, $classMethod);
if ($resolve instanceof \PHPStan\Reflection\ClassReflection) {
return $resolve;
}
}
/** @var Class_ $classLike */
if (!$classLike->extends instanceof \PhpParser\Node\Name\FullyQualified) {
return null;
}
return $this->resolveClassReflectionWithNotPrivateMethod($classLike->extends, $classMethod);
}
private function resolveClassReflectionWithNotPrivateMethod(\PhpParser\Node\Name\FullyQualified $fullyQualified, \PhpParser\Node\Stmt\ClassMethod $classMethod) : ?\PHPStan\Reflection\ClassReflection
{
$ancestorClassLike = $fullyQualified->toString();
if (!$this->reflectionProvider->hasClass($ancestorClassLike)) {
return null;
}
$classReflection = $this->reflectionProvider->getClass($ancestorClassLike);
$methodName = $this->nodeNameResolver->getName($classMethod);
if (!$classReflection->hasMethod($methodName)) {
return null;
}
$scope = $classMethod->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::SCOPE);
$method = $classReflection->getMethod($methodName, $scope);
if (!$method instanceof \PHPStan\Reflection\Php\PhpMethodReflection) {
return null;
}
if ($method->isPrivate()) {
return null;
}
return $classReflection;
}
}