rector/packages/FamilyTree/Reflection/FamilyRelationsAnalyzer.php
Tomas Votruba 1f63ed3e55
Reworking trait scope run (#952)
* try reworking trait scope run

* load configs from extension installer

* add trait type

* load configs from extension installer in PHPStan internally in Rector too

* [ci-review] Rector Rectify

Co-authored-by: GitHub Action <action@github.com>
2021-10-04 23:11:10 +02:00

220 lines
7.1 KiB
PHP

<?php
declare(strict_types=1);
namespace Rector\FamilyTree\Reflection;
use PhpParser\Node;
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Expr\PropertyFetch;
use PhpParser\Node\Expr\StaticPropertyFetch;
use PhpParser\Node\Name;
use PhpParser\Node\NullableType;
use PhpParser\Node\Stmt;
use PhpParser\Node\Stmt\Class_;
use PhpParser\Node\Stmt\ClassMethod;
use PhpParser\Node\Stmt\Interface_;
use PhpParser\Node\Stmt\Property;
use PhpParser\Node\UnionType as PhpParserUnionType;
use PhpParser\Parser;
use PHPStan\Analyser\Scope;
use PHPStan\Reflection\ClassReflection;
use PHPStan\Reflection\ReflectionProvider;
use PHPStan\Type\NullType;
use PHPStan\Type\Type;
use PHPStan\Type\UnionType;
use Rector\Core\PhpParser\AstResolver;
use Rector\Core\PhpParser\Node\BetterNodeFinder;
use Rector\FamilyTree\ValueObject\PropertyType;
use Rector\NodeNameResolver\NodeNameResolver;
use Rector\NodeTypeResolver\Node\AttributeKey;
use Rector\PHPStanStaticTypeMapper\ValueObject\TypeKind;
use Rector\StaticTypeMapper\StaticTypeMapper;
use Symplify\PackageBuilder\Reflection\PrivatesAccessor;
use Symplify\SmartFileSystem\SmartFileSystem;
final class FamilyRelationsAnalyzer
{
public function __construct(
private ReflectionProvider $reflectionProvider,
private PrivatesAccessor $privatesAccessor,
private NodeNameResolver $nodeNameResolver,
private SmartFileSystem $smartFileSystem,
private BetterNodeFinder $betterNodeFinder,
private StaticTypeMapper $staticTypeMapper,
private AstResolver $astResolver,
private Parser $parser
) {
}
/**
* @return ClassReflection[]
*/
public function getChildrenOfClassReflection(ClassReflection $desiredClassReflection): array
{
/** @var ClassReflection[] $classReflections */
$classReflections = $this->privatesAccessor->getPrivateProperty($this->reflectionProvider, 'classes');
$childrenClassReflections = [];
foreach ($classReflections as $classReflection) {
if (! $classReflection->isSubclassOf($desiredClassReflection->getName())) {
continue;
}
$childrenClassReflections[] = $classReflection;
}
return $childrenClassReflections;
}
public function getPossibleUnionPropertyType(
Property $property,
Type $varType,
?Scope $scope,
Name | NullableType | PhpParserUnionType | null $propertyTypeNode
): PropertyType {
if ($varType instanceof UnionType) {
return new PropertyType($varType, $propertyTypeNode);
}
if (! $scope instanceof Scope) {
return new PropertyType($varType, $propertyTypeNode);
}
/** @var ClassReflection $classReflection */
$classReflection = $scope->getClassReflection();
$ancestors = $classReflection->getAncestors();
$propertyName = $this->nodeNameResolver->getName($property);
$kindPropertyFetch = $this->getKindPropertyFetch($property);
$className = $property->getAttribute(AttributeKey::CLASS_NAME);
foreach ($ancestors as $ancestor) {
$ancestorName = $ancestor->getName();
if ($ancestorName === $className) {
continue;
}
$fileName = $ancestor->getFileName();
if ($fileName === false) {
continue;
}
$fileContent = $this->smartFileSystem->readFile($fileName);
$nodes = $this->parser->parse($fileContent);
if ($ancestor->isSubclassOf('PHPUnit\Framework\TestCase')) {
continue;
}
if ($nodes === null) {
continue;
}
if (! $this->isPropertyWritten($nodes, $propertyName, $kindPropertyFetch)) {
continue;
}
$varType = new UnionType([$varType, new NullType()]);
$propertyTypeNode = $this->staticTypeMapper->mapPHPStanTypeToPhpParserNode(
$varType,
TypeKind::PROPERTY()
);
return new PropertyType($varType, $propertyTypeNode);
}
return new PropertyType($varType, $propertyTypeNode);
}
/**
* @return string[]
*/
public function getClassLikeAncestorNames(Class_ | Interface_ | Name $classOrName): array
{
$ancestorNames = [];
if ($classOrName instanceof Name) {
$fullName = $this->nodeNameResolver->getName($classOrName);
$classLike = $this->astResolver->resolveClassFromName($fullName);
} else {
$classLike = $classOrName;
}
if ($classLike instanceof Interface_) {
foreach ($classLike->extends as $extendInterfaceName) {
$ancestorNames[] = $this->nodeNameResolver->getName($extendInterfaceName);
$ancestorNames = array_merge($ancestorNames, $this->getClassLikeAncestorNames($extendInterfaceName));
}
}
if ($classLike instanceof Class_) {
if ($classLike->extends instanceof Name) {
$extendName = $classLike->extends;
$ancestorNames[] = $this->nodeNameResolver->getName($extendName);
$ancestorNames = array_merge($ancestorNames, $this->getClassLikeAncestorNames($extendName));
}
foreach ($classLike->implements as $implement) {
$ancestorNames[] = $this->nodeNameResolver->getName($implement);
$ancestorNames = array_merge($ancestorNames, $this->getClassLikeAncestorNames($implement));
}
}
/** @var string[] $ancestorNames */
return $ancestorNames;
}
private function getKindPropertyFetch(Property $property): string
{
return $property->isStatic()
? StaticPropertyFetch::class
: PropertyFetch::class;
}
/**
* @param Stmt[] $nodes
*/
private function isPropertyWritten(array $nodes, string $propertyName, string $kindPropertyFetch): bool
{
return (bool) $this->betterNodeFinder->findFirst($nodes, function (Node $node) use (
$propertyName,
$kindPropertyFetch
): bool {
if (! $node instanceof ClassMethod) {
return false;
}
if ($this->nodeNameResolver->isName($node->name, 'autowire')) {
return false;
}
return $this->isPropertyAssignedInClassMethod($node, $propertyName, $kindPropertyFetch);
});
}
private function isPropertyAssignedInClassMethod(
ClassMethod $classMethod,
string $propertyName,
string $kindPropertyFetch
): bool {
if ($classMethod->stmts === null) {
return false;
}
return (bool) $this->betterNodeFinder->findFirst($classMethod->stmts, function (Node $node) use (
$propertyName,
$kindPropertyFetch
): bool {
if (! $node instanceof Assign) {
return false;
}
return $kindPropertyFetch === $node->var::class && $this->nodeNameResolver->isName(
$node->var,
$propertyName
);
});
}
}